From 6e4d970dae2e292e4155174f5de13f7e774384a3 Mon Sep 17 00:00:00 2001 From: Dan Lake Date: Mon, 18 Apr 2011 16:48:49 -0700 Subject: [PATCH 01/45] Requeue unacknowledged entity updates rather than resend then "as is". Often, by the time the UDPServer realizes that an entity update packet has not been acknowledged, there is a newer update for the same entity already queued up or there is a higher priority update that should be sent first. This patch eliminates 1:1 packet resends for unacked entity update packets. Insteawd, unacked update packets are decomposed into the original entity updates and those updates are placed back into the priority queues based on their new priority but the original update timestamp. This will generally place them at the head of the line to be put back on the wire as a new outgoing packet but prevents the resend queue from filling up with multiple stale updates for the same entity. This new approach takes advantage of the UDP nature of the Linden protocol in that the intent of a reliable update packet is that if it goes unacknowledge, SOMETHING has to happen to get the update to the client. We are simply making sure that we are resending current object state rather than stale object state. Additionally, this patch includes a generalized callback mechanism so that any caller can specify their own method to call when a packet expires without being acknowledged. We use this mechanism to requeue update packets and otherwise use the UDPServer default method of just putting expired packets in the resend queue. --- OpenSim/Framework/IClientAPI.cs | 59 +++++++++--- OpenSim/Framework/Util.cs | 17 ++++ .../ClientStack/LindenUDP/LLClientView.cs | 96 +++++++++++++++---- .../ClientStack/LindenUDP/LLUDPServer.cs | 70 +++++++------- .../ClientStack/LindenUDP/OutgoingPacket.cs | 6 +- 5 files changed, 179 insertions(+), 69 deletions(-) diff --git a/OpenSim/Framework/IClientAPI.cs b/OpenSim/Framework/IClientAPI.cs index f573c32d07..069987bd71 100644 --- a/OpenSim/Framework/IClientAPI.cs +++ b/OpenSim/Framework/IClientAPI.cs @@ -572,34 +572,69 @@ namespace OpenSim.Framework public class IEntityUpdate { - public ISceneEntity Entity; - public uint Flags; + private ISceneEntity m_entity; + private uint m_flags; + private int m_updateTime; + + public ISceneEntity Entity + { + get { return m_entity; } + } + + public uint Flags + { + get { return m_flags; } + } + + public int UpdateTime + { + get { return m_updateTime; } + } public virtual void Update(IEntityUpdate update) { - this.Flags |= update.Flags; + m_flags |= update.Flags; + + // Use the older of the updates as the updateTime + if (Util.EnvironmentTickCountCompare(UpdateTime, update.UpdateTime) > 0) + m_updateTime = update.UpdateTime; } public IEntityUpdate(ISceneEntity entity, uint flags) { - Entity = entity; - Flags = flags; + m_entity = entity; + m_flags = flags; + m_updateTime = Util.EnvironmentTickCount(); + } + + public IEntityUpdate(ISceneEntity entity, uint flags, Int32 updateTime) + { + m_entity = entity; + m_flags = flags; + m_updateTime = updateTime; } } - public class EntityUpdate : IEntityUpdate { - // public ISceneEntity Entity; - // public PrimUpdateFlags Flags; - public float TimeDilation; + private float m_timeDilation; + + public float TimeDilation + { + get { return m_timeDilation; } + } public EntityUpdate(ISceneEntity entity, PrimUpdateFlags flags, float timedilation) - : base(entity,(uint)flags) + : base(entity, (uint)flags) { - //Entity = entity; // Flags = flags; - TimeDilation = timedilation; + m_timeDilation = timedilation; + } + + public EntityUpdate(ISceneEntity entity, PrimUpdateFlags flags, float timedilation, Int32 updateTime) + : base(entity,(uint)flags,updateTime) + { + m_timeDilation = timedilation; } } diff --git a/OpenSim/Framework/Util.cs b/OpenSim/Framework/Util.cs index 5a5046e547..aaa2724532 100644 --- a/OpenSim/Framework/Util.cs +++ b/OpenSim/Framework/Util.cs @@ -1537,6 +1537,23 @@ namespace OpenSim.Framework return (diff >= 0) ? diff : (diff + EnvironmentTickCountMask + 1); } + // Returns value of Tick Count A - TickCount B accounting for wrapping of TickCount + // Assumes both tcA and tcB came from previous calls to Util.EnvironmentTickCount(). + // A positive return value indicates A occured later than B + public static Int32 EnvironmentTickCountCompare(Int32 tcA, Int32 tcB) + { + // A, B and TC are all between 0 and 0x3fffffff + int tc = EnvironmentTickCount(); + + if (tc - tcA >= 0) + tcA += EnvironmentTickCountMask + 1; + + if (tc - tcB >= 0) + tcB += EnvironmentTickCountMask + 1; + + return tcA - tcB; + } + /// /// Prints the call stack at any given point. Useful for debugging. /// diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 1f7e66dabb..14c5d6c679 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3561,6 +3561,34 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_entityUpdates.Enqueue(priority, new EntityUpdate(entity, updateFlags, m_scene.TimeDilation)); } + /// + /// Requeue an EntityUpdate when it was not acknowledged by the client. + /// We will update the priority and put it in the correct queue, merging update flags + /// with any other updates that may be queued for the same entity. + /// The original update time is used for the merged update. + /// + public void ResendPrimUpdate(EntityUpdate update) + { + // If the update exists in priority queue, it will be updated. + // If it does not exist then it will be added with the current (rather than its original) priority + uint priority = m_prioritizer.GetUpdatePriority(this, update.Entity); + + lock (m_entityUpdates.SyncRoot) + m_entityUpdates.Enqueue(priority, update); + } + + /// + /// Requeue a list of EntityUpdates when they were not acknowledged by the client. + /// We will update the priority and put it in the correct queue, merging update flags + /// with any other updates that may be queued for the same entity. + /// The original update time is used for the merged update. + /// + void ResendPrimUpdates(List updates) + { + foreach (EntityUpdate update in updates) + ResendPrimUpdate(update); + } + private void ProcessEntityUpdates(int maxUpdates) { OpenSim.Framework.Lazy> objectUpdateBlocks = new OpenSim.Framework.Lazy>(); @@ -3568,6 +3596,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP OpenSim.Framework.Lazy> terseUpdateBlocks = new OpenSim.Framework.Lazy>(); OpenSim.Framework.Lazy> terseAgentUpdateBlocks = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> objectUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> compressedUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> terseUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> terseAgentUpdates = new OpenSim.Framework.Lazy>(); + // Check to see if this is a flush if (maxUpdates <= 0) { @@ -3583,7 +3616,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP float avgTimeDilation = 1.0f; IEntityUpdate iupdate; Int32 timeinqueue; // this is just debugging code & can be dropped later - + while (updatesThisCall < maxUpdates) { lock (m_entityUpdates.SyncRoot) @@ -3688,24 +3721,33 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (update.Entity is ScenePresence) { objectUpdateBlocks.Value.Add(CreateAvatarUpdateBlock((ScenePresence)update.Entity)); + objectUpdates.Value.Add(update); } else { objectUpdateBlocks.Value.Add(CreatePrimUpdateBlock((SceneObjectPart)update.Entity, this.m_agentId)); + objectUpdates.Value.Add(update); } } else if (!canUseImproved) { compressedUpdateBlocks.Value.Add(CreateCompressedUpdateBlock((SceneObjectPart)update.Entity, updateFlags)); + compressedUpdates.Value.Add(update); } else { if (update.Entity is ScenePresence && ((ScenePresence)update.Entity).UUID == AgentId) + { // Self updates go into a special list terseAgentUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); + terseAgentUpdates.Value.Add(update); + } else + { // Everything else goes here terseUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); + terseUpdates.Value.Add(update); + } } #endregion Block Construction @@ -3713,28 +3755,23 @@ namespace OpenSim.Region.ClientStack.LindenUDP #region Packet Sending - - //const float TIME_DILATION = 1.0f; - - ushort timeDilation = Utils.FloatToUInt16(avgTimeDilation, 0.0f, 1.0f); - + if (terseAgentUpdateBlocks.IsValueCreated) { List blocks = terseAgentUpdateBlocks.Value; - + ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; packet.RegionData.TimeDilation = timeDilation; packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; - + for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - - OutPacket(packet, ThrottleOutPacketType.Unknown, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Unknown, true, delegate() { ResendPrimUpdates(terseAgentUpdates.Value); }); } - + if (objectUpdateBlocks.IsValueCreated) { List blocks = objectUpdateBlocks.Value; @@ -3746,8 +3783,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(objectUpdates.Value); }); } if (compressedUpdateBlocks.IsValueCreated) @@ -3761,10 +3798,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(compressedUpdates.Value); }); } - + if (terseUpdateBlocks.IsValueCreated) { List blocks = terseUpdateBlocks.Value; @@ -3776,8 +3813,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(terseUpdates.Value); }); } } @@ -3969,7 +4006,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP { SendFamilyProps = SendFamilyProps || update.SendFamilyProps; SendObjectProps = SendObjectProps || update.SendObjectProps; - Flags |= update.Flags; + // other properties may need to be updated by base class + base.Update(update); } } @@ -11362,6 +11400,22 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// packets (the default), or false to disable splitting if the calling code /// handles splitting manually protected void OutPacket(Packet packet, ThrottleOutPacketType throttlePacketType, bool doAutomaticSplitting) + { + OutPacket(packet, throttlePacketType, doAutomaticSplitting, null); + } + + /// + /// This is the starting point for sending a simulator packet out to the client + /// + /// Packet to send + /// Throttling category for the packet + /// True to automatically split oversized + /// packets (the default), or false to disable splitting if the calling code + /// handles splitting manually + /// The method to be called in the event this packet is reliable + /// and unacknowledged. The server will provide normal resend capability if you do not + /// provide your own method. + protected void OutPacket(Packet packet, ThrottleOutPacketType throttlePacketType, bool doAutomaticSplitting, UnackedPacketMethod method) { if (m_debugPacketLevel > 0) { @@ -11388,7 +11442,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_log.DebugFormat("[CLIENT]: Packet OUT {0}", packet.Type); } - m_udpServer.SendPacket(m_udpClient, packet, throttlePacketType, doAutomaticSplitting); + m_udpServer.SendPacket(m_udpClient, packet, throttlePacketType, doAutomaticSplitting, method); } public bool AddMoney(int debit) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs index d08b25f79d..0848979c3a 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs @@ -297,7 +297,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP delegate(IClientAPI client) { if (client is LLClientView) - SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category); + SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category, null); } ); } @@ -309,7 +309,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP delegate(IClientAPI client) { if (client is LLClientView) - SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category); + SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category, null); } ); } @@ -322,7 +322,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// /// - public void SendPacket(LLUDPClient udpClient, Packet packet, ThrottleOutPacketType category, bool allowSplitting) + public void SendPacket(LLUDPClient udpClient, Packet packet, ThrottleOutPacketType category, bool allowSplitting, UnackedPacketMethod method) { // CoarseLocationUpdate packets cannot be split in an automated way if (packet.Type == PacketType.CoarseLocationUpdate && allowSplitting) @@ -339,13 +339,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < packetCount; i++) { byte[] data = datas[i]; - SendPacketData(udpClient, data, packet.Type, category); + SendPacketData(udpClient, data, packet.Type, category, method); } } else { byte[] data = packet.ToBytes(); - SendPacketData(udpClient, data, packet.Type, category); + SendPacketData(udpClient, data, packet.Type, category, method); } } @@ -356,7 +356,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// /// - public void SendPacketData(LLUDPClient udpClient, byte[] data, PacketType type, ThrottleOutPacketType category) + public void SendPacketData(LLUDPClient udpClient, byte[] data, PacketType type, ThrottleOutPacketType category, UnackedPacketMethod method) { int dataLength = data.Length; bool doZerocode = (data[0] & Helpers.MSG_ZEROCODED) != 0; @@ -411,7 +411,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP #region Queue or Send - OutgoingPacket outgoingPacket = new OutgoingPacket(udpClient, buffer, category); + OutgoingPacket outgoingPacket = new OutgoingPacket(udpClient, buffer, category, null); + // If we were not provided a method for handling unacked, use the UDPServer default method + outgoingPacket.UnackedMethod = ((method == null) ? delegate() { ResendUnacked(outgoingPacket); } : method); // If a Linden Lab 1.23.5 client receives an update packet after a kill packet for an object, it will // continue to display the deleted object until relog. Therefore, we need to always queue a kill object @@ -445,7 +447,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP packet.Header.Reliable = false; packet.Packets = blocks.ToArray(); - SendPacket(udpClient, packet, ThrottleOutPacketType.Unknown, true); + SendPacket(udpClient, packet, ThrottleOutPacketType.Unknown, true, null); } } @@ -458,17 +460,17 @@ namespace OpenSim.Region.ClientStack.LindenUDP // We *could* get OldestUnacked, but it would hurt performance and not provide any benefit pc.PingID.OldestUnacked = 0; - SendPacket(udpClient, pc, ThrottleOutPacketType.Unknown, false); + SendPacket(udpClient, pc, ThrottleOutPacketType.Unknown, false, null); } public void CompletePing(LLUDPClient udpClient, byte pingID) { CompletePingCheckPacket completePing = new CompletePingCheckPacket(); completePing.PingID.PingID = pingID; - SendPacket(udpClient, completePing, ThrottleOutPacketType.Unknown, false); + SendPacket(udpClient, completePing, ThrottleOutPacketType.Unknown, false, null); } - public void ResendUnacked(LLUDPClient udpClient) + public void HandleUnacked(LLUDPClient udpClient) { if (!udpClient.IsConnected) return; @@ -488,33 +490,31 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (expiredPackets != null) { - //m_log.Debug("[LLUDPSERVER]: Resending " + expiredPackets.Count + " packets to " + udpClient.AgentID + ", RTO=" + udpClient.RTO); - + //m_log.Debug("[LLUDPSERVER]: Handling " + expiredPackets.Count + " packets to " + udpClient.AgentID + ", RTO=" + udpClient.RTO); // Exponential backoff of the retransmission timeout udpClient.BackoffRTO(); - - // Resend packets - for (int i = 0; i < expiredPackets.Count; i++) - { - OutgoingPacket outgoingPacket = expiredPackets[i]; - - //m_log.DebugFormat("[LLUDPSERVER]: Resending packet #{0} (attempt {1}), {2}ms have passed", - // outgoingPacket.SequenceNumber, outgoingPacket.ResendCount, Environment.TickCount - outgoingPacket.TickCount); - - // Set the resent flag - outgoingPacket.Buffer.Data[0] = (byte)(outgoingPacket.Buffer.Data[0] | Helpers.MSG_RESENT); - outgoingPacket.Category = ThrottleOutPacketType.Resend; - - // Bump up the resend count on this packet - Interlocked.Increment(ref outgoingPacket.ResendCount); - - // Requeue or resend the packet - if (!outgoingPacket.Client.EnqueueOutgoing(outgoingPacket, false)) - SendPacketFinal(outgoingPacket); - } + for (int i = 0; i < expiredPackets.Count; ++i) + expiredPackets[i].UnackedMethod(); } } + public void ResendUnacked(OutgoingPacket outgoingPacket) + { + //m_log.DebugFormat("[LLUDPSERVER]: Resending packet #{0} (attempt {1}), {2}ms have passed", + // outgoingPacket.SequenceNumber, outgoingPacket.ResendCount, Environment.TickCount - outgoingPacket.TickCount); + + // Set the resent flag + outgoingPacket.Buffer.Data[0] = (byte)(outgoingPacket.Buffer.Data[0] | Helpers.MSG_RESENT); + outgoingPacket.Category = ThrottleOutPacketType.Resend; + + // Bump up the resend count on this packet + Interlocked.Increment(ref outgoingPacket.ResendCount); + + // Requeue or resend the packet + if (!outgoingPacket.Client.EnqueueOutgoing(outgoingPacket, false)) + SendPacketFinal(outgoingPacket); + } + public void Flush(LLUDPClient udpClient) { // FIXME: Implement? @@ -1096,7 +1096,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (udpClient.IsConnected) { if (m_resendUnacked) - ResendUnacked(udpClient); + HandleUnacked(udpClient); if (m_sendAcks) SendAcks(udpClient); @@ -1152,7 +1152,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP nticksUnack++; watch2.Start(); - ResendUnacked(udpClient); + HandleUnacked(udpClient); watch2.Stop(); avgResendUnackedTicks = (nticksUnack - 1)/(float)nticksUnack * avgResendUnackedTicks + (watch2.ElapsedTicks / (float)nticksUnack); diff --git a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs index 1a1a1cb4e3..f4f024b35f 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs @@ -31,6 +31,7 @@ using OpenMetaverse; namespace OpenSim.Region.ClientStack.LindenUDP { + public delegate void UnackedPacketMethod(); /// /// Holds a reference to the this packet is /// destined for, along with the serialized packet data, sequence number @@ -52,6 +53,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP public int TickCount; /// Category this packet belongs to public ThrottleOutPacketType Category; + /// The delegate to be called if this packet is determined to be unacknowledged + public UnackedPacketMethod UnackedMethod; /// /// Default constructor @@ -60,11 +63,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Serialized packet data. If the flags or sequence number /// need to be updated, they will be injected directly into this binary buffer /// Throttling category for this packet - public OutgoingPacket(LLUDPClient client, UDPPacketBuffer buffer, ThrottleOutPacketType category) + public OutgoingPacket(LLUDPClient client, UDPPacketBuffer buffer, ThrottleOutPacketType category, UnackedPacketMethod method) { Client = client; Buffer = buffer; Category = category; + UnackedMethod = method; } } } From 08d8a3e5808b790fbbd7ba3f460603db66aeaff2 Mon Sep 17 00:00:00 2001 From: Dan Lake Date: Mon, 18 Apr 2011 16:48:49 -0700 Subject: [PATCH 02/45] Requeue unacknowledged entity updates rather than resend then "as is". Often, by the time the UDPServer realizes that an entity update packet has not been acknowledged, there is a newer update for the same entity already queued up or there is a higher priority update that should be sent first. This patch eliminates 1:1 packet resends for unacked entity update packets. Insteawd, unacked update packets are decomposed into the original entity updates and those updates are placed back into the priority queues based on their new priority but the original update timestamp. This will generally place them at the head of the line to be put back on the wire as a new outgoing packet but prevents the resend queue from filling up with multiple stale updates for the same entity. This new approach takes advantage of the UDP nature of the Linden protocol in that the intent of a reliable update packet is that if it goes unacknowledge, SOMETHING has to happen to get the update to the client. We are simply making sure that we are resending current object state rather than stale object state. Additionally, this patch includes a generalized callback mechanism so that any caller can specify their own method to call when a packet expires without being acknowledged. We use this mechanism to requeue update packets and otherwise use the UDPServer default method of just putting expired packets in the resend queue. --- OpenSim/Framework/IClientAPI.cs | 59 +++++++++--- OpenSim/Framework/Util.cs | 17 ++++ .../ClientStack/LindenUDP/LLClientView.cs | 96 +++++++++++++++---- .../ClientStack/LindenUDP/LLUDPServer.cs | 70 +++++++------- .../ClientStack/LindenUDP/OutgoingPacket.cs | 6 +- 5 files changed, 179 insertions(+), 69 deletions(-) diff --git a/OpenSim/Framework/IClientAPI.cs b/OpenSim/Framework/IClientAPI.cs index f573c32d07..069987bd71 100644 --- a/OpenSim/Framework/IClientAPI.cs +++ b/OpenSim/Framework/IClientAPI.cs @@ -572,34 +572,69 @@ namespace OpenSim.Framework public class IEntityUpdate { - public ISceneEntity Entity; - public uint Flags; + private ISceneEntity m_entity; + private uint m_flags; + private int m_updateTime; + + public ISceneEntity Entity + { + get { return m_entity; } + } + + public uint Flags + { + get { return m_flags; } + } + + public int UpdateTime + { + get { return m_updateTime; } + } public virtual void Update(IEntityUpdate update) { - this.Flags |= update.Flags; + m_flags |= update.Flags; + + // Use the older of the updates as the updateTime + if (Util.EnvironmentTickCountCompare(UpdateTime, update.UpdateTime) > 0) + m_updateTime = update.UpdateTime; } public IEntityUpdate(ISceneEntity entity, uint flags) { - Entity = entity; - Flags = flags; + m_entity = entity; + m_flags = flags; + m_updateTime = Util.EnvironmentTickCount(); + } + + public IEntityUpdate(ISceneEntity entity, uint flags, Int32 updateTime) + { + m_entity = entity; + m_flags = flags; + m_updateTime = updateTime; } } - public class EntityUpdate : IEntityUpdate { - // public ISceneEntity Entity; - // public PrimUpdateFlags Flags; - public float TimeDilation; + private float m_timeDilation; + + public float TimeDilation + { + get { return m_timeDilation; } + } public EntityUpdate(ISceneEntity entity, PrimUpdateFlags flags, float timedilation) - : base(entity,(uint)flags) + : base(entity, (uint)flags) { - //Entity = entity; // Flags = flags; - TimeDilation = timedilation; + m_timeDilation = timedilation; + } + + public EntityUpdate(ISceneEntity entity, PrimUpdateFlags flags, float timedilation, Int32 updateTime) + : base(entity,(uint)flags,updateTime) + { + m_timeDilation = timedilation; } } diff --git a/OpenSim/Framework/Util.cs b/OpenSim/Framework/Util.cs index 5a5046e547..aaa2724532 100644 --- a/OpenSim/Framework/Util.cs +++ b/OpenSim/Framework/Util.cs @@ -1537,6 +1537,23 @@ namespace OpenSim.Framework return (diff >= 0) ? diff : (diff + EnvironmentTickCountMask + 1); } + // Returns value of Tick Count A - TickCount B accounting for wrapping of TickCount + // Assumes both tcA and tcB came from previous calls to Util.EnvironmentTickCount(). + // A positive return value indicates A occured later than B + public static Int32 EnvironmentTickCountCompare(Int32 tcA, Int32 tcB) + { + // A, B and TC are all between 0 and 0x3fffffff + int tc = EnvironmentTickCount(); + + if (tc - tcA >= 0) + tcA += EnvironmentTickCountMask + 1; + + if (tc - tcB >= 0) + tcB += EnvironmentTickCountMask + 1; + + return tcA - tcB; + } + /// /// Prints the call stack at any given point. Useful for debugging. /// diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 1f7e66dabb..14c5d6c679 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3561,6 +3561,34 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_entityUpdates.Enqueue(priority, new EntityUpdate(entity, updateFlags, m_scene.TimeDilation)); } + /// + /// Requeue an EntityUpdate when it was not acknowledged by the client. + /// We will update the priority and put it in the correct queue, merging update flags + /// with any other updates that may be queued for the same entity. + /// The original update time is used for the merged update. + /// + public void ResendPrimUpdate(EntityUpdate update) + { + // If the update exists in priority queue, it will be updated. + // If it does not exist then it will be added with the current (rather than its original) priority + uint priority = m_prioritizer.GetUpdatePriority(this, update.Entity); + + lock (m_entityUpdates.SyncRoot) + m_entityUpdates.Enqueue(priority, update); + } + + /// + /// Requeue a list of EntityUpdates when they were not acknowledged by the client. + /// We will update the priority and put it in the correct queue, merging update flags + /// with any other updates that may be queued for the same entity. + /// The original update time is used for the merged update. + /// + void ResendPrimUpdates(List updates) + { + foreach (EntityUpdate update in updates) + ResendPrimUpdate(update); + } + private void ProcessEntityUpdates(int maxUpdates) { OpenSim.Framework.Lazy> objectUpdateBlocks = new OpenSim.Framework.Lazy>(); @@ -3568,6 +3596,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP OpenSim.Framework.Lazy> terseUpdateBlocks = new OpenSim.Framework.Lazy>(); OpenSim.Framework.Lazy> terseAgentUpdateBlocks = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> objectUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> compressedUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> terseUpdates = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> terseAgentUpdates = new OpenSim.Framework.Lazy>(); + // Check to see if this is a flush if (maxUpdates <= 0) { @@ -3583,7 +3616,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP float avgTimeDilation = 1.0f; IEntityUpdate iupdate; Int32 timeinqueue; // this is just debugging code & can be dropped later - + while (updatesThisCall < maxUpdates) { lock (m_entityUpdates.SyncRoot) @@ -3688,24 +3721,33 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (update.Entity is ScenePresence) { objectUpdateBlocks.Value.Add(CreateAvatarUpdateBlock((ScenePresence)update.Entity)); + objectUpdates.Value.Add(update); } else { objectUpdateBlocks.Value.Add(CreatePrimUpdateBlock((SceneObjectPart)update.Entity, this.m_agentId)); + objectUpdates.Value.Add(update); } } else if (!canUseImproved) { compressedUpdateBlocks.Value.Add(CreateCompressedUpdateBlock((SceneObjectPart)update.Entity, updateFlags)); + compressedUpdates.Value.Add(update); } else { if (update.Entity is ScenePresence && ((ScenePresence)update.Entity).UUID == AgentId) + { // Self updates go into a special list terseAgentUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); + terseAgentUpdates.Value.Add(update); + } else + { // Everything else goes here terseUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); + terseUpdates.Value.Add(update); + } } #endregion Block Construction @@ -3713,28 +3755,23 @@ namespace OpenSim.Region.ClientStack.LindenUDP #region Packet Sending - - //const float TIME_DILATION = 1.0f; - - ushort timeDilation = Utils.FloatToUInt16(avgTimeDilation, 0.0f, 1.0f); - + if (terseAgentUpdateBlocks.IsValueCreated) { List blocks = terseAgentUpdateBlocks.Value; - + ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; packet.RegionData.TimeDilation = timeDilation; packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; - + for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - - OutPacket(packet, ThrottleOutPacketType.Unknown, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Unknown, true, delegate() { ResendPrimUpdates(terseAgentUpdates.Value); }); } - + if (objectUpdateBlocks.IsValueCreated) { List blocks = objectUpdateBlocks.Value; @@ -3746,8 +3783,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(objectUpdates.Value); }); } if (compressedUpdateBlocks.IsValueCreated) @@ -3761,10 +3798,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(compressedUpdates.Value); }); } - + if (terseUpdateBlocks.IsValueCreated) { List blocks = terseUpdateBlocks.Value; @@ -3776,8 +3813,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; - - OutPacket(packet, ThrottleOutPacketType.Task, true); + // If any of the packets created from this call go unacknowledged, all of the updates will be resent + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(terseUpdates.Value); }); } } @@ -3969,7 +4006,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP { SendFamilyProps = SendFamilyProps || update.SendFamilyProps; SendObjectProps = SendObjectProps || update.SendObjectProps; - Flags |= update.Flags; + // other properties may need to be updated by base class + base.Update(update); } } @@ -11362,6 +11400,22 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// packets (the default), or false to disable splitting if the calling code /// handles splitting manually protected void OutPacket(Packet packet, ThrottleOutPacketType throttlePacketType, bool doAutomaticSplitting) + { + OutPacket(packet, throttlePacketType, doAutomaticSplitting, null); + } + + /// + /// This is the starting point for sending a simulator packet out to the client + /// + /// Packet to send + /// Throttling category for the packet + /// True to automatically split oversized + /// packets (the default), or false to disable splitting if the calling code + /// handles splitting manually + /// The method to be called in the event this packet is reliable + /// and unacknowledged. The server will provide normal resend capability if you do not + /// provide your own method. + protected void OutPacket(Packet packet, ThrottleOutPacketType throttlePacketType, bool doAutomaticSplitting, UnackedPacketMethod method) { if (m_debugPacketLevel > 0) { @@ -11388,7 +11442,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_log.DebugFormat("[CLIENT]: Packet OUT {0}", packet.Type); } - m_udpServer.SendPacket(m_udpClient, packet, throttlePacketType, doAutomaticSplitting); + m_udpServer.SendPacket(m_udpClient, packet, throttlePacketType, doAutomaticSplitting, method); } public bool AddMoney(int debit) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs index d08b25f79d..0848979c3a 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs @@ -297,7 +297,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP delegate(IClientAPI client) { if (client is LLClientView) - SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category); + SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category, null); } ); } @@ -309,7 +309,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP delegate(IClientAPI client) { if (client is LLClientView) - SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category); + SendPacketData(((LLClientView)client).UDPClient, data, packet.Type, category, null); } ); } @@ -322,7 +322,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// /// - public void SendPacket(LLUDPClient udpClient, Packet packet, ThrottleOutPacketType category, bool allowSplitting) + public void SendPacket(LLUDPClient udpClient, Packet packet, ThrottleOutPacketType category, bool allowSplitting, UnackedPacketMethod method) { // CoarseLocationUpdate packets cannot be split in an automated way if (packet.Type == PacketType.CoarseLocationUpdate && allowSplitting) @@ -339,13 +339,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < packetCount; i++) { byte[] data = datas[i]; - SendPacketData(udpClient, data, packet.Type, category); + SendPacketData(udpClient, data, packet.Type, category, method); } } else { byte[] data = packet.ToBytes(); - SendPacketData(udpClient, data, packet.Type, category); + SendPacketData(udpClient, data, packet.Type, category, method); } } @@ -356,7 +356,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// /// - public void SendPacketData(LLUDPClient udpClient, byte[] data, PacketType type, ThrottleOutPacketType category) + public void SendPacketData(LLUDPClient udpClient, byte[] data, PacketType type, ThrottleOutPacketType category, UnackedPacketMethod method) { int dataLength = data.Length; bool doZerocode = (data[0] & Helpers.MSG_ZEROCODED) != 0; @@ -411,7 +411,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP #region Queue or Send - OutgoingPacket outgoingPacket = new OutgoingPacket(udpClient, buffer, category); + OutgoingPacket outgoingPacket = new OutgoingPacket(udpClient, buffer, category, null); + // If we were not provided a method for handling unacked, use the UDPServer default method + outgoingPacket.UnackedMethod = ((method == null) ? delegate() { ResendUnacked(outgoingPacket); } : method); // If a Linden Lab 1.23.5 client receives an update packet after a kill packet for an object, it will // continue to display the deleted object until relog. Therefore, we need to always queue a kill object @@ -445,7 +447,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP packet.Header.Reliable = false; packet.Packets = blocks.ToArray(); - SendPacket(udpClient, packet, ThrottleOutPacketType.Unknown, true); + SendPacket(udpClient, packet, ThrottleOutPacketType.Unknown, true, null); } } @@ -458,17 +460,17 @@ namespace OpenSim.Region.ClientStack.LindenUDP // We *could* get OldestUnacked, but it would hurt performance and not provide any benefit pc.PingID.OldestUnacked = 0; - SendPacket(udpClient, pc, ThrottleOutPacketType.Unknown, false); + SendPacket(udpClient, pc, ThrottleOutPacketType.Unknown, false, null); } public void CompletePing(LLUDPClient udpClient, byte pingID) { CompletePingCheckPacket completePing = new CompletePingCheckPacket(); completePing.PingID.PingID = pingID; - SendPacket(udpClient, completePing, ThrottleOutPacketType.Unknown, false); + SendPacket(udpClient, completePing, ThrottleOutPacketType.Unknown, false, null); } - public void ResendUnacked(LLUDPClient udpClient) + public void HandleUnacked(LLUDPClient udpClient) { if (!udpClient.IsConnected) return; @@ -488,33 +490,31 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (expiredPackets != null) { - //m_log.Debug("[LLUDPSERVER]: Resending " + expiredPackets.Count + " packets to " + udpClient.AgentID + ", RTO=" + udpClient.RTO); - + //m_log.Debug("[LLUDPSERVER]: Handling " + expiredPackets.Count + " packets to " + udpClient.AgentID + ", RTO=" + udpClient.RTO); // Exponential backoff of the retransmission timeout udpClient.BackoffRTO(); - - // Resend packets - for (int i = 0; i < expiredPackets.Count; i++) - { - OutgoingPacket outgoingPacket = expiredPackets[i]; - - //m_log.DebugFormat("[LLUDPSERVER]: Resending packet #{0} (attempt {1}), {2}ms have passed", - // outgoingPacket.SequenceNumber, outgoingPacket.ResendCount, Environment.TickCount - outgoingPacket.TickCount); - - // Set the resent flag - outgoingPacket.Buffer.Data[0] = (byte)(outgoingPacket.Buffer.Data[0] | Helpers.MSG_RESENT); - outgoingPacket.Category = ThrottleOutPacketType.Resend; - - // Bump up the resend count on this packet - Interlocked.Increment(ref outgoingPacket.ResendCount); - - // Requeue or resend the packet - if (!outgoingPacket.Client.EnqueueOutgoing(outgoingPacket, false)) - SendPacketFinal(outgoingPacket); - } + for (int i = 0; i < expiredPackets.Count; ++i) + expiredPackets[i].UnackedMethod(); } } + public void ResendUnacked(OutgoingPacket outgoingPacket) + { + //m_log.DebugFormat("[LLUDPSERVER]: Resending packet #{0} (attempt {1}), {2}ms have passed", + // outgoingPacket.SequenceNumber, outgoingPacket.ResendCount, Environment.TickCount - outgoingPacket.TickCount); + + // Set the resent flag + outgoingPacket.Buffer.Data[0] = (byte)(outgoingPacket.Buffer.Data[0] | Helpers.MSG_RESENT); + outgoingPacket.Category = ThrottleOutPacketType.Resend; + + // Bump up the resend count on this packet + Interlocked.Increment(ref outgoingPacket.ResendCount); + + // Requeue or resend the packet + if (!outgoingPacket.Client.EnqueueOutgoing(outgoingPacket, false)) + SendPacketFinal(outgoingPacket); + } + public void Flush(LLUDPClient udpClient) { // FIXME: Implement? @@ -1096,7 +1096,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (udpClient.IsConnected) { if (m_resendUnacked) - ResendUnacked(udpClient); + HandleUnacked(udpClient); if (m_sendAcks) SendAcks(udpClient); @@ -1152,7 +1152,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP nticksUnack++; watch2.Start(); - ResendUnacked(udpClient); + HandleUnacked(udpClient); watch2.Stop(); avgResendUnackedTicks = (nticksUnack - 1)/(float)nticksUnack * avgResendUnackedTicks + (watch2.ElapsedTicks / (float)nticksUnack); diff --git a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs index 1a1a1cb4e3..f4f024b35f 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs @@ -31,6 +31,7 @@ using OpenMetaverse; namespace OpenSim.Region.ClientStack.LindenUDP { + public delegate void UnackedPacketMethod(); /// /// Holds a reference to the this packet is /// destined for, along with the serialized packet data, sequence number @@ -52,6 +53,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP public int TickCount; /// Category this packet belongs to public ThrottleOutPacketType Category; + /// The delegate to be called if this packet is determined to be unacknowledged + public UnackedPacketMethod UnackedMethod; /// /// Default constructor @@ -60,11 +63,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Serialized packet data. If the flags or sequence number /// need to be updated, they will be injected directly into this binary buffer /// Throttling category for this packet - public OutgoingPacket(LLUDPClient client, UDPPacketBuffer buffer, ThrottleOutPacketType category) + public OutgoingPacket(LLUDPClient client, UDPPacketBuffer buffer, ThrottleOutPacketType category, UnackedPacketMethod method) { Client = client; Buffer = buffer; Category = category; + UnackedMethod = method; } } } From 82de87ce99a513debabe16af41fcb2fda3db0484 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Tue, 19 Apr 2011 11:22:04 -0700 Subject: [PATCH 03/45] Converted the property request queue to use the same retransmission mechanism as the entity update queues. --- .../ClientStack/LindenUDP/LLClientView.cs | 75 +++++++++++++------ 1 file changed, 52 insertions(+), 23 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 14c5d6c679..87b86eb7d1 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3567,7 +3567,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// with any other updates that may be queued for the same entity. /// The original update time is used for the merged update. /// - public void ResendPrimUpdate(EntityUpdate update) + private void ResendPrimUpdate(EntityUpdate update) { // If the update exists in priority queue, it will be updated. // If it does not exist then it will be added with the current (rather than its original) priority @@ -3583,7 +3583,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// with any other updates that may be queued for the same entity. /// The original update time is used for the merged update. /// - void ResendPrimUpdates(List updates) + private void ResendPrimUpdates(List updates) { foreach (EntityUpdate update in updates) ResendPrimUpdate(update); @@ -4018,6 +4018,19 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_entityProps.Enqueue(priority, new ObjectPropertyUpdate(entity,requestFlags,true,false)); } + private void ResendPropertyUpdate(ObjectPropertyUpdate update) + { + uint priority = 0; + lock (m_entityProps.SyncRoot) + m_entityProps.Enqueue(priority, update); + } + + private void ResendPropertyUpdates(List updates) + { + foreach (ObjectPropertyUpdate update in updates) + ResendPropertyUpdate(update); + } + public void SendObjectPropertiesReply(ISceneEntity entity) { uint priority = 0; // time based ordering only @@ -4033,6 +4046,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP OpenSim.Framework.Lazy> objectPropertiesBlocks = new OpenSim.Framework.Lazy>(); + OpenSim.Framework.Lazy> familyUpdates = + new OpenSim.Framework.Lazy>(); + + OpenSim.Framework.Lazy> propertyUpdates = + new OpenSim.Framework.Lazy>(); + IEntityUpdate iupdate; Int32 timeinqueue; // this is just debugging code & can be dropped later @@ -4051,6 +4070,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP SceneObjectPart sop = (SceneObjectPart)update.Entity; ObjectPropertiesFamilyPacket.ObjectDataBlock objPropDB = CreateObjectPropertiesFamilyBlock(sop,update.Flags); objectFamilyBlocks.Value.Add(objPropDB); + familyUpdates.Value.Add(update); } } @@ -4061,6 +4081,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP SceneObjectPart sop = (SceneObjectPart)update.Entity; ObjectPropertiesPacket.ObjectDataBlock objPropDB = CreateObjectPropertiesBlock(sop); objectPropertiesBlocks.Value.Add(objPropDB); + propertyUpdates.Value.Add(update); } } @@ -4068,12 +4089,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP } - Int32 ppcnt = 0; - Int32 pbcnt = 0; + // Int32 ppcnt = 0; + // Int32 pbcnt = 0; if (objectPropertiesBlocks.IsValueCreated) { List blocks = objectPropertiesBlocks.Value; + List updates = propertyUpdates.Value; ObjectPropertiesPacket packet = (ObjectPropertiesPacket)PacketPool.Instance.GetPacket(PacketType.ObjectProperties); packet.ObjectData = new ObjectPropertiesPacket.ObjectDataBlock[blocks.Count]; @@ -4081,28 +4103,27 @@ namespace OpenSim.Region.ClientStack.LindenUDP packet.ObjectData[i] = blocks[i]; packet.Header.Zerocoded = true; - OutPacket(packet, ThrottleOutPacketType.Task, true); - pbcnt += blocks.Count; - ppcnt++; + // Pass in the delegate so that if this packet needs to be resent, we send the current properties + // of the object rather than the properties when the packet was created + OutPacket(packet, ThrottleOutPacketType.Task, true, + delegate() + { + ResendPropertyUpdates(updates); + }); + + // pbcnt += blocks.Count; + // ppcnt++; } - Int32 fpcnt = 0; - Int32 fbcnt = 0; + // Int32 fpcnt = 0; + // Int32 fbcnt = 0; if (objectFamilyBlocks.IsValueCreated) { List blocks = objectFamilyBlocks.Value; - - // ObjectPropertiesFamilyPacket objPropFamilyPack = - // (ObjectPropertiesFamilyPacket)PacketPool.Instance.GetPacket(PacketType.ObjectPropertiesFamily); - // - // objPropFamilyPack.ObjectData = new ObjectPropertiesFamilyPacket.ObjectDataBlock[blocks.Count]; - // for (int i = 0; i < blocks.Count; i++) - // objPropFamilyPack.ObjectData[i] = blocks[i]; - // - // OutPacket(objPropFamilyPack, ThrottleOutPacketType.Task, true); - + List updates = familyUpdates.Value; + // one packet per object block... uggh... for (int i = 0; i < blocks.Count; i++) { @@ -4111,10 +4132,18 @@ namespace OpenSim.Region.ClientStack.LindenUDP packet.ObjectData = blocks[i]; packet.Header.Zerocoded = true; - OutPacket(packet, ThrottleOutPacketType.Task); - fpcnt++; - fbcnt++; + // Pass in the delegate so that if this packet needs to be resent, we send the current properties + // of the object rather than the properties when the packet was created + ObjectPropertyUpdate update = updates[i]; + OutPacket(packet, ThrottleOutPacketType.Task, true, + delegate() + { + ResendPropertyUpdate(update); + }); + + // fpcnt++; + // fbcnt++; } } @@ -4151,7 +4180,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP return block; } - + private ObjectPropertiesPacket.ObjectDataBlock CreateObjectPropertiesBlock(SceneObjectPart sop) { //ObjectPropertiesPacket proper = (ObjectPropertiesPacket)PacketPool.Instance.GetPacket(PacketType.ObjectProperties); From 2b737c9cc2e4a4ef3520d80225381a010bd1dc80 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Wed, 20 Apr 2011 16:23:33 -0700 Subject: [PATCH 04/45] Adds the first pass at an adaptive throttle to slow start new clients. If the sent packets are ack'ed successfully the throttle will open quickly up to the maximum specified by the client and/or the sims client throttle. This still needs a lot of adjustment to get the rates correct. --- .../ClientStack/LindenUDP/LLClientView.cs | 4 + .../ClientStack/LindenUDP/LLUDPClient.cs | 9 +- .../ClientStack/LindenUDP/TokenBucket.cs | 92 +++++++++++++++---- .../LindenUDP/UnackedPacketCollection.cs | 8 ++ 4 files changed, 95 insertions(+), 18 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 87b86eb7d1..11088631e6 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3585,6 +3585,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// private void ResendPrimUpdates(List updates) { + // m_log.WarnFormat("[CLIENT] resending prim update {0}",updates[0].UpdateTime); + foreach (EntityUpdate update in updates) ResendPrimUpdate(update); } @@ -4027,6 +4029,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP private void ResendPropertyUpdates(List updates) { + // m_log.WarnFormat("[CLIENT] resending object property {0}",updates[0].UpdateTime); + foreach (ObjectPropertyUpdate update in updates) ResendPropertyUpdate(update); } diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs index 7be8a0ae59..20bfec8f92 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs @@ -135,7 +135,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP private int m_nextOnQueueEmpty = 1; /// Throttle bucket for this agent's connection - private readonly TokenBucket m_throttleClient; + private readonly AdaptiveTokenBucket m_throttleClient; + public AdaptiveTokenBucket FlowThrottle + { + get { return m_throttleClient; } + } + /// Throttle bucket for this agent's connection private readonly TokenBucket m_throttleCategory; /// Throttle buckets for each packet category @@ -176,7 +181,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_maxRTO = maxRTO; // Create a token bucket throttle for this client that has the scene token bucket as a parent - m_throttleClient = new TokenBucket(parentThrottle, rates.TotalLimit); + m_throttleClient = new AdaptiveTokenBucket(parentThrottle, rates.TotalLimit); // Create a token bucket throttle for the total categary with the client bucket as a throttle m_throttleCategory = new TokenBucket(m_throttleClient, rates.TotalLimit); // Create an array of token buckets for this clients different throttle categories diff --git a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs index 07b0a1df7a..4ee6d3a30a 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs @@ -48,31 +48,31 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Number of ticks (ms) per quantum, drip rate and max burst /// are defined over this interval. /// - private const Int32 m_ticksPerQuantum = 1000; + protected const Int32 m_ticksPerQuantum = 1000; /// /// This is the number of quantums worth of packets that can /// be accommodated during a burst /// - private const Double m_quantumsPerBurst = 1.5; + protected const Double m_quantumsPerBurst = 1.5; /// /// - private const Int32 m_minimumDripRate = 1400; + protected const Int32 m_minimumDripRate = 1400; /// Time of the last drip, in system ticks - private Int32 m_lastDrip; + protected Int32 m_lastDrip; /// /// The number of bytes that can be sent at this moment. This is the /// current number of tokens in the bucket /// - private Int64 m_tokenCount; + protected Int64 m_tokenCount; /// /// Map of children buckets and their requested maximum burst rate /// - private Dictionary m_children = new Dictionary(); + protected Dictionary m_children = new Dictionary(); #region Properties @@ -81,7 +81,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// parent. The parent bucket will limit the aggregate bandwidth of all /// of its children buckets /// - private TokenBucket m_parent; + protected TokenBucket m_parent; public TokenBucket Parent { get { return m_parent; } @@ -93,7 +93,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// of tokens that can accumulate in the bucket at any one time. This /// also sets the total request for leaf nodes /// - private Int64 m_burstRate; + protected Int64 m_burstRate; public Int64 RequestedBurstRate { get { return m_burstRate; } @@ -118,8 +118,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Tokens are added to the bucket any time /// is called, at the granularity of /// the system tick interval (typically around 15-22ms) - private Int64 m_dripRate; - public Int64 RequestedDripRate + protected Int64 m_dripRate; + public virtual Int64 RequestedDripRate { get { return (m_dripRate == 0 ? m_totalDripRequest : m_dripRate); } set { @@ -131,7 +131,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP } } - public Int64 DripRate + public virtual Int64 DripRate { get { if (m_parent == null) @@ -149,7 +149,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// The current total of the requested maximum burst rates of /// this bucket's children buckets. /// - private Int64 m_totalDripRequest; + protected Int64 m_totalDripRequest; public Int64 TotalDripRequest { get { return m_totalDripRequest; } @@ -189,7 +189,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// hierarchy. However, if any of the parents is over-booked, then /// the modifier will be less than 1. /// - private double DripRateModifier() + protected double DripRateModifier() { Int64 driprate = DripRate; return driprate >= TotalDripRequest ? 1.0 : (double)driprate / (double)TotalDripRequest; @@ -197,7 +197,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// - private double BurstRateModifier() + protected double BurstRateModifier() { // for now... burst rate is always m_quantumsPerBurst (constant) // larger than drip rate so the ratio of burst requests is the @@ -268,7 +268,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Deposit tokens into the bucket from a child bucket that did /// not use all of its available tokens /// - private void Deposit(Int64 count) + protected void Deposit(Int64 count) { m_tokenCount += count; @@ -285,7 +285,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// call to Drip /// /// True if tokens were added to the bucket, otherwise false - private void Drip() + protected void Drip() { // This should never happen... means we are a leaf node and were created // with no drip rate... @@ -310,4 +310,64 @@ namespace OpenSim.Region.ClientStack.LindenUDP Deposit(deltaMS * DripRate / m_ticksPerQuantum); } } + + public class AdaptiveTokenBucket : TokenBucket + { + private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + + // + // The minimum rate for flow control. + // + protected const Int64 m_minimumFlow = m_minimumDripRate * 10; + + // + // The maximum rate for flow control. Drip rate can never be + // greater than this. + // + protected Int64 m_maxDripRate = 0; + protected Int64 MaxDripRate + { + get { return (m_maxDripRate == 0 ? m_totalDripRequest : m_maxDripRate); } + set { m_maxDripRate = (value == 0 ? 0 : Math.Max(value,m_minimumFlow)); } + } + + // + // + // + public virtual Int64 AdjustedDripRate + { + get { return m_dripRate; } + set { + m_dripRate = OpenSim.Framework.Util.Clamp(value,m_minimumFlow,MaxDripRate); + m_burstRate = (Int64)((double)m_dripRate * m_quantumsPerBurst); + if (m_parent != null) + m_parent.RegisterRequest(this,m_dripRate); + } + } + + // + // + // + public AdaptiveTokenBucket(TokenBucket parent, Int64 maxDripRate) : base(parent,m_minimumFlow) + { + MaxDripRate = maxDripRate; + } + + // + // + // + public void ExpirePackets(Int32 count) + { + // m_log.WarnFormat("[ADAPTIVEBUCKET] drop {0} by {1} expired packets",AdjustedDripRate,count); + AdjustedDripRate = (Int64) (AdjustedDripRate / Math.Pow(2,count)); + } + + // + // + // + public void AcknowledgePackets(Int32 count) + { + AdjustedDripRate = AdjustedDripRate + count; + } + } } diff --git a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs index d195110a23..b1709649f5 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs @@ -130,6 +130,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP // is actually sent out again packet.TickCount = 0; + // As with other network applications, assume that an expired packet is + // an indication of some network problem, slow transmission + packet.Client.FlowThrottle.ExpirePackets(1); + expiredPackets.Add(packet); } } @@ -157,6 +161,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP { m_packets.Remove(pendingRemove.SequenceNumber); + // As with other network applications, assume that an acknowledged packet is an + // indication that the network can handle a little more load, speed up the transmission + ackedPacket.Client.FlowThrottle.AcknowledgePackets(ackedPacket.Buffer.DataLength); + // Update stats Interlocked.Add(ref ackedPacket.Client.UnackedBytes, -ackedPacket.Buffer.DataLength); From 7759bda833c03f4c29500dce32b835a7aef8285b Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Wed, 20 Apr 2011 21:58:49 -0700 Subject: [PATCH 05/45] Added an "immediate" queue to the priority queue. This is per Melanie's very good suggestion. The immediate queue is serviced completely before all others, making it a very good place to put avatar updates & attachments. Moved the priority queue out of the LLUDP directory and into the framework. It is now a fairly general utility. --- .../LindenUDP => Framework}/PriorityQueue.cs | 43 ++++++++++++------- .../Region/Framework/Scenes/Prioritizer.cs | 4 +- 2 files changed, 30 insertions(+), 17 deletions(-) rename OpenSim/{Region/ClientStack/LindenUDP => Framework}/PriorityQueue.cs (84%) diff --git a/OpenSim/Region/ClientStack/LindenUDP/PriorityQueue.cs b/OpenSim/Framework/PriorityQueue.cs similarity index 84% rename from OpenSim/Region/ClientStack/LindenUDP/PriorityQueue.cs rename to OpenSim/Framework/PriorityQueue.cs index b62ec07ed9..eec2a9268e 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/PriorityQueue.cs +++ b/OpenSim/Framework/PriorityQueue.cs @@ -34,20 +34,21 @@ using OpenSim.Framework; using OpenSim.Framework.Client; using log4net; -namespace OpenSim.Region.ClientStack.LindenUDP +namespace OpenSim.Framework { public class PriorityQueue { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - internal delegate bool UpdatePriorityHandler(ref uint priority, ISceneEntity entity); + public delegate bool UpdatePriorityHandler(ref uint priority, ISceneEntity entity); // Heap[0] for self updates // Heap[1..12] for entity updates - internal const uint m_numberOfQueues = 12; + public const uint NumberOfQueues = 12; + public const uint ImmediateQueue = 0; - private MinHeap[] m_heaps = new MinHeap[m_numberOfQueues]; + private MinHeap[] m_heaps = new MinHeap[NumberOfQueues]; private Dictionary m_lookupTable; private uint m_nextQueue = 0; private UInt64 m_nextRequest = 0; @@ -57,9 +58,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP get { return this.m_syncRoot; } } - internal PriorityQueue() : this(MinHeap.DEFAULT_CAPACITY) { } + public PriorityQueue() : this(MinHeap.DEFAULT_CAPACITY) { } - internal PriorityQueue(int capacity) + public PriorityQueue(int capacity) { m_lookupTable = new Dictionary(capacity); @@ -67,7 +68,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_heaps[i] = new MinHeap(capacity); } - internal int Count + public int Count { get { @@ -91,7 +92,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP lookup.Heap.Remove(lookup.Handle); } - pqueue = Util.Clamp(pqueue, 0, m_numberOfQueues - 1); + pqueue = Util.Clamp(pqueue, 0, NumberOfQueues - 1); lookup.Heap = m_heaps[pqueue]; lookup.Heap.Add(new MinHeapItem(pqueue, entry, value), ref lookup.Handle); m_lookupTable[localid] = lookup; @@ -99,18 +100,30 @@ namespace OpenSim.Region.ClientStack.LindenUDP return true; } - internal bool TryDequeue(out IEntityUpdate value, out Int32 timeinqueue) + public bool TryDequeue(out IEntityUpdate value, out Int32 timeinqueue) { - for (int i = 0; i < m_numberOfQueues; ++i) + // If there is anything in priority queue 0, return it first no + // matter what else. Breaks fairness. But very useful. + if (m_heaps[ImmediateQueue].Count > 0) + { + MinHeapItem item = m_heaps[ImmediateQueue].RemoveMin(); + m_lookupTable.Remove(item.Value.Entity.LocalId); + timeinqueue = Util.EnvironmentTickCountSubtract(item.EntryTime); + value = item.Value; + + return true; + } + + for (int i = 0; i < NumberOfQueues; ++i) { // To get the fair queing, we cycle through each of the // queues when finding an element to dequeue, this code // assumes that the distribution of updates in the queues // is polynomial, probably quadractic (eg distance of PI * R^2) - uint h = (uint)((m_nextQueue + i) % m_numberOfQueues); + uint h = (uint)((m_nextQueue + i) % NumberOfQueues); if (m_heaps[h].Count > 0) { - m_nextQueue = (uint)((h + 1) % m_numberOfQueues); + m_nextQueue = (uint)((h + 1) % NumberOfQueues); MinHeapItem item = m_heaps[h].RemoveMin(); m_lookupTable.Remove(item.Value.Entity.LocalId); @@ -126,7 +139,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP return false; } - internal void Reprioritize(UpdatePriorityHandler handler) + public void Reprioritize(UpdatePriorityHandler handler) { MinHeapItem item; foreach (LookupItem lookup in new List(this.m_lookupTable.Values)) @@ -140,7 +153,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP { // unless the priority queue has changed, there is no need to modify // the entry - pqueue = Util.Clamp(pqueue, 0, m_numberOfQueues - 1); + pqueue = Util.Clamp(pqueue, 0, NumberOfQueues - 1); if (pqueue != item.PriorityQueue) { lookup.Heap.Remove(lookup.Handle); @@ -164,7 +177,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP public override string ToString() { string s = ""; - for (int i = 0; i < m_numberOfQueues; i++) + for (int i = 0; i < NumberOfQueues; i++) { if (s != "") s += ","; s += m_heaps[i].Count.ToString(); diff --git a/OpenSim/Region/Framework/Scenes/Prioritizer.cs b/OpenSim/Region/Framework/Scenes/Prioritizer.cs index e3ed905c2f..2e80156572 100644 --- a/OpenSim/Region/Framework/Scenes/Prioritizer.cs +++ b/OpenSim/Region/Framework/Scenes/Prioritizer.cs @@ -88,7 +88,7 @@ namespace OpenSim.Region.Framework.Scenes // If this is an update for our own avatar give it the highest priority if (client.AgentId == entity.UUID) - return 0; + return PriorityQueue.ImmediateQueue; uint priority; @@ -172,7 +172,7 @@ namespace OpenSim.Region.Framework.Scenes // m_log.WarnFormat("[PRIORITIZER] attempt to use agent {0} not in the scene",client.AgentId); // throw new InvalidOperationException("Prioritization agent not defined"); - return Int32.MaxValue; + return PriorityQueue.NumberOfQueues - 1; } // Use group position for child prims, since we are putting child prims in From b5ab33b5e190e79789df3f44b24959fd229a8c6d Mon Sep 17 00:00:00 2001 From: Dan Lake Date: Wed, 20 Apr 2011 23:08:51 -0700 Subject: [PATCH 06/45] bug fix. Now when an unacked update packet is handled through ResendPrimUpdates, it is removed from the UnackedPacketCollection. --- .../ClientStack/LindenUDP/LLClientView.cs | 34 ++++++++++++------- .../ClientStack/LindenUDP/LLUDPServer.cs | 4 +-- .../ClientStack/LindenUDP/OutgoingPacket.cs | 3 +- 3 files changed, 25 insertions(+), 16 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 11088631e6..1e8bbb8832 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3583,10 +3583,14 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// with any other updates that may be queued for the same entity. /// The original update time is used for the merged update. /// - private void ResendPrimUpdates(List updates) + private void ResendPrimUpdates(List updates, OutgoingPacket oPacket) { // m_log.WarnFormat("[CLIENT] resending prim update {0}",updates[0].UpdateTime); - + + // Remove the update packet from the list of packets waiting for acknowledgement + // because we are requeuing the list of updates. They will be resent in new packets + // with the most recent state and priority. + m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber, 0, true); foreach (EntityUpdate update in updates) ResendPrimUpdate(update); } @@ -3771,7 +3775,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; // If any of the packets created from this call go unacknowledged, all of the updates will be resent - OutPacket(packet, ThrottleOutPacketType.Unknown, true, delegate() { ResendPrimUpdates(terseAgentUpdates.Value); }); + OutPacket(packet, ThrottleOutPacketType.Unknown, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(terseAgentUpdates.Value, oPacket); }); } if (objectUpdateBlocks.IsValueCreated) @@ -3786,7 +3790,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; // If any of the packets created from this call go unacknowledged, all of the updates will be resent - OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(objectUpdates.Value); }); + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(objectUpdates.Value, oPacket); }); } if (compressedUpdateBlocks.IsValueCreated) @@ -3801,7 +3805,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; // If any of the packets created from this call go unacknowledged, all of the updates will be resent - OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(compressedUpdates.Value); }); + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(compressedUpdates.Value, oPacket); }); } if (terseUpdateBlocks.IsValueCreated) @@ -3816,7 +3820,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP for (int i = 0; i < blocks.Count; i++) packet.ObjectData[i] = blocks[i]; // If any of the packets created from this call go unacknowledged, all of the updates will be resent - OutPacket(packet, ThrottleOutPacketType.Task, true, delegate() { ResendPrimUpdates(terseUpdates.Value); }); + OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(terseUpdates.Value, oPacket); }); } } @@ -4027,10 +4031,14 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_entityProps.Enqueue(priority, update); } - private void ResendPropertyUpdates(List updates) + private void ResendPropertyUpdates(List updates, OutgoingPacket oPacket) { // m_log.WarnFormat("[CLIENT] resending object property {0}",updates[0].UpdateTime); + // Remove the update packet from the list of packets waiting for acknowledgement + // because we are requeuing the list of updates. They will be resent in new packets + // with the most recent state. + m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber, 0, true); foreach (ObjectPropertyUpdate update in updates) ResendPropertyUpdate(update); } @@ -4111,9 +4119,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Pass in the delegate so that if this packet needs to be resent, we send the current properties // of the object rather than the properties when the packet was created OutPacket(packet, ThrottleOutPacketType.Task, true, - delegate() + delegate(OutgoingPacket oPacket) { - ResendPropertyUpdates(updates); + ResendPropertyUpdates(updates, oPacket); }); // pbcnt += blocks.Count; @@ -4126,7 +4134,6 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (objectFamilyBlocks.IsValueCreated) { List blocks = objectFamilyBlocks.Value; - List updates = familyUpdates.Value; // one packet per object block... uggh... for (int i = 0; i < blocks.Count; i++) @@ -4139,11 +4146,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Pass in the delegate so that if this packet needs to be resent, we send the current properties // of the object rather than the properties when the packet was created - ObjectPropertyUpdate update = updates[i]; + List updates = new List(); + updates.Add(familyUpdates.Value[i]); OutPacket(packet, ThrottleOutPacketType.Task, true, - delegate() + delegate(OutgoingPacket oPacket) { - ResendPropertyUpdate(update); + ResendPropertyUpdates(updates, oPacket); }); // fpcnt++; diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs index 0848979c3a..bd58ddc9e0 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs @@ -413,7 +413,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP OutgoingPacket outgoingPacket = new OutgoingPacket(udpClient, buffer, category, null); // If we were not provided a method for handling unacked, use the UDPServer default method - outgoingPacket.UnackedMethod = ((method == null) ? delegate() { ResendUnacked(outgoingPacket); } : method); + outgoingPacket.UnackedMethod = ((method == null) ? delegate(OutgoingPacket oPacket) { ResendUnacked(oPacket); } : method); // If a Linden Lab 1.23.5 client receives an update packet after a kill packet for an object, it will // continue to display the deleted object until relog. Therefore, we need to always queue a kill object @@ -494,7 +494,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Exponential backoff of the retransmission timeout udpClient.BackoffRTO(); for (int i = 0; i < expiredPackets.Count; ++i) - expiredPackets[i].UnackedMethod(); + expiredPackets[i].UnackedMethod(expiredPackets[i]); } } diff --git a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs index f4f024b35f..76c6c145c8 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/OutgoingPacket.cs @@ -31,7 +31,8 @@ using OpenMetaverse; namespace OpenSim.Region.ClientStack.LindenUDP { - public delegate void UnackedPacketMethod(); + + public delegate void UnackedPacketMethod(OutgoingPacket oPacket); /// /// Holds a reference to the this packet is /// destined for, along with the serialized packet data, sequence number From 3640d0204f77dff3b1c4bd50229b60a49d2745e5 Mon Sep 17 00:00:00 2001 From: Dan Lake Date: Thu, 21 Apr 2011 01:51:08 -0700 Subject: [PATCH 07/45] Added ability to remove unacked packet from UnackedPacketCollection without an acknowledgement from the network. This prevents RTT and throttles from being updated as they would when an ACK is actually received. Also fixed stats logging for unacked bytes and resent packets in this case. --- .../ClientStack/LindenUDP/LLClientView.cs | 12 ++++++-- .../LindenUDP/UnackedPacketCollection.cs | 28 +++++++++++++++++++ 2 files changed, 38 insertions(+), 2 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 1e8bbb8832..6129e10661 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -3590,7 +3590,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Remove the update packet from the list of packets waiting for acknowledgement // because we are requeuing the list of updates. They will be resent in new packets // with the most recent state and priority. - m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber, 0, true); + m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber); + + // Count this as a resent packet since we are going to requeue all of the updates contained in it + Interlocked.Increment(ref m_udpClient.PacketsResent); + foreach (EntityUpdate update in updates) ResendPrimUpdate(update); } @@ -4038,7 +4042,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Remove the update packet from the list of packets waiting for acknowledgement // because we are requeuing the list of updates. They will be resent in new packets // with the most recent state. - m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber, 0, true); + m_udpClient.NeedAcks.Remove(oPacket.SequenceNumber); + + // Count this as a resent packet since we are going to requeue all of the updates contained in it + Interlocked.Increment(ref m_udpClient.PacketsResent); + foreach (ObjectPropertyUpdate update in updates) ResendPropertyUpdate(update); } diff --git a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs index b1709649f5..90a87faec4 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs @@ -83,6 +83,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// Marks a packet as acknowledged + /// This method is used when an acknowledgement is received from the network for a previously + /// sent packet. Effects of removal this way are to update unacked byte count, adjust RTT + /// and increase throttle to the coresponding client. /// /// Sequence number of the packet to /// acknowledge @@ -94,6 +97,31 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_pendingRemoves.Enqueue(new PendingAck(sequenceNumber, currentTime, fromResend)); } + /// + /// Marks a packet as no longer needing acknowledgement without a received acknowledgement. + /// This method is called when a packet expires and we no longer need an acknowledgement. + /// When some reliable packet types expire, they are handled in a way other than simply + /// resending them. The only effect of removal this way is to update unacked byte count. + /// + /// Sequence number of the packet to + /// acknowledge + /// The packet is removed from the collection immediately. + /// This function is not threadsafe. It must be called by the thread calling GetExpiredPackets. + public void Remove(uint sequenceNumber) + { + OutgoingPacket removedPacket; + if (m_packets.TryGetValue(sequenceNumber, out removedPacket)) + { + if (removedPacket != null) + { + m_packets.Remove(sequenceNumber); + + // Update stats + Interlocked.Add(ref removedPacket.Client.UnackedBytes, -removedPacket.Buffer.DataLength); + } + } + } + /// /// Returns a list of all of the packets with a TickCount older than /// the specified timeout From 13d6e05d5a346dcb9dbaec29af37ee63be790acb Mon Sep 17 00:00:00 2001 From: Melanie Date: Thu, 21 Apr 2011 23:03:38 +0100 Subject: [PATCH 08/45] Implement agent limits --- OpenSim/Region/Framework/Scenes/Scene.cs | 20 +++++++++++++++++++ OpenSim/Region/Framework/Scenes/SceneGraph.cs | 5 +++++ 2 files changed, 25 insertions(+) diff --git a/OpenSim/Region/Framework/Scenes/Scene.cs b/OpenSim/Region/Framework/Scenes/Scene.cs index fdd5205de8..01de82451f 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.cs @@ -3665,6 +3665,15 @@ namespace OpenSim.Region.Framework.Scenes return false; } + int num = m_sceneGraph.GetNumberOfScenePresences(); + + if (num >= RegionInfo.RegionSettings.AgentLimit) + { + if (!Permissions.IsAdministrator(cAgentData.AgentID)) + return false; + } + + ScenePresence childAgentUpdate = WaitGetScenePresence(cAgentData.AgentID); if (childAgentUpdate != null) @@ -4966,6 +4975,17 @@ namespace OpenSim.Region.Framework.Scenes // child agent creation, thereby emulating the SL behavior. public bool QueryAccess(UUID agentID, Vector3 position, out string reason) { + int num = m_sceneGraph.GetNumberOfScenePresences(); + + if (num >= RegionInfo.RegionSettings.AgentLimit) + { + if (!Permissions.IsAdministrator(agentID)) + { + reason = "The region is full"; + return false; + } + } + reason = String.Empty; return true; } diff --git a/OpenSim/Region/Framework/Scenes/SceneGraph.cs b/OpenSim/Region/Framework/Scenes/SceneGraph.cs index 72f040252f..fc31b65ec6 100644 --- a/OpenSim/Region/Framework/Scenes/SceneGraph.cs +++ b/OpenSim/Region/Framework/Scenes/SceneGraph.cs @@ -800,6 +800,11 @@ namespace OpenSim.Region.Framework.Scenes return m_scenePresenceArray; } + public int GetNumberOfScenePresences() + { + return m_scenePresenceArray.Count; + } + /// /// Request a scene presence by UUID. Fast, indexed lookup. /// From 7f28dd4b3195e020eed51fa1229083123935125b Mon Sep 17 00:00:00 2001 From: Dan Lake Date: Thu, 21 Apr 2011 15:40:32 -0700 Subject: [PATCH 09/45] Refactor UnackedPacketCollection so ProcessQueues will handle Adds, Acks, and Removes in that order. --- .../ClientStack/LindenUDP/LLUDPServer.cs | 4 +- .../LindenUDP/UnackedPacketCollection.cs | 56 +++++++++++-------- 2 files changed, 34 insertions(+), 26 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs index bd58ddc9e0..aff90c5259 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPServer.cs @@ -672,7 +672,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (packet.Header.AppendedAcks && packet.Header.AckList != null) { for (int i = 0; i < packet.Header.AckList.Length; i++) - udpClient.NeedAcks.Remove(packet.Header.AckList[i], now, packet.Header.Resent); + udpClient.NeedAcks.Acknowledge(packet.Header.AckList[i], now, packet.Header.Resent); } // Handle PacketAck packets @@ -681,7 +681,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP PacketAckPacket ackPacket = (PacketAckPacket)packet; for (int i = 0; i < ackPacket.Packets.Length; i++) - udpClient.NeedAcks.Remove(ackPacket.Packets[i].ID, now, packet.Header.Resent); + udpClient.NeedAcks.Acknowledge(ackPacket.Packets[i].ID, now, packet.Header.Resent); // We don't need to do anything else with PacketAck packets return; diff --git a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs index 90a87faec4..793aefe6e5 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/UnackedPacketCollection.cs @@ -65,7 +65,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Holds packets that need to be added to the unacknowledged list private LocklessQueue m_pendingAdds = new LocklessQueue(); /// Holds information about pending acknowledgements - private LocklessQueue m_pendingRemoves = new LocklessQueue(); + private LocklessQueue m_pendingAcknowledgements = new LocklessQueue(); + /// Holds information about pending removals + private LocklessQueue m_pendingRemoves = new LocklessQueue(); /// /// Add an unacked packet to the collection @@ -92,9 +94,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Current value of Environment.TickCount /// This does not immediately acknowledge the packet, it only /// queues the ack so it can be handled in a thread-safe way later - public void Remove(uint sequenceNumber, int currentTime, bool fromResend) + public void Acknowledge(uint sequenceNumber, int currentTime, bool fromResend) { - m_pendingRemoves.Enqueue(new PendingAck(sequenceNumber, currentTime, fromResend)); + m_pendingAcknowledgements.Enqueue(new PendingAck(sequenceNumber, currentTime, fromResend)); } /// @@ -105,21 +107,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// /// Sequence number of the packet to /// acknowledge - /// The packet is removed from the collection immediately. - /// This function is not threadsafe. It must be called by the thread calling GetExpiredPackets. + /// The does not immediately remove the packet, it only queues the removal + /// so it can be handled in a thread safe way later public void Remove(uint sequenceNumber) { - OutgoingPacket removedPacket; - if (m_packets.TryGetValue(sequenceNumber, out removedPacket)) - { - if (removedPacket != null) - { - m_packets.Remove(sequenceNumber); - - // Update stats - Interlocked.Add(ref removedPacket.Client.UnackedBytes, -removedPacket.Buffer.DataLength); - } - } + m_pendingRemoves.Enqueue(sequenceNumber); } /// @@ -179,15 +171,15 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_packets[pendingAdd.SequenceNumber] = pendingAdd; // Process all the pending removes, including updating statistics and round-trip times - PendingAck pendingRemove; - OutgoingPacket ackedPacket; - while (m_pendingRemoves.TryDequeue(out pendingRemove)) + PendingAck pendingAcknowledgement; + while (m_pendingAcknowledgements.TryDequeue(out pendingAcknowledgement)) { - if (m_packets.TryGetValue(pendingRemove.SequenceNumber, out ackedPacket)) + OutgoingPacket ackedPacket; + if (m_packets.TryGetValue(pendingAcknowledgement.SequenceNumber, out ackedPacket)) { if (ackedPacket != null) { - m_packets.Remove(pendingRemove.SequenceNumber); + m_packets.Remove(pendingAcknowledgement.SequenceNumber); // As with other network applications, assume that an acknowledged packet is an // indication that the network can handle a little more load, speed up the transmission @@ -196,16 +188,32 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Update stats Interlocked.Add(ref ackedPacket.Client.UnackedBytes, -ackedPacket.Buffer.DataLength); - if (!pendingRemove.FromResend) + if (!pendingAcknowledgement.FromResend) { // Calculate the round-trip time for this packet and its ACK - int rtt = pendingRemove.RemoveTime - ackedPacket.TickCount; + int rtt = pendingAcknowledgement.RemoveTime - ackedPacket.TickCount; if (rtt > 0) ackedPacket.Client.UpdateRoundTrip(rtt); } } } } + + uint pendingRemove; + while(m_pendingRemoves.TryDequeue(out pendingRemove)) + { + OutgoingPacket removedPacket; + if (m_packets.TryGetValue(pendingRemove, out removedPacket)) + { + if (removedPacket != null) + { + m_packets.Remove(pendingRemove); + + // Update stats + Interlocked.Add(ref removedPacket.Client.UnackedBytes, -removedPacket.Buffer.DataLength); + } + } + } } } -} \ No newline at end of file +} From c5159ad8d00e5b2dc19eb100ec2e74a7a605e139 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Thu, 21 Apr 2011 15:40:38 -0700 Subject: [PATCH 10/45] Add some locking on the child list for the token bucket hiearchy. A few other cosmetic changes. --- .../ClientStack/LindenUDP/TokenBucket.cs | 37 +++++++++++-------- 1 file changed, 22 insertions(+), 15 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs index 4ee6d3a30a..8ce64d5236 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs @@ -29,6 +29,8 @@ using System; using System.Collections; using System.Collections.Generic; using System.Reflection; +using OpenSim.Framework; + using log4net; namespace OpenSim.Region.ClientStack.LindenUDP @@ -177,7 +179,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP RequestedDripRate = dripRate; // TotalDripRequest = dripRate; // this will be overwritten when a child node registers // MaxBurst = (Int64)((double)dripRate * m_quantumsPerBurst); - m_lastDrip = Environment.TickCount & Int32.MaxValue; + m_lastDrip = Util.EnvironmentTickCount(); } #endregion Constructor @@ -211,12 +213,15 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// public void RegisterRequest(TokenBucket child, Int64 request) { - m_children[child] = request; - // m_totalDripRequest = m_children.Values.Sum(); + lock (m_children) + { + m_children[child] = request; + // m_totalDripRequest = m_children.Values.Sum(); - m_totalDripRequest = 0; - foreach (KeyValuePair cref in m_children) - m_totalDripRequest += cref.Value; + m_totalDripRequest = 0; + foreach (KeyValuePair cref in m_children) + m_totalDripRequest += cref.Value; + } // Pass the new values up to the parent if (m_parent != null) @@ -229,12 +234,16 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// public void UnregisterRequest(TokenBucket child) { - m_children.Remove(child); - // m_totalDripRequest = m_children.Values.Sum(); + lock (m_children) + { + m_children.Remove(child); + // m_totalDripRequest = m_children.Values.Sum(); - m_totalDripRequest = 0; - foreach (KeyValuePair cref in m_children) - m_totalDripRequest += cref.Value; + m_totalDripRequest = 0; + foreach (KeyValuePair cref in m_children) + m_totalDripRequest += cref.Value; + } + // Pass the new values up to the parent if (m_parent != null) @@ -297,10 +306,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Determine the interval over which we are adding tokens, never add // more than a single quantum of tokens - Int32 now = Environment.TickCount & Int32.MaxValue; - Int32 deltaMS = Math.Min(now - m_lastDrip, m_ticksPerQuantum); - - m_lastDrip = now; + Int32 deltaMS = Math.Min(Util.EnvironmentTickCountSubtract(m_lastDrip), m_ticksPerQuantum); + m_lastDrip = Util.EnvironmentTickCount(); // This can be 0 in the very unusual case that the timer wrapped // It can be 0 if we try add tokens at a sub-tick rate From e0002f6b64e7aeb29de9f1db92ed39d3d1df578c Mon Sep 17 00:00:00 2001 From: BlueWall Date: Thu, 21 Apr 2011 19:04:41 -0400 Subject: [PATCH 11/45] some mesh config asthetics --- OpenSim/Region/CoreModules/Avatar/Assets/GetMeshModule.cs | 2 +- .../Avatar/Assets/NewFileAgentInventoryVariablePriceModule.cs | 2 +- bin/OpenSimDefaults.ini | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/OpenSim/Region/CoreModules/Avatar/Assets/GetMeshModule.cs b/OpenSim/Region/CoreModules/Avatar/Assets/GetMeshModule.cs index deec444e8b..fc1ddef38c 100644 --- a/OpenSim/Region/CoreModules/Avatar/Assets/GetMeshModule.cs +++ b/OpenSim/Region/CoreModules/Avatar/Assets/GetMeshModule.cs @@ -70,7 +70,7 @@ namespace OpenSim.Region.CoreModules.Avatar.Assets if (meshConfig == null) return; - m_enabled = meshConfig.GetBoolean("ColladaMesh", true); + m_enabled = meshConfig.GetBoolean("AllowMeshUpload", true); } public void AddRegion(Scene pScene) diff --git a/OpenSim/Region/CoreModules/Avatar/Assets/NewFileAgentInventoryVariablePriceModule.cs b/OpenSim/Region/CoreModules/Avatar/Assets/NewFileAgentInventoryVariablePriceModule.cs index d651cb26f8..3d4c7b7527 100644 --- a/OpenSim/Region/CoreModules/Avatar/Assets/NewFileAgentInventoryVariablePriceModule.cs +++ b/OpenSim/Region/CoreModules/Avatar/Assets/NewFileAgentInventoryVariablePriceModule.cs @@ -72,7 +72,7 @@ namespace OpenSim.Region.CoreModules.Avatar.Assets if (meshConfig == null) return; - m_enabled = meshConfig.GetBoolean("ColladaMesh", true); + m_enabled = meshConfig.GetBoolean("AllowMeshUpload", true); } public void AddRegion(Scene pScene) diff --git a/bin/OpenSimDefaults.ini b/bin/OpenSimDefaults.ini index b134b7ee36..7ea98d49e8 100644 --- a/bin/OpenSimDefaults.ini +++ b/bin/OpenSimDefaults.ini @@ -453,7 +453,7 @@ [Mesh] ; enable / disable Collada mesh support ; default is true - ; ColladaMesh = true + ; AllowMeshUpload = true ; if you use Meshmerizer and want collisions for meshies, setting this to true ; will cause OpenSim to attempt to decode meshies assets, extract the physics From 3534f4492ae747baff492f4bc10bf06994ee1bc6 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Fri, 22 Apr 2011 14:01:12 -0700 Subject: [PATCH 12/45] Various clean ups. Removed some debugging code. Added a new "show pqueues" command to look at the entity update priority queue. Added a "name" parameter to show queues, show pqueues and show throttles to look at data for a specific user. --- OpenSim/Framework/PriorityQueue.cs | 7 +- .../ClientStack/LindenUDP/LLClientView.cs | 5 + .../ClientStack/LindenUDP/LLUDPClient.cs | 24 +--- .../Agent/UDP/Linden/LindenUDPInfoModule.cs | 103 +++++++++++++++++- 4 files changed, 110 insertions(+), 29 deletions(-) diff --git a/OpenSim/Framework/PriorityQueue.cs b/OpenSim/Framework/PriorityQueue.cs index eec2a9268e..ea718c4e7c 100644 --- a/OpenSim/Framework/PriorityQueue.cs +++ b/OpenSim/Framework/PriorityQueue.cs @@ -174,14 +174,13 @@ namespace OpenSim.Framework } } + /// + /// public override string ToString() { string s = ""; for (int i = 0; i < NumberOfQueues; i++) - { - if (s != "") s += ","; - s += m_heaps[i].Count.ToString(); - } + s += String.Format("{0,7} ",m_heaps[i].Count); return s; } diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 6129e10661..32a075a79c 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -384,6 +384,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP public ulong ActiveGroupPowers { get { return m_activeGroupPowers; } } public bool IsGroupMember(UUID groupID) { return m_groupPowers.ContainsKey(groupID); } + /// + /// Entity update queues + /// + public PriorityQueue EntityUpdateQueue { get { return m_entityUpdates; } } + /// /// First name of the agent/avatar represented by the client /// diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs index 20bfec8f92..103ec66590 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs @@ -228,26 +228,6 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// Information about the client connection public ClientInfo GetClientInfo() { -/// - TokenBucket tb; - - tb = m_throttleClient.Parent; - m_log.WarnFormat("[TOKENS] {3}: Actual={0},Request={1},TotalRequest={2}",tb.DripRate,tb.RequestedDripRate,tb.TotalDripRequest,"ROOT"); - - tb = m_throttleClient; - m_log.WarnFormat("[TOKENS] {3}: Actual={0},Request={1},TotalRequest={2}",tb.DripRate,tb.RequestedDripRate,tb.TotalDripRequest," CLIENT"); - - tb = m_throttleCategory; - m_log.WarnFormat("[TOKENS] {3}: Actual={0},Request={1},TotalRequest={2}",tb.DripRate,tb.RequestedDripRate,tb.TotalDripRequest," CATEGORY"); - - for (int i = 0; i < THROTTLE_CATEGORY_COUNT; i++) - { - tb = m_throttleCategories[i]; - m_log.WarnFormat("[TOKENS] {4} <{0}:{1}>: Actual={2},Requested={3}",AgentID,i,tb.DripRate,tb.RequestedDripRate," BUCKET"); - } - -/// - // TODO: This data structure is wrong in so many ways. Locking and copying the entire lists // of pending and needed ACKs for every client every time some method wants information about // this connection is a recipe for poor performance @@ -259,12 +239,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP info.landThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Land].DripRate; info.windThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Wind].DripRate; info.cloudThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Cloud].DripRate; - // info.taskThrottle = m_throttleCategories[(int)ThrottleOutPacketType.State].DripRate + m_throttleCategories[(int)ThrottleOutPacketType.Task].DripRate; info.taskThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Task].DripRate; info.assetThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Asset].DripRate; info.textureThrottle = (int)m_throttleCategories[(int)ThrottleOutPacketType.Texture].DripRate; - info.totalThrottle = info.resendThrottle + info.landThrottle + info.windThrottle + info.cloudThrottle + - info.taskThrottle + info.assetThrottle + info.textureThrottle; + info.totalThrottle = (int)m_throttleCategory.DripRate; return info; } diff --git a/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs b/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs index 6a24cc1092..ddbc079579 100644 --- a/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs +++ b/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs @@ -81,6 +81,14 @@ namespace OpenSim.Region.CoreModules.UDP.Linden lock (m_scenes) m_scenes[scene.RegionInfo.RegionID] = scene; + scene.AddCommand( + this, "show pqueues", + "show pqueues [full]", + "Show priority queue data for each client", + "Without the 'full' option, only root agents are shown." + + " With the 'full' option child agents are also shown.", + ShowPQueuesReport); + scene.AddCommand( this, "show queues", "show queues [full]", @@ -119,6 +127,11 @@ namespace OpenSim.Region.CoreModules.UDP.Linden // m_log.DebugFormat("[LINDEN UDP INFO MODULE]: REGION {0} LOADED", scene.RegionInfo.RegionName); } + protected void ShowPQueuesReport(string module, string[] cmd) + { + MainConsole.Instance.Output(GetPQueuesReport(cmd)); + } + protected void ShowQueuesReport(string module, string[] cmd) { MainConsole.Instance.Output(GetQueuesReport(cmd)); @@ -155,6 +168,80 @@ namespace OpenSim.Region.CoreModules.UDP.Linden ""); } + + /// + /// Generate UDP Queue data report for each client + /// + /// + /// + protected string GetPQueuesReport(string[] showParams) + { + bool showChildren = false; + string pname = ""; + + if (showParams.Length > 2 && showParams[2] == "full") + showChildren = true; + else if (showParams.Length > 3) + pname = showParams[2] + " " + showParams[3]; + + StringBuilder report = new StringBuilder(); + + int columnPadding = 2; + int maxNameLength = 18; + int maxRegionNameLength = 14; + int maxTypeLength = 4; + int totalInfoFieldsLength = maxNameLength + columnPadding + maxRegionNameLength + columnPadding + maxTypeLength + columnPadding; + + report.Append(GetColumnEntry("User", maxNameLength, columnPadding)); + report.Append(GetColumnEntry("Region", maxRegionNameLength, columnPadding)); + report.Append(GetColumnEntry("Type", maxTypeLength, columnPadding)); + + report.AppendFormat( + "{0,7} {1,7} {2,7} {3,7} {4,7} {5,7} {6,7} {7,7} {8,7} {9,7} {10,7} {11,7}\n", + "Pri 0", + "Pri 1", + "Pri 2", + "Pri 3", + "Pri 4", + "Pri 5", + "Pri 6", + "Pri 7", + "Pri 8", + "Pri 9", + "Pri 10", + "Pri 11"); + + lock (m_scenes) + { + foreach (Scene scene in m_scenes.Values) + { + scene.ForEachClient( + delegate(IClientAPI client) + { + if (client is LLClientView) + { + bool isChild = scene.PresenceChildStatus(client.AgentId); + if (isChild && !showChildren) + return; + + string name = client.Name; + if (pname != "" && name != pname) + return; + + string regionName = scene.RegionInfo.RegionName; + + report.Append(GetColumnEntry(name, maxNameLength, columnPadding)); + report.Append(GetColumnEntry(regionName, maxRegionNameLength, columnPadding)); + report.Append(GetColumnEntry(isChild ? "Cd" : "Rt", maxTypeLength, columnPadding)); + report.AppendLine(((LLClientView)client).EntityUpdateQueue.ToString()); + } + }); + } + } + + return report.ToString(); + } + /// /// Generate UDP Queue data report for each client /// @@ -163,10 +250,13 @@ namespace OpenSim.Region.CoreModules.UDP.Linden protected string GetQueuesReport(string[] showParams) { bool showChildren = false; + string pname = ""; if (showParams.Length > 2 && showParams[2] == "full") showChildren = true; - + else if (showParams.Length > 3) + pname = showParams[2] + " " + showParams[3]; + StringBuilder report = new StringBuilder(); int columnPadding = 2; @@ -224,6 +314,9 @@ namespace OpenSim.Region.CoreModules.UDP.Linden return; string name = client.Name; + if (pname != "" && name != pname) + return; + string regionName = scene.RegionInfo.RegionName; report.Append(GetColumnEntry(name, maxNameLength, columnPadding)); @@ -249,10 +342,13 @@ namespace OpenSim.Region.CoreModules.UDP.Linden protected string GetThrottlesReport(string[] showParams) { bool showChildren = false; + string pname = ""; if (showParams.Length > 2 && showParams[2] == "full") showChildren = true; - + else if (showParams.Length > 3) + pname = showParams[2] + " " + showParams[3]; + StringBuilder report = new StringBuilder(); int columnPadding = 2; @@ -314,6 +410,9 @@ namespace OpenSim.Region.CoreModules.UDP.Linden return; string name = client.Name; + if (pname != "" && name != pname) + return; + string regionName = scene.RegionInfo.RegionName; LLUDPClient llUdpClient = llClient.UDPClient; From 08e58e7ca6602782e1727bed9ade89fa42e9b55b Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Fri, 22 Apr 2011 14:02:34 -0700 Subject: [PATCH 13/45] Set the initial rate for the adaptive throttle to 160Kpbs or about 15 packets per second. --- OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs index 8ce64d5236..677d3d1713 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs @@ -322,10 +322,12 @@ namespace OpenSim.Region.ClientStack.LindenUDP { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - // - // The minimum rate for flow control. - // - protected const Int64 m_minimumFlow = m_minimumDripRate * 10; + /// + /// The minimum rate for flow control. Minimum drip rate is one + /// packet per second. Open the throttle to 15 packets per second + /// or about 160kbps. + /// + protected const Int64 m_minimumFlow = m_minimumDripRate * 15; // // The maximum rate for flow control. Drip rate can never be From a3bd769cb33ee59b883998205454bb340d44cb9e Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Fri, 22 Apr 2011 14:55:23 -0700 Subject: [PATCH 14/45] Added a second immediate queue to be used for the BestAvatar policy and currently used for all of an avatars attachments by the other policies. Also changed the way items are pulled from the update queues to bias close objects even more. --- OpenSim/Framework/PriorityQueue.cs | 103 ++++++++++++++---- .../Region/Framework/Scenes/Prioritizer.cs | 34 +++++- 2 files changed, 114 insertions(+), 23 deletions(-) diff --git a/OpenSim/Framework/PriorityQueue.cs b/OpenSim/Framework/PriorityQueue.cs index ea718c4e7c..8eeafd1024 100644 --- a/OpenSim/Framework/PriorityQueue.cs +++ b/OpenSim/Framework/PriorityQueue.cs @@ -42,22 +42,40 @@ namespace OpenSim.Framework public delegate bool UpdatePriorityHandler(ref uint priority, ISceneEntity entity); - // Heap[0] for self updates - // Heap[1..12] for entity updates - + /// + /// Total number of queues (priorities) available + /// public const uint NumberOfQueues = 12; - public const uint ImmediateQueue = 0; + + /// + /// Number of queuest (priorities) that are processed immediately + /// [] m_heaps = new MinHeap[NumberOfQueues]; private Dictionary m_lookupTable; + + // internal state used to ensure the deqeues are spread across the priority + // queues "fairly". queuecounts is the amount to pull from each queue in + // each pass. weighted towards the higher priority queues private uint m_nextQueue = 0; + private uint m_countFromQueue = 0; + private uint[] m_queueCounts = { 8, 4, 4, 2, 2, 2, 2, 1, 1, 1, 1, 1 }; + + // next request is a counter of the number of updates queued, it provides + // a total ordering on the updates coming through the queue and is more + // lightweight (and more discriminating) than tick count private UInt64 m_nextRequest = 0; + /// + /// Lock for enqueue and dequeue operations on the priority queue + /// private object m_syncRoot = new object(); public object SyncRoot { get { return this.m_syncRoot; } } +#region constructor public PriorityQueue() : this(MinHeap.DEFAULT_CAPACITY) { } public PriorityQueue(int capacity) @@ -66,8 +84,16 @@ namespace OpenSim.Framework for (int i = 0; i < m_heaps.Length; ++i) m_heaps[i] = new MinHeap(capacity); - } + m_nextQueue = NumberOfImmediateQueues; + m_countFromQueue = m_queueCounts[m_nextQueue]; + } +#endregion Constructor + +#region PublicMethods + /// + /// Return the number of items in the queues + /// public int Count { get @@ -79,6 +105,9 @@ namespace OpenSim.Framework } } + /// + /// Enqueue an item into the specified priority queue + /// public bool Enqueue(uint pqueue, IEntityUpdate value) { LookupItem lookup; @@ -100,32 +129,62 @@ namespace OpenSim.Framework return true; } + /// + /// Remove an item from one of the queues. Specifically, it removes the + /// oldest item from the next queue in order to provide fair access to + /// all of the queues + /// public bool TryDequeue(out IEntityUpdate value, out Int32 timeinqueue) { // If there is anything in priority queue 0, return it first no // matter what else. Breaks fairness. But very useful. - if (m_heaps[ImmediateQueue].Count > 0) + for (int iq = 0; iq < NumberOfImmediateQueues; iq++) { - MinHeapItem item = m_heaps[ImmediateQueue].RemoveMin(); + if (m_heaps[iq].Count > 0) + { + MinHeapItem item = m_heaps[iq].RemoveMin(); + m_lookupTable.Remove(item.Value.Entity.LocalId); + timeinqueue = Util.EnvironmentTickCountSubtract(item.EntryTime); + value = item.Value; + + return true; + } + } + + // To get the fair queing, we cycle through each of the + // queues when finding an element to dequeue. + // We pull (NumberOfQueues - QueueIndex) items from each queue in order + // to give lower numbered queues a higher priority and higher percentage + // of the bandwidth. + + // Check for more items to be pulled from the current queue + if (m_heaps[m_nextQueue].Count > 0 && m_countFromQueue > 0) + { + m_countFromQueue--; + + MinHeapItem item = m_heaps[m_nextQueue].RemoveMin(); m_lookupTable.Remove(item.Value.Entity.LocalId); timeinqueue = Util.EnvironmentTickCountSubtract(item.EntryTime); value = item.Value; - + return true; } - - for (int i = 0; i < NumberOfQueues; ++i) + + // Find the next non-immediate queue with updates in it + for (int i = 1; i < NumberOfQueues; ++i) { - // To get the fair queing, we cycle through each of the - // queues when finding an element to dequeue, this code - // assumes that the distribution of updates in the queues - // is polynomial, probably quadractic (eg distance of PI * R^2) - uint h = (uint)((m_nextQueue + i) % NumberOfQueues); - if (m_heaps[h].Count > 0) - { - m_nextQueue = (uint)((h + 1) % NumberOfQueues); + m_nextQueue = (uint)((m_nextQueue + i) % NumberOfQueues); + m_countFromQueue = m_queueCounts[m_nextQueue]; - MinHeapItem item = m_heaps[h].RemoveMin(); + // if this is one of the immediate queues, just skip it + if (m_nextQueue < NumberOfImmediateQueues) + continue; + + if (m_heaps[m_nextQueue].Count > 0) + { + m_countFromQueue--; + + MinHeapItem item = m_heaps[m_nextQueue].RemoveMin(); m_lookupTable.Remove(item.Value.Entity.LocalId); timeinqueue = Util.EnvironmentTickCountSubtract(item.EntryTime); value = item.Value; @@ -139,6 +198,10 @@ namespace OpenSim.Framework return false; } + /// + /// Reapply the prioritization function to each of the updates currently + /// stored in the priority queues. + /// { diff --git a/OpenSim/Region/Framework/Scenes/Prioritizer.cs b/OpenSim/Region/Framework/Scenes/Prioritizer.cs index 2e80156572..a7637c041c 100644 --- a/OpenSim/Region/Framework/Scenes/Prioritizer.cs +++ b/OpenSim/Region/Framework/Scenes/Prioritizer.cs @@ -88,7 +88,7 @@ namespace OpenSim.Region.Framework.Scenes // If this is an update for our own avatar give it the highest priority if (client.AgentId == entity.UUID) - return PriorityQueue.ImmediateQueue; + return 0; uint priority; @@ -119,16 +119,40 @@ namespace OpenSim.Region.Framework.Scenes private uint GetPriorityByTime(IClientAPI client, ISceneEntity entity) { - return 1; + // And anything attached to this avatar gets top priority as well + if (entity is SceneObjectPart) + { + SceneObjectPart sop = (SceneObjectPart)entity; + if (sop.ParentGroup.RootPart.IsAttachment && client.AgentId == sop.ParentGroup.RootPart.AttachedAvatar) + return 1; + } + + return PriorityQueue.NumberOfImmediateQueues; // first queue past the immediate queues } private uint GetPriorityByDistance(IClientAPI client, ISceneEntity entity) { + // And anything attached to this avatar gets top priority as well + if (entity is SceneObjectPart) + { + SceneObjectPart sop = (SceneObjectPart)entity; + if (sop.ParentGroup.RootPart.IsAttachment && client.AgentId == sop.ParentGroup.RootPart.AttachedAvatar) + return 1; + } + return ComputeDistancePriority(client,entity,false); } private uint GetPriorityByFrontBack(IClientAPI client, ISceneEntity entity) { + // And anything attached to this avatar gets top priority as well + if (entity is SceneObjectPart) + { + SceneObjectPart sop = (SceneObjectPart)entity; + if (sop.ParentGroup.RootPart.IsAttachment && client.AgentId == sop.ParentGroup.RootPart.AttachedAvatar) + return 1; + } + return ComputeDistancePriority(client,entity,true); } @@ -197,8 +221,10 @@ namespace OpenSim.Region.Framework.Scenes // And convert the distance to a priority queue, this computation gives queues // at 10, 20, 40, 80, 160, 320, 640, and 1280m - uint pqueue = 1; - for (int i = 0; i < 8; i++) + uint pqueue = PriorityQueue.NumberOfImmediateQueues; + uint queues = PriorityQueue.NumberOfQueues - PriorityQueue.NumberOfImmediateQueues; + + for (int i = 0; i < queues - 1; i++) { if (distance < 10 * Math.Pow(2.0,i)) break; From d5256094d975583876766b630bcf0ede1d3dd3e4 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Sat, 23 Apr 2011 00:16:55 +0100 Subject: [PATCH 15/45] print invalid command message to the console, not the log --- OpenSim/Framework/Console/CommandConsole.cs | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/OpenSim/Framework/Console/CommandConsole.cs b/OpenSim/Framework/Console/CommandConsole.cs index 52bcd5599b..be36cf2c10 100644 --- a/OpenSim/Framework/Console/CommandConsole.cs +++ b/OpenSim/Framework/Console/CommandConsole.cs @@ -592,9 +592,7 @@ namespace OpenSim.Framework.Console string line = ReadLine(m_defaultPrompt + "# ", true, true); if (line != String.Empty) - { - m_log.Info("[CONSOLE] Invalid command"); - } + Output("Invalid command"); } public void RunCommand(string cmd) From 44e43d9d240735695668849fbd7aaafd339f262d Mon Sep 17 00:00:00 2001 From: "E. Allen Soard" Date: Fri, 15 Apr 2011 06:17:26 -0700 Subject: [PATCH 16/45] Added MaxAgents configuration option to RegionConfig.ini allowing region hosters to setup regions maintaining more control over system resources. --- OpenSim/Framework/RegionInfo.cs | 17 +++++++++++++++++ .../World/Estate/EstateManagementModule.cs | 5 ++++- bin/RegionConfig.ini.example | 1 + 3 files changed, 22 insertions(+), 1 deletion(-) diff --git a/OpenSim/Framework/RegionInfo.cs b/OpenSim/Framework/RegionInfo.cs index daf0a256d6..239ce3d067 100644 --- a/OpenSim/Framework/RegionInfo.cs +++ b/OpenSim/Framework/RegionInfo.cs @@ -369,6 +369,7 @@ namespace OpenSim.Framework private int m_physPrimMax = 0; private bool m_clampPrimSize = false; private int m_objectCapacity = 0; + private int m_agentCapacity = 0; private string m_regionType = String.Empty; private RegionLightShareData m_windlight = new RegionLightShareData(); protected uint m_httpPort; @@ -547,6 +548,11 @@ namespace OpenSim.Framework get { return m_objectCapacity; } } + public int AgentCapacity + { + get { return m_agentCapacity; } + } + public byte AccessLevel { get { return (byte)Util.ConvertMaturityToAccessLevel((uint)RegionSettings.Maturity); } @@ -821,6 +827,8 @@ namespace OpenSim.Framework m_objectCapacity = config.GetInt("MaxPrims", 15000); + m_agentCapacity = config.GetInt("MaxAgents", 100); + // Multi-tenancy // @@ -857,6 +865,9 @@ namespace OpenSim.Framework if (m_objectCapacity != 0) config.Set("MaxPrims", m_objectCapacity); + if (m_agentCapacity != 0) + config.Set("MaxAgents", m_agentCapacity); + if (ScopeID != UUID.Zero) config.Set("ScopeID", ScopeID.ToString()); @@ -943,6 +954,9 @@ namespace OpenSim.Framework configMember.addConfigurationOption("object_capacity", ConfigurationOption.ConfigurationTypes.TYPE_INT32, "Max objects this sim will hold", m_objectCapacity.ToString(), true); + configMember.addConfigurationOption("agent_capacity", ConfigurationOption.ConfigurationTypes.TYPE_INT32, + "Max avatars this sim will hold", m_agentCapacity.ToString(), true); + configMember.addConfigurationOption("scope_id", ConfigurationOption.ConfigurationTypes.TYPE_UUID, "Scope ID for this region", ScopeID.ToString(), true); @@ -1055,6 +1069,9 @@ namespace OpenSim.Framework case "object_capacity": m_objectCapacity = (int)configuration_result; break; + case "agent_capacity": + m_agentCapacity = (int)configuration_result; + break; case "scope_id": ScopeID = (UUID)configuration_result; break; diff --git a/OpenSim/Region/CoreModules/World/Estate/EstateManagementModule.cs b/OpenSim/Region/CoreModules/World/Estate/EstateManagementModule.cs index ab90e90b78..3aed6bac8d 100644 --- a/OpenSim/Region/CoreModules/World/Estate/EstateManagementModule.cs +++ b/OpenSim/Region/CoreModules/World/Estate/EstateManagementModule.cs @@ -125,7 +125,10 @@ namespace OpenSim.Region.CoreModules.World.Estate else Scene.RegionInfo.RegionSettings.AllowLandResell = true; - Scene.RegionInfo.RegionSettings.AgentLimit = (byte) maxAgents; + if((byte)maxAgents <= Scene.RegionInfo.AgentCapacity) + Scene.RegionInfo.RegionSettings.AgentLimit = (byte) maxAgents; + else + Scene.RegionInfo.RegionSettings.AgentLimit = Scene.RegionInfo.AgentCapacity; Scene.RegionInfo.RegionSettings.ObjectBonus = objectBonusFactor; diff --git a/bin/RegionConfig.ini.example b/bin/RegionConfig.ini.example index d45fe9de2e..ff00ddf022 100644 --- a/bin/RegionConfig.ini.example +++ b/bin/RegionConfig.ini.example @@ -28,6 +28,7 @@ ExternalHostName = "SYSTEMIP" ; PhysicalPrimMax = 10 ; ClampPrimSize = False ; MaxPrims = 15000 +; MaxAgents = 100 ; * ; * Multi-Tenancy. Only set if needed From e2c6ed236d45e91ffb354b1c59e9f5a5a5b7952d Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Sat, 23 Apr 2011 12:17:10 -0700 Subject: [PATCH 17/45] Fix a bug looping through the priority queues. This should fix the problem of not all prims being sent without reprioritization. --- OpenSim/Framework/PriorityQueue.cs | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/OpenSim/Framework/PriorityQueue.cs b/OpenSim/Framework/PriorityQueue.cs index 8eeafd1024..3e6fdaa7f9 100644 --- a/OpenSim/Framework/PriorityQueue.cs +++ b/OpenSim/Framework/PriorityQueue.cs @@ -101,6 +101,7 @@ namespace OpenSim.Framework int count = 0; for (int i = 0; i < m_heaps.Length; ++i) count += m_heaps[i].Count; + return count; } } @@ -171,9 +172,9 @@ namespace OpenSim.Framework } // Find the next non-immediate queue with updates in it - for (int i = 1; i < NumberOfQueues; ++i) + for (int i = 0; i < NumberOfQueues; ++i) { - m_nextQueue = (uint)((m_nextQueue + i) % NumberOfQueues); + m_nextQueue = (uint)((m_nextQueue + 1) % NumberOfQueues); m_countFromQueue = m_queueCounts[m_nextQueue]; // if this is one of the immediate queues, just skip it From e3c3d626fb00fbc85dfc10755be0ee140f0b5a6f Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Sun, 24 Apr 2011 16:14:24 -0700 Subject: [PATCH 18/45] Commented verbose debug message. --- OpenSim/Services/InventoryService/XInventoryService.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Services/InventoryService/XInventoryService.cs b/OpenSim/Services/InventoryService/XInventoryService.cs index e602412be1..0af35c84ac 100644 --- a/OpenSim/Services/InventoryService/XInventoryService.cs +++ b/OpenSim/Services/InventoryService/XInventoryService.cs @@ -244,7 +244,7 @@ namespace OpenSim.Services.InventoryService // connector. So we disregard the principal and look // by ID. // - m_log.DebugFormat("[XINVENTORY SERVICE]: Fetch contents for folder {0}", folderID.ToString()); + //m_log.DebugFormat("[XINVENTORY SERVICE]: Fetch contents for folder {0}", folderID.ToString()); InventoryCollection inventory = new InventoryCollection(); inventory.UserID = principalID; inventory.Folders = new List(); From 507f0e9ffdeae9c8941171f9128f2948873f6e38 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Sun, 24 Apr 2011 16:21:01 -0700 Subject: [PATCH 19/45] Upped VERSION_NUMBER in master to 0.7.2. --- OpenSim/Framework/Servers/VersionInfo.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Framework/Servers/VersionInfo.cs b/OpenSim/Framework/Servers/VersionInfo.cs index c9d4c93d49..53a3f17b86 100644 --- a/OpenSim/Framework/Servers/VersionInfo.cs +++ b/OpenSim/Framework/Servers/VersionInfo.cs @@ -29,7 +29,7 @@ namespace OpenSim { public class VersionInfo { - private const string VERSION_NUMBER = "0.7.1"; + private const string VERSION_NUMBER = "0.7.2"; private const Flavour VERSION_FLAVOUR = Flavour.Dev; public enum Flavour From 35c09cae8c572c3b0dcc4b680dfe908d1253e8a9 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Sun, 24 Apr 2011 19:06:48 -0700 Subject: [PATCH 20/45] Removed unused libraries Castle.* --- bin/Castle.Core.dll | Bin 73728 -> 0 bytes bin/Castle.DynamicProxy.dll | Bin 63488 -> 0 bytes bin/Castle.DynamicProxy2.dll | Bin 106496 -> 0 bytes 3 files changed, 0 insertions(+), 0 deletions(-) delete mode 100644 bin/Castle.Core.dll delete mode 100644 bin/Castle.DynamicProxy.dll delete mode 100644 bin/Castle.DynamicProxy2.dll diff --git a/bin/Castle.Core.dll b/bin/Castle.Core.dll deleted file mode 100644 index 2464e661609b339ea4d1bdee520f723bed10ab89..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 73728 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 diff --git a/bin/Castle.DynamicProxy.dll b/bin/Castle.DynamicProxy.dll deleted file mode 100644 index 85404951075a162b510e9c41e98d03865c17e4f7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 63488 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%0D#GyV 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 Date: Sun, 24 Apr 2011 19:14:39 -0700 Subject: [PATCH 21/45] Removed unused libraries: RAILS.dll and dependencies. --- bin/DotSets.dll | Bin 24576 -> 0 bytes bin/Mono.PEToolkit.dll | Bin 131072 -> 0 bytes bin/RAIL.dll | Bin 200704 -> 0 bytes prebuild.xml | 10 ---------- 4 files changed, 10 deletions(-) delete mode 100644 bin/DotSets.dll delete mode 100644 bin/Mono.PEToolkit.dll delete mode 100644 bin/RAIL.dll diff --git a/bin/DotSets.dll b/bin/DotSets.dll deleted file mode 100644 index 522565140999b575215d7614866299583cc8cfc0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 24576 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 diff --git a/bin/Mono.PEToolkit.dll b/bin/Mono.PEToolkit.dll deleted file mode 100644 index b4c3e70e67a5cdf15e088754a6fdfe448d2cf948..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 131072 zcmeFacbrs3_6J&hyQkyyFik^GU}k#A&Gf(kB7%T`SwKNR9AuR&fXyLceQ$G3d%W-LLzcHS9-dAuOD|m6 zxOm~JRjJjDinmg=MJ8h8>5q_n%n+)17|0W?HeEvhbzy=U(e`d0!%u9cFsacY;)G>ZZ zZQl51|IdeX&2jS?6!-c%7S6(w)A2!v~8wsZ=tN? z-z0R`u=LWyQYa|4!e){6MM1fp9o2qL{_KIDJ@B&!e)ho69{AY8mvSQrGicUEO!Ezv%#j*tV9H# zmb+stF2~&|0B>`-r4y+#pr!_^oq0{};U0A`)IfKnS1^r_!_z*JFi9Nb4P~Vu-a9bV zyt6OS6LLd7DL1UvCq;+#^hxPqQJ+*8)+kWFvN}Li% zqGLyGLz&tRHozAVcf7LIYA5^555Ei zC!(6w)(65nCNtTq1J~aCn1qj}kybmo?Yj_+*1+8%$j;g|Crgl>g)a@0?eKra^TDht<1!o`*437;;)qx{^ z(KWqN1aaV2`{nC`0_duD=wiC0E~uhUQVl4ms~QEPEh~ZJC9m!aw9TWAf<6Rer0$F) zx;NyDrSWSiYYeVUcL?rdaH3JHkxpfYI9k486eK!YEjZB^<%4c- zsvka^&`;=BL1}PkD19B8(bgXWWR0UI%RS8%eoqPm%q#skx7=T| zqP5%~|8B+n{Jhq3UD2;Q4>r=DXX4WjThRxM>J+>gL|S^C=~ysX1x!o4sb(;irG|of zt6GDI%#48}Uvnn@B-O&IH@Raiq4e|g+Hl}C?xEgw>yg`}3%RT2rbd8& zBqjx`X-C8Kq(-u?4O5t5f`d3#sS{(>tG2>l0s1TbHbhHQxYOIa)Zw*fU2v)s6nt_Fz_FtVkH`6kZyGv^#wUWOxUn6QpVi(=wP>f+-gH+^HoHs|%w(rpDmV&2RLMc#bTC zrf6BDtl1YSON|5nwI-$KlS9I zGd-*#+0$3GAe5}|RfGC{HK3Na$?fZ@5}32Sv%O8hT8v7W3o!Il2)cW!A9`c?vwA8^ zY{yd7Lm4AE&gP&T|7jbCdee8JvzpR>1=SYDs~5+UBYJAEVKqu7c0#j~(M+~Pk}!4i zr0Z!>uxZ%3N%&a57rV(GJ9>O`zQkEG#`o;npcNZ=sMWY0tQH>_Rl(#htVV^ZEj4bd zVX@>auo{0vf%y%KT4z@Wlj8t?{4w6Lo!f$ijrSsJffcV@Y!_Ob!6g`=VKwAhOxq(>uO*{9-TPgIGN!S-lbtmPRi0hwOevZfHT!6Px}TTx%cH?+!R`8fUO zW`)vm3vRbib4|pTisDn2si@zI_*1>HM0K!_31S57Xo4yN>21s|ZM0%Z1*>{H69%qd-TgaWV&?AZ|ut z3>E8G%m=Ze-5T^}dy#XATipjaJt3zToXM$SC{~7|GDE&DWQH7(QSMLu#lm&g;r_Z* zEutVCpuB+ejQ2=%zQwUcUc}RW8PvrRy)cFn)gaB(EH<=nbuq@VpmT_7o++O?8&YLY z)rM}vr4$PN0ts)^u*RKdP1$)p=L@XodbW#o%muV!9S4w$`D#8t_tBp+WeVra<;;^Z zxO~)wcedJDH(og(>ob?=oiwAt#tl}@{A%#ANt=g)8XRP`!y1pV`#>i9pa~RDK0goQ zr(Vkf7#A2+5f~_T_UfR0?i2z(7_xjFleASDw+=*=#cJFm4BBh%g^H|l2 zx#br2O03_oXn_{0n^C?v-eZ3CqA<2BRMq$+=HL0cfSFI(e0Pe|CgPkQQNS8}`U5NM zODKn4E8^*3Q<&>r(TYh1Q&Cy%iUVUSh6QUSwR~>WT%`?^%d`l}i90acQLr#O=p;B@ z+=E!;`hdXTHA$o)b4|1x<{{@b(Gh~P*F={Kwy%lm(hmr(cU~i9_p=EMR(p$z*$-=< zjjF8#CN%GcNwR&du@ackY?QzSrzXRxHJsWEr;p+E$#Cinr!K>( zH=OzmM;fBH)eWuKBFr6F9veRuo1$c$J3ZIU{hPzu6w*OGZAvR~VI*62gfM(ic3=Z})esw6uG2=BJ%mnN;Q0olX=LKg2w#>u)Y`|_W*bdl+^isj` zfHQ*oWUk2C*@O*~B}#Ewb~u{b+}|(Fi1Y}^l zL06>yi{mODJzOG!YOJ;7I{7RsSo9r(;l|FrMNe6F-T12GOq|KR1X@E#RC+8!&GCid!*UyE@K&o)nK}&wcVk z=10l5Lq8QqLmI;}85$SoNmEFcjg=UM+tgV<+<29aCRAQmtN)mZK9 zU>V#S{k7@O;ah7hdsqrMC)Gab?w71kUQ`H)G(X? z2W7VkX5Obx8A?_PttdW@n9D|oEnA!4eSLGfujt+}fqGr{ z-^CEBR#p3YSC{egz+uOZMC<*<*c{>w_?YVKNM-)ruH)I>)@TyYnxK!G~ zwd7jU4&m&a=!=Ws>elMkTKNwKGA7{?`BxB=*L2lHt}Aj?-GS>Uk4MCst*wMBe3K1V z;au+I$Vi2U^8)zF0uCB!_A+Y=*Ou!fXtSG{R|RMHzagJ9sO!?>1ZUTlLj^n6mdiA! ztF?t~-GJQ?ZcX5rk8RvgA^YQMtDWsL>ke*E{1%QYx=&(8H^&S8R5)J9EjeCTK*EX> zyB;$1Z(gKr^!G_`MXRl~bag>y?!;bnKKwme)a8EoY6#Wi=|&Xec+#9=)HpEJp?j;zjSRK1bV2&*=d_!#x}RV+H5!<*8H8>Frb|*LPOxHrQD-T$$0$aHYkC z;Yx1FaAg7MwY-gV(U`jvW_1~JnDG#E)#$7HQH(LCIqWMB?$GIf1#{iKDBN}U{SHF) zYMZ-1M)b3PwA>i?8q48!iOKXOc8W(l!So}5)>;b6i%zSZQe+b*H$5_U&WpDF!BTDS zq&af^xz)5?IF9SjgSu+}BSqWaC+BM5I*8BRb6j7*xq&8doEF=M&^e`!EQ;cbaG`4AJj@!{*nhr)4;g{57Mg;$IEusX+MH$%K0 z3-Mr1#x0o)!|wPqErG7$zC2WXmEFc%wk^J-c0DHAsn6+4YWqL<^377dyp=7GIrV+x zOU?O-IrY7weyquH7uLwU$it$;IkimY>KNx#x=7EdTkVw0sa88{I~&is`Hi3GyecRf zM<14IJ63!XM_7YpEDC2^#^R?%+mGhUSiA#*^;oRU@d#G8JdfN(6S|6@Z}Z|Oe=H`0 z&TVj_5#F%csoUIfr#hz#sVv`xxffPNefVW5AKuLd$h!2k@u6@sJ}j|^`gchm{sn?F zdnnm4-9t{!b@9DaUVQA$6W?UV-sys~d#F2$`Hnh`@2E_+Q)YFis{e!U{#(j-_tAVA zdoCQAb1l#uz4zQ5^9uhJ>}P)L%5UN&Y>o1~^Z?Z`#vJ4B6Et6syFMX%r#(mFhxK*C z^_dZC=QsSP$DJDu>T(UlMhETYnNZNQTXTwCJNQwe=%Oc${i|*o+UB?xpPFgWZ0^qV zV$PEJ&De%(@$ypb*dp!Vel%*@p*cU%f02?+dI&;uqx@mrNcP`6#6#Y^*`8@F`_Dux zh2*t1eH{u~Ybh)*CwFTKUl!TZ^e*KGY%ooK))_x&PO<*1E!p@-byIWu^Y%;&r9X8$ zSTesI+t8m4rP_g=wVBHXnsx{$GnbXvoAvLiS+X8H2En;=8TN>#mvgT=o}dXFt3FkD zUHB0kQEHD}Q(j~}pzaM39A;Rl0Wh-Y9-A56++(u=g?nsrOYX5*K)TIpXH#W$%G*>I zjlaRLu*>=6Nl4Q3=r9!HJgPZAF%Fwixa+CqDG1FShfh42!!Qk1s)}-V^SH zVjPdc$@E`|d2|m{C^{yk|6YdR-2Quo9O=KhoSpWoG(nHmcC4AWHg()9|5W-ZyEm*4 z=U(IPEjaHQm%8=7NFVM~bDrm{cB_zzFUG~9LJhd%y)}WqUgYtpnI2+#CJcnPiM)z$k>*4 z99*g$IIKWBYO$WIH0=;hMZChc>2TP_e+6R+4`&r`mdrTGP2~M zg~QOiqT$MnZiXwHSQxJ4mJC-GkPheLU5}$B)>^tIF9tUHb8lF&uZmr#q+oZK@%$0w z>V4u-D8_gePCvQHS=%b>g@FKcQaO>$v{r9nRv{ z-xMLYBF)1LteoeJ+>K&UKO9ra4}YgUa_(@P@q_04#D2M>RQo$Vh0x4SN}f*ojB40V z9yI|zjq$wiU39Q+ceIB+bIy@-2i7;=Ib=osb;F#4hOYYj_)=}fRg{^-er4LKImPC% zlS?-63*8)vyF@{I&OOLxS-u2+s7AT%G=d9Q%n@;rz3J^DieyFbHWmp zC|r>~Pjh4vCzpthasKN?hPwbO?N z^|9LNKZEMbsTCcVH~2UD>#fw;=;EUDHZ!_8Z?j7Z=WTLJ&f6?t=Iw_6twD^4LW9UH zX%Gt-gSKt0I1j$+a;^9l()8ZvQWWD_p*h9ojf+vD=)xca~bB z^9Bn^-{)?2iq0FqE@k)kVmGgyHyOKyleumvaqWC{$!7cj!MUUR#{v%CfuOGy4nuo$ zueth1a)CtXgNKu0y95omH99%~_S*QaFi`TXGU( z0W*nVn*U!K#Hc7Vh}@C}v4B0z8}~7q3fz}F(V#KkhZ4^nbRHDr8LkCOgwW*;|#)preOxv7~j*fJl_{y6c&xUdrSErckbYS zu9puP{|jeZ*2}+k<$syGdqHrn|8Wm3j}xQv|Jzyf>*8B`S~lw?b&A`p)D{?5bS`5? zH|H`MUO07-TXO1P0o`MI+GT6=dN7f*g6lmJVlMZ4S&zWtF6WIZ2-V;7dkV!kZwRL% z-b;@kk2-;a2mcDj&l9LnbncM;tA^m*{;MhA;AUoCnLZg#JdkrFggc`cSUm56o%!GQ z%fGP@?#0zHyE&R5*zRmx90;}tRzv!7!I?E!qQUWfKQ_f4Y32ZyZOQ8kIS_ghJ}DYD z%;;v=&^v`;LvG2iVF58sUI_EQG>Dy3Xb`z24PpUf5cU)7NOOR;4Qt*ra9fwLTMtos z?tdA@7`wvB%>5p9gbv-p*nJTdipH*7a^Nm^HZCQa<4%s0Y0Pl)BDF8A;&?%3C3~I) zGNIT`#}w`PF6OOQVP}_p-w(p{e&-z&W8Z7ePwaQzLgB9a9Zw--`(1h&clPyq?s#@& zXm9%r?m#=`e2$qy@}>qeX;4^RPVV+cu5kLCyH@A2xnX3;44%pIwm9zk`SC52hHRZi38k-$s0YxXw-PrEgd$kRS zohn>&SZ({`Xs4qEqOEq)n5+x(%A-^(vfM*mO?X8EUN_h=9NFHgk!9}KInAMTeawF6 zP!(TiHNovaXXq%m@2p`MWZ}TGF~868_NAzi?i1;FfeuB--U{Q*SN7Yl?m_>%)oHM+ z8g}tLH#DbtojW#i4c_8I=0e7iNvOknboSRdsRO_Z`P_+b(R92N53k~z zN;@mTUgK{b;lq10Qu6?g-2wHcgmtYzfjoC$F5g!ZH187gs2Q-UhW_GydYF;M;>RXkakb z@^Vvc`f{|J@9<#Hizuua`Ld2GDr*h6xw5VkSyd%v9amHqt_kvF-6^uFOUn9HQCW!D zJXu>sR!vD+=MjBPwgkQeu-Ygyq^vT!@Qh%DTb z&69PFmQ_(o)<28N!acJ*S@&pJJxa;KU6LYp;dVfttQWPc%2Kkts8u)@1`A|;u4P3^ z$?8#57S5>h>wX#6g z%UYIWAu5!WE-H)90Ojr{KGL!r3sIr0_M)=*98IpQZ?!DPLR2X0#Gy|8k30cVNmaNJWvXIv;S&b!RA+KAqCY6wdyl%-_ zUqTl0x+P1wx{fd8bx)R6LYCD{S)LNIJl&M#FCoj{O<9g5F>k)dd+xg_tFeS#jop+r zsf4Ua-ITR{Rdyqkvpy=EIj`D=m{_Ej+P)~yWl&|siO+cUj8!O-$?v8}tcFgJg)&*7 zn=)ZfX_+k0O_^97OUq<|Zpy?eSz0CwbW>*IPswC~ZpxhWQ!-hgn=;q`luQ=rrcB1C zBRmUZi3PeV6Y*DCCJJ;{CgQKOOcdy@OvGPlnJBOgGI_v$1Fwm2-**9SIPuD=LlR&t zvKf-Vttn$qjmV(Iw@*3$#2N9z0)Th@;p-ang^)<*@?f zi8+ub5b^}W-lev^k{FH08JVb>U$v;UX_!IP^IMzjoZ9(Si}B?%em@N7&8iRY=fzft zdHkeUce4+7JLA>7afpjg{i7;vPO)*ki3pNp4Q&2N=_bIb1ac=H*&CyAey^PRaoF(17duP%>; zlT&&}J-wr4x@y#$tcrLdy#8I@iaUzls>S?0hbsKAL&Otk!LQ83R&4i1q^$R_9{8ax zw_o0ye}%ucwH!Ad!vRaI7`C*aW$`L)tPz&*Q zH~v0EZ;Zh}J0E|3T<5pq?|1n78h^79W`D)sm^f;u9ESI?4aYZ_r2Sj~5QTEJJ_4WP@wY4frr~b}{&1IEEx_L)_&Xeb$KY>0 z{(gnOv+;K^{;tB`m`xy!`16Lx{J|tW;SARm9>bC>V1>t=VUqsvm|vP?Ae>yoT>K19 z8X>oe>56djV1A;B2l2C4I5~%(k#KS!e%6MQyYmz7-HD%#;pAw3_75kA@pF)I)nMbQ z1YNbVPswf=j~;bb!BbVXr`WxA7bhzz7hNFTiY|zyzbGy5@2I*h-D-!7*T_a&V(G)` zO4@ohW!J;j8rxPzv~4TH5%=s>Kl~;6FcQO3{DZI*|G<9|rnXKA(+^AQtfS7K@DKV) z{3D^cq`qVuaBlgN@@)GM8?p2&rRA;N26=Y>L2fKPyP>3gcr<65`frf&C1Ow7j+l$3 zpDHc;DfCHq`Vajk{fB;({)7F}fAEXwJe)eG#7ng&IX^cTF)>KMc=1_*zZm|Kbr^s- z0b(}_2E@`&mQ=v!MLaTt)?=W?g6@znGuMuR@IhsD)o{H?~{E%oWrb;#%0Q`&z3zMWwVaZzGKtiJF3rAarmwkRr+Ma=eV1?YCNo8o9Fw+tV-N(c9(~fTIe28(W9jw}>YW(8#jAoxTX z%WpRCYt4wcW#fg-DEEjLS1HwianuV>ugTBoVHYO9pcnO8^5c0?pOv}^o9VIFqT%tL z!Lr!-;hr9E;DAIU_ExcStL=N-(un#4(Lgd?jh~^3c(CtUh;`QE!;KDqb!&N~tdNb} zS_D6Qm|wd4Z2lx-G!Sb!G2)MSVu2K#0LyQo#mCd)@?cq2zQr!&MuUN9C^^lFhsqKS zXi0Hfy0E+pc8IowVlADKpu?71X$zjaQSs_vS#`cGWwfO{5DkyH1Vw892WD8Y5-qAN zOT@csZC#1HNSJL4$HpIkU+0RHJDPVJ8qhqWHdt1Z-?RXmRuPEyNY<578|gvOJz_1d zL@FF|@1Wdcsx3i#T6#_eN+yB-V0Mq|#x)U3SCzs5ce$3PgL4Io~c{*aH2l3gyK;QJmY= zh%D}(hEj3Sn{DhJ8$X|M(aX`ydu{jC=eMect%?PrRmr-NqN^ynitdX!hn^ z8uI1VQ*L!2T9e$Rc(aRhyK+Vi)z`#YZjMwt)IUh|tDqihb>2kSm-=f1(LTws)g@e3 za(qVmP-~x9%iWP$ht@3+ote+Dj^$|{NX>PDXnk_3)u+Vh@2aWA`#n-mTk2yi*G1|a zwmeK*)*~J{HshguTZYhmv;gQy!%$yinvcA1}{0 zWhhN)3`F}TXIkiyE~j}b9xRa?>B6MGG^uZ_yY{5j3Y?Alg4U z%BuN4DC|##{bMcnNBTJwK0zO3x}zdr>Go7QAP^ncWp`k1?$R?|wS6E>8OZJ!;4tMW zwtbz1MUQ+t#?p>Kf#~4mp2fqW_|jKuAVmh#iovm#KSTyOtayf2EMpH;<{L4AMkE5! zA<2=&)sf2lu4#%zD)f5s5jaihz>11$;5k=n1LcgsdXqPnP!L9m#B3nTN2IJx)-$$3q)Iz z^`#}ZP;v{N&W{XpNPY#?voYE$U;5sZK0FW|(N&CgF>jTgNk`C>5scB{4pUxZBgM77 z^Uc6JgS^p^f#`NW;abdJUGAYG+fnOwbnQro);FkCTwAbx-j7;G1)|%-wIwBQPs!WU zwWA!8-=gF-GUf{QxpOIcbRfFJ|7f4I11;Er`=rqh3*Ld2u_xK23I)4TT#$RCI|icTl5?XB%_TD< z<7n2nSj$zB9UW%9550PxDA>;~rAgxh(FvF*y12BMK$#ObPmFiS{4gW4V2`?-GIt6@ zCw4veOr*?-l(~~b=EtZw_Hw$f0y9CumK4V)-sq%2baHZ8@$R&+r|Y`QiZ?G_a^9Ux zJ156l&WcQO*okvbZ}XAnZF{h^V4HaaE!{Z~-6grBlubhl)47i&zRA-wF}jf!@QwLBWx)uHGMDw@hUIi5dV9ZN;K2cml< zW5pFk_Mpf;Vl8h)c6W&U3iZa8;WQ;)2eFO$Vml~y&p>pqF^?b!ic2ou^Z)O;#kF;{ zb{3DNu13&abjDt>@lCu?+0)^Ve`Z|VH{Tb>(-%_$(Wza!cq*k%rHiLHME;v1kC9Dj zzkHb|QRcKjbnoQO#hcq5Ze(xj-#gZFOJtfu|JTql7GE6EUEDw4gi~q4K7r`;F^{5) z|KE38aS6rS@xSTJE_PGX>5b{J@k7|h`#2o(Uv|hL9IpfNopA=8F(VM&H@RJLi{t%E zt@ry<<-W0&2O={Z`o3ca;QMzfUJ0E8^L3s>oihW`S;_gub#{ka+P+z|Z&s}3qR32# zeLv72Ya~Di07ATxTjonNS^@4{6=NoYe zjhGXN?w?#&+=xGR)>WiSlGOhQ9mT#x~pm30fFeeBpN zJbQ2*CwaLKX0wn7B^Qg{j=81+Z|c;+Jziv}Y&7*F<(7>%u91ml%U zqM=BrIUET^f{j1^h-Ze#4_B>#|H82qEl;C1pFSKW_4a|lQuTaVM6}!%7AfB$fb;wE zxzS*8{g7W^(qtSo( zR+Bv50hYoOKj-2h6M1Aw5@4KXGb91V`8GomxR2vegHg8;anN{^505icwJnE8UGS*G z5-R`_xGx$E_Ed?(@X05mF0h420t|dnOY4;c7#GK9$sI zinr4gZ>K5VE>H0yrwO+6{dCvjvoHMTZjko{2feX$@4k3Bi`V?HgSR?8S+IFIix)qZ zE7-i8#aoxYM6h{}mY2^%1odmP@VJwh*l05(foG-gtvpX+N8F8T8ja`i5c}Q^vRL^y zc3DY)agEK81Q@@y8Il0wTALvWF!1#~-3Up5@q3#g2{1Zsh9tniT&b%_0*vczh9tna z!DdJTj6c{6Nr3T3n;{7>ZnPPa00Z9z)LJCL=fl24Z1JBR^XvyZ=Gg^yn%=kEZ7a}= z9dxT*o*uQ+bfH~_-m}wmoSml6>@?kFr|Bs>O()rD`o~VwHFla_vD0*jou)7BG~HmQ z=>a<}jYpdqr*``>X4!OmJhJKbcx2Nhts~2wIt)%gc{bfH&!$US2aqnKW5-Uj<#w7a zx6^F7oo37JG+S<`*>XG0mfLBz+)lIQcA71>(`>n&X3KTjo?8yVWG{0|yeTueJS~$& zc5;dK<7AVaT=o;3om?&wY)>u?>6ZoDFX{8fgXuo~DIsoNl8JSDAybBneIP87|;}+vdd5zt~k{i?CYffLA(=R=FkjUw8a|WbO(wu=dXHfbP%^7TS z66xaNZaVE z)4{gUSFb0ijXn%_h8f=quV*Bru_!si>8}!-HP4Zp3?8iJh`AF&*mypJff=`KTgm!%`H@TN^HITMp*> zn_$m6l31db2UE|@8QgITl9?Unj{1>TOy-B8TGJY9ElcY4JyD68_8cX(4mR0|C~`28 z$&}H{1%MT0NsVuxwj$JbRk$Qmcm0k+!V2xJz=td@Hr9T-`)o(^#MNl zNFMjiw~>i|lZ5^>sfgcEp>7F0-uE`ww8#6-K4f`Y20jf)qqT9sPexM*o( z`p8v{%U3n-I(=5-%G8pj!@}X<03&(}TqI8+4SWix)1E{s)5iED!(Wl%OY8LB40}A) zpZ%O(h@Q|d{lx}NH&zc~Z&msNy?_7kxpmr15*&ESlN_!c!|j5@$2j2Rd{Oh9S*jxbWJt4WHP&N34ON z1RfJZDYks*T&taqXoGw3k}&&na}#{5UqNPh_4Q^N`J!KYDr;2X)u&Y@PJP4YOJL)R zCM(@$zgm2#Zu1bdw=aJAWjMYa9YWtU!!~^X8@2@)n zA6Sy-xO_Hw%fx^774*PF#(!Fh&c;QwGe^Qo=ksb>8*Ak8vKoDy>H8F(%??U@S&7q` zmpBEqd5xn8inF!tKay~QbB2Tjs}{FdM7JSuy&V$oSuJ){O-7Ld5stD`c_lzUy4CIQ zC*~mdkxp}`+ELY=s%MwwyUH!@(N}yi`LNpQ19xgPO%m(fssAwTPHfE=9fRr`vvj6540m8?cgBB$wMf8QhqP;w8d&?z8Pq=K9zvd zpf{8p(AFPIQ@P)C$&RwbaR~ZT1CaL17gBsU7{Wt^&owdb(FJ@z0DQ+{@?DhF-p~NvZ`>y47>EV zqVzb&mP8GOUDZ_w{LmxlwPxYHTzi!Jd{xi3?lF_pKhwLXnG;xc*iV$*%ak33vMnfE zYvj&0a%)5`s9&T*6?FKjg$oL8Q?O3D73&ZGKh>Wq^%tQ22-I&d`u8*X>qQf2j{Z8~ zf`Ut1%1M2FgbNBT?Jg(v)e09Bwoc3F8u|Z$uF>lrN9#mLNMT zoRcNU&Z@}C5@cuLt3D1h1ld`YIaz}2tVm9lAUmsPPL?1$E1HufXdAM6<6ZnYEu^m@-MfkP?JXn@ z-_rYPj^p>j&d?mk?}a_0IgZ~8D@%$T`}e|d3_4kJ%rU4B$DkdWV~#<6@mTs!&9Q&` z8^@sEXpY?g9E0xAERyXe@EBBc>_*@i^k&U5$Dlsh`Nk{duo7QQ)ZH&XRK~uS_xH2o z`HLp)?DdWQTKpEIByd*A{@&=vo|~P`-*;?hPjB?&VwT;!5#J@W+S$oS;nIX%j1&&c z*uhBQtdiY}6wWHyxk%xxl3j}w&MMilnIB6smj*afO@Aq^u$_U!RF2mauJmxUYCEJh z9UV%4*zSOt9LXual(JpYkRBqX9Q_IzITr*)uLtl|di#37Ud8Rt?BV72T;%0?0&Bkf z!>&i}S8@$Uns5^Z)4{+V&%$+= z{)gi`nqc1(p$VxD7y~{{u+ucbKH{SZcA6$g+Snl{ua-8!q(!@=o9w%M)Up9PZ@@cqcA6UuJIx)1o#qz8PIJ$I^sbC8ZW54|c;eD;r@7YKX)f+a zi(angcACq#U54wmo#p~K%CJHc0ADycA9Rm({zI! zqjZCvrW@=u-C(Eb20KkR*lD^Ug<~PcH{D>%ryJ}v-C(Eb2D>iZV5jK@yA0i6r|AYe zO*hzSy1`D<4U*1`2e!`K^=9K7cfIWq!*-eyY-W)XY_Tc920Kc~m^upUpp2;jnKAWO z858#XwCc30g<~c=rX~o^j;Uh>XUEi^1ZTz+*1?Z7$8kHb?{LbDJ8lQg)f~s|z;iUm zaXau4&2iigEE^$m9Jd1}YL0C_*1=;n$8G@D!OfauHv#M5zcj~g1lGarMv7#|?Z6{6 z$8kIGI?d^yb7QZktd+d6x0Vs?IQ_@ny|oNvCkyuo7|%`?9=j-$C8*X)cqP%tO86v! zC5n+MKc6AD{urxfqQRQVXf+56>skh^Q!tlU2JVIp*vDe0f_0X$Dc#aZr2j$Yfd5WiIamZ6YzuXs2 z)>^6cC>iy56F5S|SD_QX#D}-x+14PxtsRcFyy1_QCCj}~5wv8zor(AR;PI`pSj)#v zmcGBm$ID_`f+ukTH7~MofgeJrh{~aUcUhtWXpr7tTIw;_gr{TpPKanY67Dm1H$OO+ znm?R>%6n*Vnj-I>!6R^LJ^se<5l$=Q+q2v+#|Ze*l394w0}nr~a9}SU_sM}m)Dwt! zkds_t@x?q-XrG)dAW;_a&aMq6XIN2R#5a}CugN(B626F^Iisy;AQG6y=hx)S0SP>0 zwqHXqS!eNSwrP0&tX$3?kO)P}nPXMq1|_e9Zh$GH>FZ{{J9Q#FG1Q7Rxl<=G6?Lah z=4UAVHw+AY14d^Btx&Ao9UqlC1^H}6)E~ol^bf@Mx@>uaKEIl_i2b zgNf6Sz}NLtr<1|tcQ`L51`~nOZba*%!BJd@`1m0$kKY;Hq;N?qfW5j>v@{yPM zn?@cNN94ui#<1M#l|u&%82RL_UP+9`@4+)Xp{ z%D7oEpIJucPNt;0RktI9=W6_fHgO$Yqwh-F_n4P%$Df2{Q?lb(4r}={R>R~r!TSEl z&FpI6nZCsIq$F@L-4BfR0p#KghQQq+XdAKy=F}2o*BX?QCCJVioRcNU&PwED39_?> z|MHhyy!&qla{u}}2E7K0Io@qE+9_akq^}RCUnPdJgia z1E6!P=)@trYfvnhtU`pdgS@F`2waxJiFnY9i4T|cWT#LwIPz5o2qi=Am=rU{eQ7!~ zwG(4FyPcgt2z}hcak}C^dw<7%x7M;B9akH6Ci~D@%kFY!u4O0XFxWLthS`s@TOdmI zrtAk>tT-MvNUyq2^llwV&XgS~VX@G)LN5|}qtHi$ zz9;lYq1EG9euU5&LRSjCYTSwAyy`BYTgOrIY|-+iKySaR}CEICBvFBZxd@2QEa<$WB`<*Q)3D@;f?`0&qvSkCI~tL z=xSIo9f#-^T`TBnb)BG_1#MEd2)awqwRqPfr93XEQ*9CSil9HJCk4GP=qB}& zpl<}-s@@h9u-Ky8)yINr1l_IvAt)i}KJ}fT9Rxk7%HXm}wWpvhszT6gL64{^K??+J zRgHp{3wlxw5p;~8XVgeRrwDpZjT3a9pqJFHf_^LLHMO^(TLitO<_Nk^&^v0rpvMKh zrw$SHlAw>&5rWT>AgD&r_v#8kg9W)<*9saf zDCD|P&_04HT(=9FFDUA|U(hl^Rj#dqjuuq!dQQ;kg8I2$7j&thfv)!jT`Q=`^|_#1 zfvQ!D>uVRFx*w=ajc|Q0d5;1OR|^b!!5|lcyi&buP??}F3K z&`Lqa8+3%AvkW>?&}9Z4E9hE-It2aMpyLHSX3$B3UNh)aL7y6Qh9I9;x9=Q5wFX@v zXn;YN2pVnB<$|UebhV%Z4Z23qVFvx)Iv%mq0aT4RyFv0c06pkh;0n53n3FF83ae$V zIv}Ms0hIyub2A2S@ZRGMsN;!{_XJQG&;Y6T22fa?ON5di7&J&qekCO@aqTF1-%8#U zf-sN!Snnp+A(9ss^k>)6Qm+cAOx@u+PV$CJ-u*-{cyw0UjGJI<55qkjmRrgL#>H|ZWlBUi1tP&($3{TQIs4B6j6s7 zbbSVSy$|?;YR>wtzA)(XLO&Mj@{5a^gf|OQOl#Y1YX4| z_*LM$OzVQda@U;oH9{LfT`Cc*1RWae1v)ZV4a)Lv91$#Yxs^|7d1!^pqZWs50%b`& z)fWn)51$NqtkLT4QsxVx{|q(a8^7NQwaO_WR33!19_1^Lj+HaFPH0Pcf24OPAB?d% z5oKIzccJ@8`8nlH$ekW$-BwVSS}643@Jy6HJj`0Zl-vu$+19PCSYXXrf4tC>gq{v+ z$I00hOOSh+&})P)1hsQ-syGC>w+p>j=oX=m34I>aF8NBuVJP{o(9ebbAT-#6{2HMH zg^mhQs{p`?J|{t z8#86Heg9%;a_qF9qMOyV8+9 zTFRd&^gIWDcu)4%1ff%e&Jn65ZUFGnVmf+0&M#`5(vu*Dcb`d7Hb-BF z#!--C&p!*IUtykECNw4VXrW6h$zLz&c1fQm>ANI-o}{l9dY#Z)ggz+rX`ycj{aEO~ zg?f5XW?1Ox2-7i1_Y=BGIKw2plhD0{&Jy|yp-Y6W6uMeyyU-JbUMTeLo|JsGq;D1a zpwLHzz994)Df5A(zZCkt&;wv6epIWTE5hOR^!725KCXT)c0MNyJwxb4_5Cw*_(2W# zyXUMwSm=T%)60aWgsv3+Q9{=XJx};o3%y0?slvCytUEeFx=QF?y_lXQv|Z@kJ;`}c z(vJvzLFnuN`JV(y&xbVj@3rv5(1g$yp<{(k z61oTIX!Y&Ddp%x!v4!gq_a}YT5rcxNuUZ4@QYQ}LmH(MSFByb+%y@P4psKPt>+ccz zfY678J|XlOq0fWjiMl~a(6 z+*RShGqH~MNqDRgs$XJ3Syu0Ac<+(~=@mjZ3B69}Ekf@Qdauxjgg!0wMWL?>eMjgA zLO&7uxzK+I{YK~yLcK$%t5Rrhp>;wB2yGTRTIhJ8y9(V?=ro}-h3*gPR`Z4o!^=t* z57`cM#Sr$|YN6|do+k7np__!>DD-xrTZBF)^jV>A2>n>-mqNc6>T9CpJ|h@QeMii7 zyH!GH%LvBU7@@lfoju|%lsQo7!9vdx{zXEs7J8k?`IFGiLjNN4uR@;``tk_wEkBX+ z{}B4EP;2B=v@|f1mh>D+?R|s}7TO|oN1?lbG8!3&-ym(r!^q%O-q9+(-8GQ2aC=H% zth&_^l0J5OrcV@lhR}<*r>?7o#>(&Z+T-r}?eF#?Vud~@^kbnv3hh0bxkH4G7dlVq zTA^nP-6V9g(8q+nE%bY#;TiParm_1rEXbjLo zK_?>bP26=jQRn#|5Bv@BqyKFwdnxcS&<4r7P`wxU0#{!@2s%OPl__6emffOQ(yJ;3 zo$UV(2h=__T_cpNbNSWLCU11GpDUnFG-z^g5YV{>O+#K#U1`vM$P20)3_2*7bcNI% z1}zJY!Tphk4LUs7;wo298MHPy3g{(+P700%O7x?hC#!3MI{}?<(B|N-*w_6_(245q z;3)h|g{QwR`E+oKt3v%s&=&OqwD(ZIGkMnpR|3r*pi6EJ9_fmx&omNWN11_tTkxWq z4p$a^7dYO5`h`w$;h9+Zx1}r@I@84;$lyPE{(M)J29UzG*Je;l=+_R!_GR+QXmzdn zcgB`mU47Jmfg&?NTk6yZgJ?^gI>;c}Qm;c8ZQ3DW&OR5c{1;hgH{LOyG#Y7%t3^z=|QO3;Ps)ldj|w`EXIt3{0&!g?2~ z4@0$BKU0F}i56?5TBni!fb!AScIpy?4leJ98;MsNw7NVA^ngL@1#LCx-10GMJN1@9 zzbWrr{X0Cz8lxW3c|fnZ#;O+#a#xJPovqId>RB<*8ms>t=wvmzVv1{=dXTqfF@tf0k7pA9wuG4Wjlv)Tx5B_C3_ug0%KM)OQ9^`(A1- z9)Ys8Pf@=Xq_s~`zsaCZYl^yAmlW+&)vX3m`&4zeLDW7~{Z)|GK2<#?NNb;}4#T5H zO1&@fvbQ>hH|b4;?4vFgq_yv(uF9ZJYajJTT~f49S2r0%?bFrm22uNTb)O)ueY$#3 zkk&q3ePQ^0QI9TDca{LYl8SJK<)Dsj=|Y~ z0cxM8B7(H`d1`HG=r#p zzB>>1JtLZZlS84M5H~xNR1Jswc{Cp4C=HNsp*3Bcwek$ z8bs}j)m($9eX&|7NNZoLmI%_?7psR2V%wLh2PZ51)7lSK?+Mb{4_1HApib*x^{p-` zZC|E-Fo@chsj{6#igJfwJ`ILHEkk+13FA37xQ)<`Us89F%5$Z8P+Vg4kcR^Zv zTKzMFI<2&F?=E(VcAO>~MD1-VY!J1#sosLL_BK@~NNaCXXB)(}AE~PM$hQ3`HAaxu zew3P&L7mo7YPv2d+K*N<4Wjm=)m($9{b;pNkk)>*S|UhmKU&>o5VaqxdhMCjzDA7_ zq_wY6<1?t!TBD}wlA^s`?PCzNx2ydOqV{%mpdhWiT`du$wYRIM45Ic9)w)+!`#N=u zAgz6!TAx9k);e{TE-BiNQ|B5)?Z>H045Ieq)YXEt_T$v=1ZnNZsq!h*r~CbQb-p0o z@4r;H2-4brsqV_4PV1Lyt1cRtwVFPg83%sM9)4ovurY_S4mw22uOz>H>qP{d9G?Ag%p$b(J8k{dDzD zgV^t9s9)?&TeS8w)mlMX`pMOZ`re z)_#_H%OGk$M-A8~tNmQHt01lYTs19&I<0fnJY7<>pQjEoh}zFnOAVs-^VCW~TKjow zwIHqiJoT1A)P8|_4Ufm#ZNE@`CrE3*PL=MD3TT_jEm(X)jfteQkZC@Z^4Z26bAOsy>3W_RCa*LDYVk8fXx; zU#5l%(%LUm!v$&Wm#ISxqV~(x^}3#Dzd}7GNVok8^;`yZT34udbxHC3mFj(isQpUy zsX^3!rTUj3t^G>%ogl6KO0{4n+o#9-)#|WWL^|p=sttm)_KoWN4C=Hts!h72Xy2rM zYY??>Qa2bx?VHq}1!?V@)LnwK_DzaUl-t^`Q8xczBGv1uT|d&(%P?8`t`6{`?czDY$a{&zgJ%i(%L&!^&IAD?VYMIgF3BF z)gnmu`*muhLDYVo8fy@>U#BJu(%P?6y9(0UuTyIcV!z*@M(t01TKgZ=Y(ZN4AJi{0 zsMGp`I#idGw*OJBGKkv$sE#y<+W)9J1ZnMmRO#{bn`VAZou^O)!YsZ|2))@K0;MSxpn9wco5BHi+8)tUL#>q;C7I zYJec!_FGkR26bAus&Trcw0*PM$slUqtadYq+BeI)O8kIqv)*WgWT~f5)sdhJr+V50*8$|7Qs{I7%w%@5*1!?Vfs_PA+_Pf>B zx}JFcFRJ<%v_)(Gi)zfEPU|nKMUeLVJ!+&u)P9c|YY?^Hqb3W|+V4@j3ewu|QELsN z_WRV0x}IpiUp*yAYrkK;ltG=={ptfdG&wLhfR z3ewsiQnwgH?T@Grbv7o@dsRs9U2_N{7&LDargjS!@@ zZ&hOiY3*CpuMML1$5pRI)Ti72gc>DCYkxwG&!A502{l!h6wg1YrW-`UKB}MzQ>J@{i{aN+4 zLDc@N`bd!0{;c}DAg%pbHR<52_UF~bf^^$oP`3%v+FwxjW>BZ~f_hSy6zwmnXAGkD z7uCxKQTvPPZ9!W5i|Ru`TKkJ?!LqFOmsRy4*|xu;#t72d@g#W$by}~e>AIw7e^t#i zh}vIOa}A>QSJgs6TKlW&5J6h|tLia>*zd2amzHO>zoEVsq_w|+FJ~~1sMC5w^$?_^ z?oHLxAZmY8)fhzWZ>qk6wDvbuLXg(}rn=1_YJXdue^^%gJL(oeTKhZdt_Is9W{ay9ELDc@PdR>s#{;qmYkkS94U>ONMt3DVj> zR`+I5r}eRVQkN9%pQvXHqV`YJ%LY;VC+clMTKgyJLqS^mCu+&zS?!;ytB%OF{WEp1 zAg%o~^>7AtTA!&GbxG0wxq8JQYX4llZ4kA8u09f^wSTU@5Tvz#t`19QwSTF8*Ot}( zm3ml^zFYQ{dd}qWZrMN7VXIk^cgx5 zTY=UK;yt(XU0!{~*-RP+2cYWxn z7jk{>s26g5<){~OednlG?izJ;wtYQZ;|yZ^dboBHq}$iSwWrRL_EoypJL*MUr#R|G zU8gzfMP27R>h*Fx=BQWYde%{|%JqVyUX|-DN4;uS^)XpH`?%r;(at`ueuA`}eO%2t zPwcF7EpgO~yAE~Ki@Q>edU4k=j(Ux*8y)rfyEZ%O^>^LrsMp{1prhUZ*LRM3gI(@p zwbit9u*)Y%+d0@(sq@6ngzNul?@a)ss?NUgbI$AuTLK73qG6A+hagbakPIP`%_MAs z4MQ>{14$-kCO{M`2qG$P)!M49wraIi>#n$C#a-)$6%{M3P|>Q@iY>NSseZrbIp@yY znMn}ee*OQy?|UKK-<)ST`*WVN+&g#X8cnj#k+Q-TZ=|fY#TzM?+Tx9pKeWXgE3dZ2 z8!NB1#TzScw#6GK_uAr3kbknpn;`#ei#I_Yw8hJpqt1%=!z4MuP*lH3auQRzev{-Z zE#vx4mTPVCrpj||@utd+ws=$JR$IJj@+n)qneqi&yqWS9TfCWaw=Lc*+3W0hISXY! zLs2;knTt`V2BQqIy8`=x4GY$&Q7RoDa@fOPKZ1EP!-L`m3g3tBcy)4cFAy{wg!HO2OFww!4ws^8hNkSSfivt_9*-a5I- z7Vlhnu`S-Y@)BFTbLCaGcZ1FCU&)MQ# zD#ZqYe|lYTx%3!{%6Yj=XG)jza@j}AxST(frM7ri%4%D@E9F94yes80TfD2}RknCn z%WG}%u9i30;$1Cov&Fkc?zhFeR{q%*?^^k(E#9^AkS*SIa@=|G`rRNW8H(z6gPh8g zuHOx^NXxi>H_COkcsI$7ws<$m3z*WYyqhGYj~47M`xZNHj#24-=h>*fklhlK>b&6kNlOq(NI(RRyuwqUpG`SWVgx>47C8VTjhX_l;37=5VG6kC__adyG@>9 zsEv@_E?W(CIb^rX>kV}SWOvAIhPn%~JLLO@dK9ud%qE@RJCNNaiwyM% zWOvEchWZk+yX96xr5{)6xLZDMsDa1rk@v{mhRTKP9@+hTo#sTy?v+Cgbt+`{%EgAN zg>0K_HB>8P+vG0{wE?pGkT^^iRxA2QUBAloi?8tNC2ZI`YsI?ekadsJo_>S@Rxm8Ti%b;y1#8x8eG$bKz< zYN$^kdrWRKlcj9kQn+y@7$O_1BO+C4+`~1+u5*Izzn!+0*iF zLwyR_Gx9k@eFfPw@?VDPK46c0R`$7ArUqdslzR;I zHe@f!Lx%bbWG~5a|E1HE$M2CZ%ThySA7ANsS#C1a6v$qY*BPoDvRCA8L)AgHQyw%_ z7_yym)c19oXG8X?oNK5{AbVA=H`EP~{Z?LMs7E0Ct=wg(*C2aM9x&AVki8~zFVSg! z0og8DWT>=(m5yC zEAEzmH57Tp-SSJOHjAHSFLk^n`&`Op5x>l?bNo)8YN$IQ+av1@^&n)wmp{>z_lfLP zj(6nG4Ye~n4C+=x(NlAKI%{4gdae5*Oe6SGLdt_PaN;br+%WTyc2%m z_&~m9sF^3+;rK`%V(JP}a>D(NkL5M@aDLd8=ZgdK#uzmj)a^{^{63QpGIfQx=ESh` zGr4se$MYUIvBUXS`9nh;KJi>oKQ)v$XS4Hf@)w56&bbKG9fr!w87vRV`wdl;GaS?o zL)GS7;{06x%}~wI{9Jx%sLjy)TsrU5xnB#}-(@#L-3!^@WnV)*i`@SqvkkQux&K2B zH`Et7KXe|FV+_@O&`&^3))YU>_=UX2Q1mP#UT%6n=O^eC?n}9iDSZ-kSnkp?Zo$KH zkEZZ6=wNwR?l%-Yb$nQUVyLpgo1OoXpBpMT_##kW8R{I!zLM?-bp3t=*;lfsp>7{M zQhqJR8*0bkr=4F*{>fW-zLy5S0BW|8{ciAYokE>zsQrU?gQ_;vUk1P9lxmrw)DXP8 zK(!d^xFJtFm0E46p+jB(b*`cEAake-40S4G4t1%amP6)LR~c#zWKMOxp|(QiQnwiD zr;xeSJ%+jiGPiozP``o9t)4X0`;d9m3x@hTWFEE4Q0}2mJH2X;p^h8+0;u;4H5@XZ z`ovJDK;~0_H`E-+($rUmS_D~|@;s>9dL?A(s;8mOf-GJ2H`GOtbyGQp`YB}H)JcZA z3$pHNyrCX}th@S-p`L-Phnj7u*C6Ym<{4@)WIa{2p+17Fr&_8hw!vO%=|kG;$Obdi z(udcmGmo{^1+Q8Q$3b&94qzdmZdp(ww; zDy%85Z}{iVM2c$+!0IEVC8$9<9Vl!I0aN6O^LZ2?=C}?G39)MG~buiOm$;wvvo-h2Wn{k^;dc#FOP|r2kBlgF4OjPQ%KId8AECAz*{%`Qt`TbNvs!c4h{@t4HQi9} zjd;g-vO1M1JzJctYM8o0jK~eTPFCv;H9mKxYoxl(P}6e5u2Je^L(R#Jx<;!}&ruFn zh|_XATw~N#hFX&QJ=a*}d|t~|D{4uvCfPP~G2`l*4yb52kbu z%hYk2^7cM?k6fn47;5mzo1M$lR6~tA`65syhC1csRgQpKV5qW_!=UO7RS(&6b*7=t zhHSYy*HAw=d9bWgTMc#n$-_bY#87uZRVWj`0UsgUT}0O2``3NrvixtWnK2)RvKhWk^*R>I&o+ zQfCSVq-aLv0y7 z9Mon*T?JXYy4+B|f~;L#Z>a5%tyXs!>UqdktL=t*WAtSA8uh%P_K%(c>PT5$y8Z#VJ_m^}_ltXrwI^IzAke#J+4Rsb|XRC>Zx)idr zRiUAN3E4VTX{bjaTc^HjsGX3VqgEMeFJ$Ma4nrM)?0agnp}b>X*r?Y1HlD*K^|_|J z;qfEo`KteG@f^-qZ!)EGIA6UJqh>(!fTlRj&Fb*3c$ya|*BeZE=jDx*Ths_cEoQ33 zP%D{gVoKL&i~7rMil^t5E$R?cmy7%JD&1SuxVK1lx!93+x_hg-IY!mFFHui1b(z?e z*W$ihUGqDw`EK57_m9;4w~5kT|4Ox-DXP(Md8JyVDeu>LgXL8!VkqB);h@%Oiu>Wm z>LRAD@NP^0p8F^22Qlh=_tolZru4e!8g-+gXx(#-dfR9g4BzVhsrtxJWuUHAhYi(x z!dCb7s>dEG=M~-)LEWHo3^fMSFVsXs6@a=)ooc9s6RvRIqE0td9jIH?N~Uys-mZRT zWb}Qt+tu|<>6W-%-OiNm+uPNBn&MON+ts6nqHjswuAVW}SrZ1!JJf52`aW9e4zvqoJbto1OQmOAWO({~}O7 zGt`Zc-LGyn)ON`3R}UHLi=1D%A5hO2s{5e3K<&~LkAMf&+&^&r*j^q|cW4TBHbXwF zb{H9bd-M@i`A%GOJ5~@<{BA1YGd{vU*ur*>cN$4^(+I!FOUCXai1S0%(MTL+(j<-* z?l+w@36)9b1m+B8j!MTnSj7nD=yXPNj^8%ct(@9joDPMh*v4p;T!E9`jrdR7q<_V$ zP?dO(IUfR*_>{v385gj=PQhN^vKf?)<{!s8cIoLXjn~d`qY)lE_;9X5TbLkXxSo>-Z^R*_$qIZ1K zZ;Wdly)wU?Q$G5hvadh&n&If<{_oXHEs5sX`Su)rGpXIW1Mx=O)G+Td>~VGNy4h9* znxpYuPC+Aa6y6O+>1+H4OOK4SJHS4LHIIc;+ri#<|96E`ttizpQ#mw3@vC#N2YQDt zjZ?=gwpiZnc0*Y6VZ%urr_lS*BIqbFhhvexGJL%c)orM)S97%WzRCJ;vHlr;yMM_BwUkmO~LgYTvKsP z!!;e(3|upD&B9fHYc{U$iN5qmA-=aR)ILFX={{uq3|Q&-g2Pf0-^ZB2IDl~w<4KI; z8K*E70Oxy71Los>%@$5Zn^1VMe1GavPk=Q;tkWSUOgqyv9@3A*g>u2Pjh+kTQs5L3 zn0AThGTAz<*m0RWXWBf+6mikCA9)IxUx2fqkHnAVP1CORESC39y8(Edyv0)@9-lT@ z+`=i}BI&(BQ^f9RI0s|S6!F=#M?H7TFQ+{YB&={u4B18J{& zPUF&^Cg?jJRL;rx<(R|1cfcPzeLrwy`bVCJkiw^&<3VudPXE&LxLhz@dZ`Tgg3#rC zTrQu!2k*h6`;*TjodMprxFkOQz8#f$oOhpmW4cp)$o$2e?-bE{#%AXfkv-!ad_!-< zj1uo+Ic~;Eymh74`zmm7dbxKo%2VT|(q7~wB>pKg);YeARFWy;v>B@$l>QYc&jUDX zBy@SH%v9@pr`?V;@13>{c$|Eg!%ul{;oNUQ?aC3R^luR>XT0IXH)B!v#j<_IKJ^f{ z$t{c<^ul>0hcl)Lw%?Es!$ydRdB7Q&P*S-v; zrWZRN2mcSglhkK3-ox$Qg%dso{%gi1T$e9>*K^yBSJ%w!okryxoHl^(8y6t`2aqP+ z{aq{zhObVec3ziOqCTCuA#ES-4sLYqlXNe0pQL-3`y}1F+$ZUtW<8O%SWTJr9Ps-yUQT;el+MDFT4E_)|q@FTBJs zU=~KX7xY+QF6)e9{c#wfUX1KHz{z4Ba2kG}#fz~!A6SH6v-0A%U21`)q8?Z&Lcki) z0$j@JEaPP<#eK)&RR}q9jCLN)7ix7Y~ggaiZw{>Qn3#BLwE_VxC-wi z^NMS@7T1Z35xx;}uee$K2zVRb3*Z%ZiEDw|An}R^#Vx??;tt>r{9>(FJO!RtJjX5Y zqIeYHo%n?tuh=D?1@6W#1$xCE@hWhycmsHVOL&m$a)|4)31ihOwn%E3t&EpSYK^Oy ze+_f4!<&6jHoRRArNZ0)P%4&glhl$AvUEF3cS!1s=UC@Oyip0|kxbAbosCg4GNA@C5Vc34u4o^w(-7>7+96c2XJk z;Pnltqmy#nkFR|rO(&K4V<+|I0Vnn5K_~U*At&|1VJG#1a8WNfUDOL+7uBwti+Z7# zi+Z7t>mMjXe;4&dmg_LWIWFpvp)Towb-ActM!Be8#<|?!3y$#&%+6&y_dJp)N>m%TEu1|q4 zx()(&y1oGJa(xBd?Q%HLo-QwNud6$7zbg~?fvX?zV^g?B(_Y`?%)<`@81>v)tvt9QSF!q3#-Bu6rSHl>2nxIQJRAeD^Zo zWOp5KntKIsmOBJ2ayL1Bq7lDl;S){n0${6q9x&>D1k#V)SuT$_;64#}&^;J<$UO{r zm{E9g5q5e=XBy)y&zlGrd430+>-jyf)Uy{@>3J7e<9Q#r$nzm^spn(hGS8>Lde2{h zjh@eeO`b!*R?nBfsOKx-T90&l#5#`?xZdLdZt|o7w|MX?d}6DIYI>=s7s5aE^aft# zISzP@X8`az=G^GXLilFSiNM=DgMoK>h5@&Ea)A$eM!J3CLGN*>^>#0{!w$??KJk>7 z+Tl5GE$~I}K;TYq4se(E3E*z;6Nt6P+pc_KulFqAe(xsW2i|AF|JeH?@PKzLG!J@D z1|DKO?0pq{;UjeV-T-=iZv(se_5yqP-UIgWeFW_9`xKbvI|$72eE}Tm`wE!rb9j7W zl+Ozs=j#s4_hkYn`}zT=`Lckse1m{RzTv>Rz84)nQR;gYSm}EMSmWCZT;v-G=~7=N zdViU39K!X!iNHo*2HwNk>dSJuMbtOb=@x5!vw`b;-|@J`df#;5Cf_XJ7T;X(xB3bZ zzSK7d_(NX_@G4&^@ETtQ@H$_WXM~8r!;BG2#Hs0HL^Wd_V>)AR#(|6z7^gB8GR^}Q z;#WH|#Jl2Z(cp35-BcN(4sUH~@N8tP!~3Z+#74%U?sNsIm!yR~2%qdh(jh%LRtBLL z^{Y&m{9?NAV=s!uJbrC2;Q5|si|1<3-JSM|vlDr+N#$i@mG7YrN~c zTfIN^{=&P%`=a-4@4MbVd;jT8_x1PX_@40n-6zsKX}M{m)AG`$rWK?Wr&Xk#o>rUI zlC~MYp?5{v&(eOGc1POwv^UeBzUx4-c z>4^DV%-zeeUSEL}R$|_6!Fs(7DXqqO{Vc54zlZhuc}R6L=JkuPUjIJU>pwsaS75#V z6U^#A!+QPa$nBR{um1}3`yE)X--Dd-UJvmw+V0m_uRnoOJcIT6Z_tJ>W9IuUO7w;( z6u%RG%#d@$A4Dn0b^EV<_Hi?t*jDx52P&u9!fg{_^-6(z*EwH4s@mc5~y_w({DRU=?;$jxbN@Ke=7Yj z@clFxj`(M{Vxa0yxp(VM`SxKP+@128z*xYz)m;kBIX$RU6+KA0h%wNEQfp$|-jn$6 zrxCu`^C!?h8cs$pDoLv5F}`lYW9@LP9qeQ3|80A4rF{Z#aC&d@A$Bi!EUdl#|KxTk z7&zY{L@{IeK$>?JG1f7*FrLM@k?~T-pD_NC@lM7ajBQ>@;d_jiF<#I35=-|m{+aPG zqc@B6`!HrRj$)k3Sj>1jVte4FtjMwLx@^=2H%IFfND zV>#mzMthq$PS}9;hOvd?)tHzPPI)`!d zILhS*jDK)b_#uzqEyQ)Kc`x&yVcgC55#yJPp7Er=VLY|)x;(Q4U=|G7+gb)1g zj6JchpkL!m2PXqOB{H$QP}pN+0JHF^u#4ymofCixJBWV3!Psdi>>>sNheNKggE$d7 zxj=>8!yw=&$Q5=F!+?2^OY9m5Ct{bOuzMgp1-c5mhS9)6>@pP21jhsY(8c{1=t}Gl zCL%l+dklp=!W7`C*k35@52gc8!`?z+Ur+$7Mj8@(gF=MoBMpW9K{0R{;wcfp&O%|o zQwD6nu0n}M#8x7NcnW);`M@xC6H2s-#lSPM8&xp7<(!f1vu?l#hXa-({lyFZOX(+K3J5z;yQ#P?HO`wo(0VUqbri2IYSi{p(@*SX0?gOUbjb*s+ zEZ+lW$Ugxy<%huD^3T8l@)O|kc(;}k1LbGHEcrJezB39Of;U?!?DM_=4#T^vV5{<9 zz)^Su6>Js1x2&+^a{%+D3pi1FfTu_waFR?1PM6(*Gw^;VC1%PD;4Il2I7jvc7R!FX zxpDxo1n*W-_&xD#V5K|}c$ypptdc{3i{voiV!YQ#iPPmtz$JKd5v){>1}>Llfpu~` zuwG68u9OpjtMHy6C7R?EV6&VGjL7N0sGJFGmj%Gpct?*C>!cru-$MX?PnH1BmGgk- z%QE0*SpmF2o(9|^tAXE_^MRMhg}_VcZj!i6E&*N%f2(jRT?_oN3;=&3>wwqEAn%lD;LGw7;4AVn;7+_VM2R=#kAQE=D}lS^ zkAZK=tAX#xp91&FYk~XZ^}s*k9UKa0-@gEUC~pFOByR!!8SlJM;xoL#0#AnEeH2Rk z4R_lV&cyEp{u6J0P~tG&;{YqgTNz-b>JgwrJqmQH$ABL7IMAz}1g5K}f!)-zz@F-P zU@!Fou(x^%*hjqr?5AD@_E)a~2ddYBS?W#TiRvw2j(Qt7MExE(RJ{Wnf$xJWk*nSX zj#TdfN2xyn$Ksxw664gLffLjxz-=2wQ2xx znHmVJQ`x|Jbt14q4FWc*A;48?7_do=0EX2`z*aR97*V5vQ8gC0MvVuqRTF^tjty{~ zngl#oO#!Y~Q-K@Rbl@g66L^6t0B%u*z>Ae1xK$McFHt4HOVvE!<*E$$LsbF1Qk@38 zN>u}|R`Y?^sD;35)nedvY6S_b_f#VQ}sP!dEN$o$v>M@YO1U@Iyc)9#-^=;g0|@BGej$A7y+@b%670 z#vSTx@E>PSAzqGX7T4uZ_RP_`13T{9QmP-c*+%{08H0bvZb10Wn7?`UUd0ftVxI zl?eYHsKg%>{TBHa=Z}p075$3udyIcl*Mt8)P>K)L4G4e0_>rPt zD*rPO^NYF(;ZJ~=U(_uKe+pFMfVvgo&wv=kihjNPZ;YR-JHa`~_;+vi_znF92_^J z*YPCyKE`g2r@>DLVh!ea7UAxUy&cbk(+8+TU&jl;;~Xym`#D|#9`AS+IMDGLaJb`j zXpR8F4>;aL_#~hbCp+Fkcq9-$!SOc2V;sLncr;LoagKKo9?Ll1u@9U)#(c-S;7?%u zj^jP>rvl+69DhQ1IuJ9R<3ogJ0+pEM_%pD;@d-Gy8H*gBf>Q{@Z%{ivL)gz)?D!iv zbAgx#9iJn7DiB`9@ehPcfmpvezCgGf2oK}<65&e5D#yRTIgPQ}@iq80j0+snfpr}a zzcS}=AiR)qvBL$<>5NMq9`Khip5gF;|6L$Fk|Q1AWkAfZj_wFAXKZow1gDuX?8pGW z6^ON;qc^a_(HFSE(GR%MF#vdhV<2#gBO7>`<3!*CjzPc&9YYZJA;w1>!@zkMi1F(f zf$(<5Upr0$=P@AGV~&vsKMutBb&N*%NyeugW5Ia}i1F(fkMJ{$&p9T5^E?ov*f9~| z7l0VWj!6i=1jHzIOhNb+AjYp_D#EWazUG(?&TkoaIc9?YI^&y;0`T7eV*EM^5#G)C zJBJ^fw}BYNj$(v=55y>Tlpy>L5Tn>J58-`4j9*6?!tXM^=coW@KjZt3)4=}|mhGEJpZKAV#rc3BsQNF^U~$ApAEV#;>Cm;m?7Xg&hH) z%UK6>JA*)vvjLdl3_&Loh~0{F6>zAt8JuB^Bb;GyhBM|m&jkM@#*xkl_$LGLM5wbJ z;ZclZoNK@t3&afT>_B)t5IY*@*$7Wyoaj6UoP3}Zlbq)wdE~v!^{ym1hBXPF$W)k6S4y0j+mVv!lCa(3^A#A{P9D9 zZK37u(I84tR2^*#2ATz8)Cb!@*94X~1+CB7fk;p+2sE_^IekbeS4ihaLrsWnidGd4 zBiCTGFs~xU~`})8mf!fvaT#?S&h_eoT~5|QQqFvM4!50DMUF} z1y@+XA|0F)3O3bOM%#$ZVUEO^8Llo5u`2^@fo8+yFfmJ7qQSNmfx2KxbF0n~9TRLb zITeOmB2g5XGz;4!(QtFDoW$ZB%K~kY#sJ4H3f47M2kY9~LeUP6(iDh9N&_9?c20rS z<772bt}kc`w*;$04W#a0jUFruw8p}wu*@fBWm~v4*cRcw$w*04*E-hTaaOFF zY_YZmt!@q0ks2wO^mT(SC?K}Kys$K^OIcoMv(fQO+hZg+SYJe6edrWKy^&AXX2H^zJv9J)b2!*YE1_^oAuc&Vk9g+)Y1^) zF~Zi4QKZdH+rBZlI02K>1h)$s-A3u4ZuOW%lUigMpN-PN1Ov2^F(#-<%LomXXoi|; zH$=^`o1%#rH%40|ZH}6eV33++nWT-1RgYy_U|AvzBasnPwluUj$vo{$vOhyxCaOv5 z4>D3s88g*{WvH6crWy;!%48XwmRMBG_7d3@)rmB$F@@1ZR5o0#Os1m=%*PYYlg0eWqAFihO%~ca3-}K)yMwCCK3x>FwFNr3E$TJ91n0+dT1c(>EG;aA z6R!%MSuZGnR@75`OQ4CoduT-nEn1CA!;qRpWvHbEb>dP-^a`jU7_Bvnp*d~g=BmIN zvTRX^-X@f>0#1Jk*5gzbp`0WvmI)c4#$4Zlm>4Bdy6A z>)SYH#flY?U{v6$H9^iE-Ojvf3J|9}(5yKeAdXco4pX`mz@TU~>&!7F6=HU%CD4Y( z<0Tz^7|uehPtXedfz}B{IbKb0ZPZAL+M8S9GE5(sl5t@;e=BcZ?{VBP>LDUj+7>D) z7rC8mtb!J75TS6~Kn3-t$?CDhvngOq>-F)N!f;cQ9<>qO+okYBXgZ`4@iELUp02pj2%7p-8D8gN{d? zrO!4{7FY|n(140Ih~~h`1!%+OPz!w^$za=RSW**~#c*h7IkAG*_AT0CA~ePtqwM}7 zY(o*T0^`3jjK+z!mDH~l4KOGAT;3F3PM@LX2Ckcx0M#n)Pl-a#5gLQGFqFxgPxI}0m4@rUuqGSb?aMo&Jo$0+%m%>* zKuDOsYqCA{sc3kp4Ai-Rq}Ha*{!t%lS0 zx75L50cpvF5M_bgQ3!=$yRwJLV@6uB!L%G4CNtyUc;1O^C?nYKM}sZ(5xuQsx2G+P zgFK;Yku8c=FmpCJG@Fb_`|jH&;@t=*%OSlr4#E~}I@Q55+k-7AcWj43YP4M7lK@eU z23f5S1Pb&jt1{Tu9Ey-jf~i26S2o-rJb817Hw~OKhxDG;mMe30srCMX)~wd%L6_KU zb7Ndv?D#4M%lmL$$n+Vnnb~3K_M*iQ7P;&?usvvn%hXw-$zVJ@$?2e8z-Wgur@f_) z#-`Pgu;vw^hIXCK7_*-U;WWkCT9C^xXle*!x^2YKil$&LWr3C9Hp@|#g<9go?DkMo zy*5dV#MM|$gj)<**xtko>DZ=2R8&_s1)?i3yfJ@OR41eg(w=T?SzFPj>q3Rm-Dnk- z)>yP%v9+dIa-zf(;#RQ&+M;cn6&ejqaK|kTdZA`YJv-cv#a}h=+qw55dJf_hR4FFQ z66VecwbYkm{lOCi52r#zjAFVb)Vp#n1eMYhOOFLw`RPuj5L+_^Vm3>I019Tkiv;z7 zPV%}{CuWXvM8aa3>!Y{HTFZnJcFVD|8y(=X#1Cs?rW!A8+){PvbvdmaFV~U1EAe5X zI%nF_=-n-S5Zmg#SRig5mZ+jSCPeF{%E5#Rbv+zV{DdXtc!kTvrzfVtO}JnYcNnZG zRIe8v93r2n`<5T#MKNdvZ0JDW_KoXjJC_Is4TSdO*B*!J&_s9-!kG@2`z@vh#9w0h|!ij zX`5)NC9#b*x1BX2jA{bU=p_smqnE(5a*v-sSuJ57TJ~Jc2(xEuSVk*x>>0g8rqSd{ zghq-{iqUqt&0<(iv|XMM*^*e3k}YvN%fz=>wk+(dBrPi=OU24dTU*>`n93xxO=_CW z9wkAMIhIb$)mSPqhS8vb$1cby#AqY8Jd+Vyq>-|vkiRnG7;WUnjTxz-;xb#K8c%0r zmRGaVjnPSRw4gqRPF9Qyl48IKn$lQlf|I0U^GlWzu~N0-xg@0t^|&lGUTPr_Azlvl z(yN5R45qgw4RxAsSQ& zS(a-L$8FN40K2nMWax=xr;)V1LXmA1p9!`JI>pw&vl|u2xc!j6%}`eJ(xRok8QU}* zoba-e&RekdOe`vSXwo9={gBE(^y2d;uuvGY`_Ur zu+8iY3y@6tHX?Lr|oaSp^bU^wQF=)GB$zqF#Ty0*Hes-&{gUxc3% z_ZQBuDVSaAuPrGnDE5oel5&4-`TR1Q4q_CP)>bdJ^9qWKtNhg^3;eYUt80;1nJ6u& zD)!eF6cvGAU0blQpvo_1FRt-d*H-x#_^Yb@MYW|BqO72BF6A&GZ+1zIDDuyqUmPz? zmA|ZF0hgs}etGSjs*1AO>V*ZBqTF9oo5-nNTwUWY6Gf$^VtyrxMpSJ@dFkThEa#Sp z!jjV3BLAF%`K5R!InY#x!#cik0N2ACRhULSMsag14l+fggN%vBBFw6~qj`B4ceKN; zbS8=$C)`1GNq0oZTC4+ovK1>#e7*0fSP?rWB83ztY1V|AgE&?Ua!J@LCUQtd9BpO@ zXhn)z84l5HYn;$>dD&1ng-!(t#WEF4D3#%n9@V#E1>JiHxLtX51=Y-a_BSKrKo_uos@&@_{ zwGvXB@>ae|7t1;)(1wPCgCsud(_TM`&-$J4Nk2)IlQ7S4aA(7{@Utdd7=1EA$>|Vj z`>9>HleC8;X)=#9GLM?*I{$kn-h9euP{=3)80Gwp0Kxi zQ9}FL+cK%qOe+-yO{3^$py!mt>QGyha_8Z)07QGB$&`j$EQ!U<8R3}zGoi{iqd0{7 zhJ0hH1xK$~>3~X#in~;lVMS{Y_n~pOtq@0U`c?*(lr6RObX$^+;etBYv?9<4ybk*VneZit4+)T&+JW(Hx%tf{8aC?tFTZ5WvqYrB? zr8`*7^?WOk%79y8e7g}RdZ>Z9eMwIVAWCsC_Uu{2+{&T0h)tDLEnOwL9ge33^xRhx zDJ(6~ei43&>d3iK3VMQy=v$cg?hGsu zH+#(_h{7k^tf^KkbzM+OCRRv&NRE4XIEAcU6KKW8WBe%`nja4$G$Aj>G(j*=$&WD+ ztO@HFHDME#8PL`Y4(LdlfhZ*tCZB?P>wrpsc}>auGEp^Q+_-VIv;8`Nd+i`a)p+1I zv9QozR#I8LAWsyOR?fx46LTlfRa9L`AGozL7v-5dQ4~~_iJH0d%Vvv$vI!GJWyL~& zRb`>|Ij2&T6d-R(3RUH#3^}KAKC|&jl0}tNtE!U13M?E8{d0tWb|I21BJBxzRbcrs z<<`$>!Yz>&+!Dk`kbN^fW`Y*Q(w#zZ;XJEQIA}dFNkqgviAUk(SPoYOvCbx?03To&>PwAQk+k|Zs7X{&)z!WBX zVsp@bn$0VK8$|36$~dHXn9xf@EvtgD`vL6I=>|a}o3&{=Ph_)pDx0+R@`yYqXKEwk zQgR2lm5Om%kL6~-rkE^=$HC3LE@P$2D4y|!6)WNch;!&UJ2Y-4hh^?y0IstQqoS5#L`cZ#PUdyP#oO*osege z$Hu%}_gE^Ibg??n&sR{AF@hx8tho|)>!l~$co1s2H8oK5L=L49d%}&xF@jT%Js`$0 zV-(4Xa6h|~y-z3Xcsa4Vsas{rZ!2yBJ4qW)pfqF7j+dO34L7cf8n&h;Z09EGrX)+M zwp3{m8Z;02QQmf%QnEH*Jx5=*W2On8?x{#cn_clYNrr zruv-D{EE$q*gOW?4SS0e-;g-qk2Ki1=v@p{j+ZdRSUZtJ-aciBuwDu`b$G-^gJ~nS zd>yu|z7vzw3JG>+V;MV$r4qBOILD~i&7!leM`$_etjy{#fpHJjLz4q!euh=td1h^eGju+Hsc)Y7YumQR*iay z@(HD3HPxFT*3#aLWrN-rS5#Y0mui||Dm;eS3t1|0S{=M2F*8xNRy_+=1&5 z9<2j;0t3X*#enq$R3o<84ophK-AntRL_~8mLl~PMYY6H+*L#K)0JIYwNhw zjC%Hz$viHi-k5Rskw-IIE2c?K0X(i~p7)IVTZ_YaSqE#wq4lp+ZKsT$|W`03-P_ERAEghYI|7GdL|Rgc>ea~ znE11(5n(aoym)e!%sj%%%eT0+Inp37+0Z0|`^Y$5!^t|H^2fc_03OHGj}^wX@epHh zE!HN$iuNe2C9r0Sf6s~@@D_BR7wbqo@lNZ)Ry;z_2M1$n8nHy;bs4^n6{0Wv;IIUW z{=5U#|$w*FY#WqoMiycq$4-f@-b~z5|6vmbpZDnzJF=$U)i{;hP(lb*r3%fjnKN1Vn_F?TPb4pv@rZ({ zHhskm4ZyLvdDn>605^a+@sUyQ=Q3Dd*sHQ`eF z@&(C6c=<|EKVH=5iD~9LPxM$y{M%whI2y+n#%GA3Ls544ig9aC!O8Q4Np+*ggV)SY zdf0hpa9G_|-c?jNPO*mS}j3Ic}8@|Ma_qE~MN_a1uNFRbX#LdV1;fjP9QZ52`ds|TCB>jB{ z^x1g-^*MMkUYWGbB-VkEp>z?*wB7_rq+^i~8M@p#cr&1B&w-=SFP;BMUpacF z6n`Q^YcDZ*qSie%XN=*9Ny%+j#yu23zXs5X0hERMqJdj6ig)o*yP_qBm4S=xEs!jN zO7=aVz2%`bvJf?@=hh%idyiVSL4BT2eLk`bx#9bHoNFE42^hB~t*h;Wb%#-@L(nWm z`gMrc#H~g#v@W$Nl@UV#xmV+=gLVis4vJ!0)V?B2EkdQBIJ$T2l(sWVHX%VXKY^N< zK%J66O^Q*v2jV%1-sNc3Cj8}~tknN#w*kw+qwzwPK(^3gbVT>%&}j#aNA-mmTa22u z;A-RE()EkCn+{nmjk=EQYNXoZSglR^Wf)g_OxZOO!?qS5>qlwo;1y^TcJ-(Ax$TYq zY?PinL+A0dKKHw0FkY-W7%At%-V#015*|&$bq2mf*Xy^8r zB~+uVVR&xKbLpJeU&UGqiw2agE~aCzmA&S3kaB>1qg^90Y){`0i^H^IdNNVZqO`tl zsX6F}Ao8e(&C|$B^rAY3*5{T&&ylBXME$ysq4g)yDCizfA1&6?9vk%7xYR0J>ZiSS zsXn7CeIBXkui2lc%QIA8p@q zZDO|5RoRKJ)7O1_^sS})wi1>`Bfj|?Z9(^KT%TGuPIX;Ad+U<-JQJgaY_;=oVAV#q z*3tVI-CBj*x-^#@U3tj&)5yU$w7+3wa(Oap@YYxw<3+euuzgvauJf~MquXL7=*4Hg?^y%yJ9DVuYJwmhJYLuv}xc2hnrVVP^`D|j<#cm;$ zNEctFXTL=AJk7FXHTKp@t}o5-dhI~gZ(kYM+f&y!ZXuoPQVe_fkG?&1`**cAwdxvA zUzg|Ty`gUZqaWWoeLwdUS=%@HdQQK4T%X2woU)fcF>LR>xPI4uu5rD^TAMLysm(*M{?2@nJ^iDf zp|l0yYgG87wQ2~tbUwE1<*}zf8}aJVZt;@a>#NhZ>yw|uEnDQ1xGJ=#fAl_DTO_T` zTk)qyZ0B~8+&;~i&8g4lp!{uE*HU?CFJP?=wZ1ODo{>o}D!MA0z5I6li;Q=_bH~CU~F7{TVJs57!9)-TP^4TbV2xY;$TM)AP$DX=1a`n1|T8h^H zRyjJ0rOTt&C7s(vXX*301S^qt-YwIJn}@toZQLr)DZCEr$};RZ=<-Zr{X(SQ1dGzU zR=ZX_to8ZS0p%etLu(v8TH6v@UdPe;y1yzg%T*!=8qq7bA8>yX>x_8nR{7)lG?I?g z$2xs&Ka~DA^|991{rgR6Q@QNr|E9F#`no(vZ$G*`M{hsY9_MJ++15HZ*?ua}gEi=h z_^8(P)%JGu_M`3X=F5#wEolO(e|ci1)6#2%m^=`f#uPj-~Oldx%~%pu21u0 z=MsB5R?q8}KKhdCp0wsuy@o#0K3!XirGKOr*ELH_-_q}F*RAz+`K=uswVhsxh2ID|iztg>T}beG^yr;zWD@Nf z)^qXxvA1^eDo@0A|JA$$sl(ZprEk?G-Xi)ONnk%!j`2um%MGwKOo>mM==A^D7 zMBii`n%PCaWUjH0lRD>jlJ3CHH92)3)r;1SRML7GvSAzK9AybOGrvESU?#lW7XY(c_Pju<_BP4hez+N4me!KVt+K+*S2>ugXf@wRD?O}ZXoeaATCW3xNM($W z?eR1tM@Tb-+{m}nNxZc~r?jd3bl*(a2=0$yv&pL~=t-U8vip|i`D@eMNWv$Tlr`Dx? z39YTq=DX{-Ct5HUaRNMl(yU8syVO(n#FU8AsqwC&po(WvEB~A$oKo4-vDLFcyAu1; z1yz`l!`!>}IEi6f?F)|Pj43fTQSmkr1+@2~6*lcGDv>w!C0Rh?xj}LoItAWMq8$=f zNvVCC96MeHK8Yxx)i$kp^|KS!uD7#iE$k`jGVs|%LHrC-ui>ie$@sc52m6M4{Izl0P#X4foXTTVr>#?%LML>E zejaP&321aC#`&f)jK>{+N`vgW3FX$SNqeqcr(jvcczzZ~6c)i6XunDKu3d3b+PYjDQ?_*9ZfmdI?<;3$pX9(2c>C38IGoJ zt#Yt^7FMAhsPtr8$^ND5l2~c5g>rk)9+p#1=1E-WQ(sa&lO(+p`-93Y22jsayR~wh2;$L< zf%;EP^nl;&du)99!f=RWZ$!QmX?moeu~RXd;3<@ z!MeY6q>}5n?4PibBQHzwsQsx&X|>W>j`oqrerc-pl$S0soiXY7U6n!S%l>F8`Bux9 z9I32{qnZ5>%1rj&fpXJ|t`Q}+v^xLP`Lh3+S_U7Cw`U+mF+3?fpQJsiKDkPbuj8`c znOcKcfMOm^c_}3wm;KLlI(yP~ug?{zwP_r7m6nccX2j|C)7(TWOD%CcXEPU0FTqS+ zgS?LR&&r#vYr1_FOw5}k5%f9r1KA#pK&#ef{+mvxZYAK_N9B>qVdc#hHeHVdvQT=` zh32m#|5>rw#s<-vOrL<#N}OWPK}q#9cMfoWcHo9z732=U95JL$$=u%Y2Kptr}zgnHS#y(40g5S)Z&eLKaladzQv;L;ka`F6NHU3UQ%DPgFwIbUweNpyPq?eH|3ldo9$_eHRoyHcI0-mT8u4fO5)v7?aey(%9bUL7i?4Kmhd>H9Id?) zh4r0D%7aEk8^#CucKZsNPHV{f(w$7&1CwW7?fN~Fzj(P>cpo@x>4E|IdoNUOufv_;bxLSCaS3NHf*gWz1o(YZ zz$`rEj$0Sb94UK1zn4>y%t1i0oGt?VSgLS(2)qP7f;5730AlrVb|b1gK@WnS1Q`UG z1bqmOBj`^sfM6g&7C|<_i3EcPW)Tz+_z5Zr<^vq=%%Lt6Gjk{^p@Z2jw~(3r!N~>3 z;m&qJA!C5ol}m_%Ww7a^z{Kt&eK!L!bvz%63t5(+K*rMmFOh#$k?J{E&1m z6mv-txlAMIuiQT6?%~N%GBZ1KD1Qz@0dj;F(K2hiZb-bDHJnC`C&wYPU0#Sp#)e)< zjmiflS1!};%#&EJKP8i~VJ9L>`XpvXJ{48R+==8BnBHDqbdT4C?^fmefWC};PH87n zRt~T5Dz}1ybPMH$t_L4IWlsrBZ#HCZG{tD;PIJ05Hyom@@IjRyjpAi)cn%q4&hk`VkyYa00<#f*}M$35F32Cm2DHOMv~baGp#sl3)}7O`V3EB!fc3rYW}zjir#6 z?xtP}Ixrn6T^=;qC@KVMdj@n}9&|4=P~|hw{u%ihy}W6{;YD+1^vd>nQGDoUQ{Tv5 zXq$8lEXWYb=KAbJ1B$N+B!ME};pnE2m%vAmM$nC*2SElwAA)`a#}f=B$RQX)Fq~in zK`ucn!I=bY)Q-EA6be7Gj=n@0dkF=LTaSK~;Nkby6>LlNQX)r*SxT%|-JmFaJ*eNh zxnSJqbz{>#uN#`iO-%z3837V)*sZO4H<>CFa-o1wfGu}7<-H%JkZ?7b8$Lz^ptemS zAk?wqVT z>;;_Hi$Wi8&L1G99y-A^{=g_7)`N5XfK*}I(v6;w9uL5a?vOrohfG6v$neDsMW ziy((!9KkGtN`hqstpw`{E+x2*;4Xq41Um`#5*#29y$EIztcTB3F7%mDD0iAr=|XiA zs=H7<1Q&AsFVMR^dbm6)vwsgCER}!MP;?V6nYsRPFI`@z2c8aUGRsYBG)alf_0PL< z$N@S$G7a9bhYQJM_wb-$dt_y0Aw5to4-$DD!^VlcGS|Q5@OT{faFYLV!1GG!bb6GR z$SkjmCYLM(8G7`986m|i&TAjiKnRSi?jAR$pzJIZZC_?zU55|TJm^eJF8GhaX5_nt zgNlK&d11{xF-v(}9tRL@k{P6gvM8sltPGE@r?aOs%?nR86n`6@OZT|aC`_p%t<0f4 zJ?_j|EXo{}8O_Z0c|2Zk8vVyu=z$WVQm89^q7dHfEY2KFN>Ytf5ZyH3_&ih`e=kYX zbmn5j>H(*ZauS=$iAkV_&63(a4Y|R!;$Ipin3dJdgSkal7PBBKMXq=KC7xxZgZ_X~ zkNW@rKlR#%4C);i3Odg-(9^@0xxt5Z#fAg$P4G-`UB0w5pU;=(#!3R^reU_BZ+F;f zMgV^ssQEJjWF;Qh9knn%$p6uRnhZ6v(2Nb)L{4*&`DKx%5@cc$f+li2L^jBX{HVFS z7^G=1!}DTYoZX{`$JGN23})_j>d-VA%NXfB&`U6UvYTPx_jLDkx#-hlKH(^s_L2~)7%j@mw^Yu(i z>zSUO(d@%0%iJ(NW5Yz4TmQ@uTMfIU=FCukPr9NMNQ3@|GMXvq?N2R*>YxM?iyMz4 z8^eu;n%9LjjS%N;od)MZD|^}t(UX~p-%+aLbvf~;i^14^$9KYGKqWp|H|7W_4Hn%+ z8ob(({%F?_f3lzQ8;@&-E#@>nb4)EQozT$!;om7|1gAyGHR`No=*|V+ipOU{cqJFw zYdPP|NIVyqjma>Bwyo{sKt1 z4%aEjX9`Zc^O4Fpydhu;TI`!!I)`^?scn5UG2^>WZ4vZcwN`T~Vn6%Vb~IRwH9a^D zq0ZWq+B|A>V340kGE4q<0)=x7k@V5arl&=CiGCa223Z!MA2^8A;pa?)WAM^K2!|AZ zJq+p4cJ0`|V=eIiU<){CTZ$)I#CkfNn2S$33t5IweAxbBn*$2xD(gxA(iAzy+qpvA z+zB5NtT2<)Bd!p;8SS9?AJ|zO*8D?V@VyYv!Y1qkDx~?B>ro3XuP^=pES};>v&(1*1I-^*}DtYof zC+)8--C3e1^0IUj=Qh%@X1DQ}bMi3{jlmgfK8Gff7&h1&@srOG<0Yb}Iq2Rotr`4i zGkVgg33qSw-E;idIR5A#j(m_#Ddva<)CxbC486oVmiD^E^>i=PBOaanw;^R}DXm+{ zWvk#GpgU0deAB8cQFIqW;K_`mE2D0E{R}#NlafwE<2BWJ7IAwf>Lj=7H@7MEN%C{; z$#*2axo@Z);49cBX5kZiJs<|RZQL;<2J7|6U%y@;gbRPf0_@na1DvnQ>itdY)5XLjaCw{>pW)_L$z$90}>Z9cQNzCJ!n zXMOalK;ZD2P@r+s?^f8Qy%A`bo~SVbfj5GIK&L>D{DU1PZUc(pO~N+;h=2ZD4m{xW zhY-MsaNK@6Xr}qE-vUt|iAUTB`O~0WqCv8WY1)LQ{q1Nb3(XWSO@C=BqMEt>(ll?R!yD;zBcZnO4x%F96G-rBA)4zzdrRF;80#bz?UWJn zEFN|{fhH6<4s8G(Q6)5JIB4|%KY|s8VBF|YdBC;@)4i1KJpAl1ia>>TV?Cm#)CT~E z((@6@8ijtBLfGW|N0C1gsnGs)NR=J{QSXVN{Ue~)8$8GjB10{O5|-xd;A%H`2!g@F)xtN-1|qv90+AwegGAPH z-Coo=i+XXvX5{hy`>`&v8f<7u*(*vw{*0F z=q(-Ngsb)7Sm2-s=n5V^fV%2jds)||gx=Ee9yqkvp}~Tq4nr2@HgXs!sa;f@D+r5zsIPzJ^EYFb#-YHUGGHs7WIHH9Hkwa^A1ky%?$$Co=vBTQME>&;o z5)WdBi_rc3ZRPoxN1iZt@6gkU!WK@E3(_Kl)8R5b=3;tF=;RL5m%3D%w_`DO(O*Yv zVHc3Q5DoK4vmYI`hR2~&95CxLUz1v_>?PBhZ~S zqOlx-4z3YRGEpmK%^xN9|B3hyibxVs-XkXR7)D84$?ppK55=t4IPLk?KG{`*xr?;D}A8ZQpyLK zEL~L%<@{lh!4qWe!_(E(P_74T6w}o91?5T!wIWQ0?r^^ou?G@}w?T1WiBn(4U%NS|&WF*Y`;E*f2GLRWRWKtuq^+7i6cPv>N-4t0| z=S$^SdQJ5KUnUtk4mGZC;%LMjs@F#Qm7qSoektN*$ zEiGg>`jMsfyY&>Z5BrfNtH^HhBTMaf_?WYA{+N3OYL{%Cb@mYk1Sb5_ECQ_ zS34xPTl~mU`<+ow%MJc?l>(!&oUvS!-R4J@tRnlEA6aU@;{fQ~KJG`BtRmazN0!>} zcnvzY+x^IrRb-#=BTMafTn(Mu9e!lVDzZ=dlU0iMP9L%+cU3_5n7e$bPO$wDP4y`s zszm!WK^kc|4@j8v#laZ?ZqOG8X8^dxzBo7+ zz@6iRLp4@0a*IE5H6w@ok?0uvAuiFEiKz@(- zQl01+OQHIrFV%1b)t7v!)>lwH>PzJg-%{dZzEtZf^7^tbm22uHug86fa)v0RIXMC)ib_Su2u-uSAD5mtq`iO`BJ%BAyi-YrE;}GsGjwuaquQ1~5-b3@-5-g< zfV{v5naIb2f$iy|m>};%fO3y!WWgVK93zpwa^QdfUa5wX0OoQM`)6~EXAhh~6{1rj#2N-Ajienx_%8ky8c z*wreeK`YwG^%7qk$^s5{&nMji#=(wN;lRy8M#2i8h0ne(rvKPhTomm_qeSUn*Ca zLiH|lB=wg8vKtzA;iY_l~^YU=%Lm*_=zC_)X zr+#b&3#h5FD~>YVrQ6a6h~!54`7r%FLqF7iOJAg)H}E6L5|nJcZFfq0340U$+)O{@ z{FkUhm%c-GeNeuy7Q)F1~Tkz^(%L+{Z{UQ5{P>F1;L^ECaC>E~sH((4FGvKC5qx|8f4 zBDsZrK1n~Ej0=C5qHeNDl&r={=JLm(=ZKihYw2m4>IUJ$%eO}rx4uwUjjd;!S%tyKM)@yOr}wQk}_MP6NIdPf}*SY z1i@zOY^paYBb$#vsIN9~S|AWW=T`ao@ax+5P)*}TGFT)IcdtimpOF&26U{a0w~&6H z(dq*59z4E#6O*;M&>9z-4k8R=LwiuJMp-Z*N^^CoJgpEYlJhSJ6?;);&XMs{oF^pIgikzxeKFKBH8SmL8m#IvSGQY2*U0yZ;G3GA0jLh8s)1qT}u3Kox~3dTWFpL-1? zG2vzgP;sMp43MFLh!Gy6kWoC=iBrh1nz3&vyFi}B6B$`2vG5p0HD%Wx@ zqbspVqb9LqwL>1==)lBu^359EXKs_KLD$lG;+@gidff}^AfYa#A?~wv8dwh?V8mL_ zRDp$x^AQUcsX%KX3bGnpqP77$CDhMOnR0+=WErjPbbUxxvUkNy`@|!7@8E(}4ckg>Ketw;6dA!v0%f_`AV?k#6B+2MIyj z0Mac4h0ZN2;1q#RbitDZ47QE@RSwwbehShgO`shW;SlexwI(6|o@D+rAQOmZFRgBi zrKle|DC}1&(%CLu*=wNQHr-GnccG+i0TD_IcK=)#-dK}P0iK<%109XM$^1)x#RtRb zUEr1scLxgY&O?JKAW^dJ9Ai^OHii1celQwmUJ1wMT#WkqxY2njdfU=_Kw)$u~2OE#qMU_L?RnP{P8nUlx+U+$IoIsx^CPbe>vl!_Y>fEGalA;0{j(> z$4ET^{=JMxn@xbflJOYvCct0Ccy!VU@EF%ne~iZy;NQo1^t}o2*DxOabprhR2`}x! zW7g-;W(jIr(I`C%vJ%MXiGh#DQUO&Cp)<9e@D8C&=@}cPG&C3QZ7Urmym6#=Y6bEY z4{{(DDlUY=!4=Uxz}gNqGpsi10;mLOV+hvG?t-@RH$nL)g?;)SjBF>;kn;0$29%!z z?{TV64uS9_G-s17jdy>!yrZ?Egv%xo!%9`v`KpP79?F0p2@a8J; zlM3K~e*uKDrceP-mDiv{p@j9d{z5`0SJNv zZushSI}~Cd;@@I(v_$wx65#-3LGy~oA&`2nOlZh7qs>tMf9CpG|EA^r*FFfs^F z&Kg2%RJ7MjYA+FlleB~ub~Ebjs6?YBT=YOLSfmn0SD}P#rmRo`S%WAv;T2D$h4$fB zm%67c+fo0V>RRff)@&qahjzXSU%Mrnq2E0)-Zk|}vo_NbQ?)9rYR9cE(Ct&TsbRC0 zK$tE3)XPZJngfEUh0VqBm^Bx_kObF-q?OH za=8UcK^<4pDHi7qaG#ybApoR>XP|M zC38ULoK$vWb=gG0j&OyJW&O&IqwEb)|4smZi_HQ*sca7TTSY=yFoW^#MY?tc7@hbI ziji|)m`AY_G%l2W5HJjX87kr$>@O#Gv7-*m>;*GHanUf&>jP?JoS2lj-~jpLpFsUF zeX@N8AsJl(rNLZRPmrb4NV6E+A{MuSCMFfZwAP0>!aYVbMH8P5;<~6Gaa1NAEaCju zs0PfgxY>PF^U5|)psK%|K<;Q)mODS$<^%&ncq*>z7=eRD`pstKdF__u{7wlbl69HJ zNOU}I$n-v%8m@~*(B*#J&G!5OyH^B{jEm2eF(TIyK?Sbd z`MQiKGD8k2c`m$^!HDvtnuymu+yxGqJYTzI*~ReTy*<{j=0k?|qEfcrUh?@LC`fmb z&#uDg@*gJ1JUkZqkz$NJMG5lSNnwDX3&0tswdBj;v4|eLZXE8>ryWCf0NO^?fTFSu zuN~`?Gz0XIe5o`jKg~;u|Mz*o@-{L!D@tz=rfw=)7F#aNs87 ziP?`zL3x6qy`8%d-_S$Bcqld8RFheYV8`+eJ(xW1<;mHGX`6yT?L9pnY9)fXgjj}* zH*~ZLrlV=o=|Ld(PNQcVleUA@Xg#@b0NAKABI{G@v&m>M6dB%2ojTsdx`V}+%uk}! zapPU`fNq2X1@IK6dWY8~BXWsIVVEyVSrkXU-FSdp+Nm~B4`wmr!(5yj z1@1!JT7nkk=ofh8v{<%%ERI5W>n1v>MMkbTHAmVXm~>w z>zM>CmI#$vZyku-M-D<>LSbykCh}*Jax4X62j>;vFt{y*cJGK=%K(g8%P9?-z6M3O ze6sTFDN`O3)~bTw<}w0n1+p4NE7g#L5G$@k)H(=1rqNtSM|kzf-4ViT>yO> zpe_~3##IDP+uuOht)he6a8c_sXa=r;gOWfsUeH4}dE+ojSaMo7R z4^uTMg{-i+5Y8G(rG_&>0L;P_Al^rD6H#4!|NBwN1u zr6B|WQK&}MWc`L>`M*)av02U!#_dknF8&nR@t0A1tQe`5I9YP?z*Fm21p$QDNsA`| z74DeadqD2v2ug)mo3+(o`w-7v=OAr3^#NYItmJ58i$%5K~=mPW$&=Zlt;iCR;+aW5|NJhz-Dh@%$1m-GOoxNVZqR4-^Unf zV-;iGL^#p(3+Srx%%C?Zl`(}oZ&?v0b;WQ{g*7z&JS8^mb7>SbCI_O@KNC0*}fE7yB z8qe(oi_|^8sHU z&m5qm^J0=N+&7!(@#dmj@KPR|^<_uGk??dvgS($SWV-&=gd-wso13|hLYuW18A8P) zP#GRGL&YNzltikQ)f}%p$*QGaf`COgBLR$h0j$d6-5|Rcobg0F#A`j{ub~W5<{-vj z4%GDaYDn+8+@FAvn9;jK0`0vQNnmpCNRBL5Sm)-9yLe$%>yG)gJXXO&)clH9<4r>& zab46eO;V?k$)@}U#i?bRA)~Q3FUOkIhrHt|Oe&iy8B)nB@hi?|u|@mBH=}k~3Ag>W zcW&Hx$Q5i|v=hmVzGi;b^iGj@@6ya69BS`fz|YCO*~~&|P@UQ<$dD^Y8CyMG0I|t- z=^JQ!ypceEYDwhY0??~sp8n$LEmazvXsL(l`zI8CW|lt^soZ?)W7(Ey@hB*4jDH8D z|7{akc<6F4r^C7nj`-)wMI!6|dH(paQfb8G;{JvHyY!7WX)by+SnTpCKvvab%8FouchaX{}-F*I%hYm+RL+$S4(6~wdO|Jvp;K$JP4dHO%8SX1M7IOLY=B7 zn%vU`gKgHfkVH(CG@CPr!-Z-dHz5t{1kg1Y7WBigcH>9fs19JjYup<4psyyH^#H&z ziD~;jlaRtRUSHUyk@??Z?8*NPKi0`WjliRVGq0j{ zA}$6-!$2ZfI=HcBU z*;1ZngU89u#ttw!{p{dM!@*BK3XVE}?BXWMTh0Tfw#!ZfDH=^B{98@t15R!b-6wPo z*KRTEGV2Xhhj=~s<^Yv=?UwcaMb?dLry7Z<$hrxNtV2U#5`lcFR{bAEKqd)s*TJ_1 z)MC~fSgm5$z9DHg!YL=6=}>Ut$IT|ypc-c*yT+7i(q6f(1}5tA-H_{zdwQ&}^}(~?jGX~;=OnmVjLI5Ac4^u+pZ+S7rRmYON^Y8FPI-s?7Q zI`7Oin|s%68#(i3S)g%Nn4C2yEH&XBJT2KQqem!QcmmyVY&jxgw~fGa$hCvU7hbKZ zbrz{=$FPivl-8+GzOj`Ehpp2Pl+~JeL)bIMm;2}mA?%4^+3aAX%Va_9dj87eS&hIW zhj$%ek0P3W1I+OP7#2~2;aK69#O!J=dIO5iM#I+W{6x+x0LGFB7~BX@tbifM5TLzb z^M1W-Rd~}zVPTFc5)2mp)6N+ZqI!0-o%G8B|L-t)-oD#{yxFya|8+5MxrX$B9(j$!@EscDf!mL&`Y z3@d>ASYk%^`NwXQJwqD;huuv}!hyiC);;p*7HWF#R3mo1DJ@ ztGjc8k@R;Fs|#U8h?&MXW>}j*S8qgPyv>YP(j-y;eqk*H7JiLnverv*i!21MGf1a7Jr)ieSjKB$}UP>|W(`X3PZ7z09Mhfs3(p|yce!8k%UR8<1UC^m$%F<7K6VBNln5C)QR z2o=d%dKRo5!`9`=j+H6SzlmghEiL23x0>OkX(X|hH(+3YsC6!2Dzdi|rYA7aNe7g| zHV~`3*VRGV7tl{#D_&|&DZT2!d$B1vG!a!hY(|=c{5H$%QeCSs>kixnmke{3z_Z~% z;3xRM72#UI&#A#O2Lki)|5w1^c?Xif&j9^%h!+s2wD$q_PyACnmjyAOdpGAa&*AsY zDDFKFZ{LwRz-QVWY~KMknBK@c;PqxK8Z43^t)D{M;Hx|ZCE%;)uy#oU4If~1=ZE}BsH(RMh(`Hvy&1*J=$wN5>{d;mhZZ**qM_n);;q4Y5*!si%WSoS_Gi>yPoi_l z;;Ef%ui0qV@-;6Ly}dU0fiiH)20!Qp4|IFQ-pFEy-_m5e4^cQSw~BwOo2jCm zq#JpX^$bxZRdN)oDZ#9Ys^kcT3TQM^$JM~Ba4;0IFbZZT;U> zlhu^5BalNxP1c;<0xJ@=6Yecs!Gb2WLdZpMP5H_OGm7eFwmM1D+esO*D2eu&2F%ee z<^sZefH9P2k&DR_<_0Itl>GZd&r^W!J`>gQG?up4!6J=Jm|3&K))0>IfQtBK=t;Kl zhao{_3}a^t=9gPlc-hl`f9tKJIW2RTp{B|EvuRKbv!fxbI*QapoQrJGg3S(^ovITU zQUc!0s>7<&+68XaG>CGVnr2wA^6~zAlIHZxSVT>ioZ<8b%^7M&kpztC59c>CY0k>z zP@I6(tQ#szoULX9NzE!H&Q!D1?0sq$ikgL>on8NzG?kjkdkfTb5KLFoBh1UP3uyr} zO}=tB+AYzCK{Fl*bYa{Mv2U)2R~}?xj!Bm6JDc6PXQQC%76+Pn#=?uW6*<*6hh&V* zs0@#h|BGDzg5LXTBoO#D{-1;mMv4HY_<B0YxVZ)apz5@6^AbdXl;a>(Q?c;#e zfQI6^Gy*b4W}CAzMuzM$QUWqY+E|Q{8q8y)4PK9aV|d0$!!t(K=6;A0WsJ1xbtZjB z*tzi-X=D2q*~$G2 zDI*?Jal4EZ%81EGS(}SfM1@D-wgjzv(NHoh*5>{Q8GFV=yDB^o+NJl-x4}FRI^cF2 z%mbkfUj8LCuUEX`kwz-@Hb7<19(g2q`IuPzo^r~OLqG>6%gM_CIO&JA84ty!l>lt- zHwqYs3~McZy8Z=5V(?vfWUkHq1gdFQiU()AQW%`ICh_3xfKxV@2WJ~RFxPIE@JIvI z{Uaoijnsii73>xoMQJizOWwZ2HEOFUi#w6ftiv*is^bZ&MF$Lm>EBb%t0-rB#mYcg zE$jBwoN?Zatb2%<-qB*c<3PhI&t-(>zIHJ*`K4F06D3#m*~S>0vriI#wYiU@?~r$KhHLOxhHQLc1APQcasi1p^*X9*pR(efwiE49a6Avxo^wj3o3k#Eog@=d*Y>!|MXAp{j zRbcjU%t`*aPx8nxr6$=jL{MZnsZxfMJTgo|hLda=QW6#~%PQn~ zC#-x-O|qjmAqtjPpaO%RCzd@rVYarIZK~B(*-F8A^tiK1k6S(3+m;#d7lF>4)u>&y zqcB(H+f*BI-e$AVc!J|?4J!#b~|tsFekT|Q{V-?L(i*LgDDfK!DLZ` zDKRz0E*Ohz0dYTDJRXg>op8yZAtk88Y_mt)&K@|D;Sr3BLF-b$`OFObjj3uHyOR+$ zRXc685ks8A4%LBN9^qVi#X0QdT(I@P-gYRCVkdbk9dK`^qYMYDrjK(mx7z-M7m3B# zmlTq)%NSK1y7X(H0b~|p#OQlbw0MoRnGu4*?Fd@$Bk0xmf!}u;K@$YsLr``VYcq!u zw2q*A3CeC_ZKj`~^#uJ8LD@ybd|v!R@kT2BLqi_Q0oNtoqmh*h` z)I6N$nzl^bMW@=c>6#;(CK~R9em5r_v}Xji8-d|D&2p z%@J>q*3;ohM#USv2;QKH3cTz@_YHL7{|C^mf8ZaMi2uk>Y=Lo_gZ?4jqs)7G zSt)J|E{C4+;Z>}r9!cX;9I`c4j^1=A<(>*nMEF!_c7}MC@+&ZF1W6+411Y0X3i5f; z>|9Mld;a+#gu#T`HH}8nOPvIApma`I)(o>cDXrwc%q4FEG+RMVFfaLXn;b`E`y(+v zHcC6-4n-y97e26?y$YvS*eUohZA1=_J_8}Jy8Rw6jG%5x)D17Rpl;2!y4B{6fV_0J zwVuYtk|-5Oo9;bO+Hpnhw`n$;>=5(eCnwfBvkLzTEhgC1%fE-FRR*6G#xyZ93a2ng zZy^VtBt*=*1bGj_%L`!gL}tc17e9bsjK^}C4g!8Y!q~Sa1bA48Yi6U?nk+R1uts#?C^{Qc9MF%1jP_ccg z0h&xMNZD)SjT}gkq4b_y`DEFXu9qd3#9L4AeC0M;X(Mf`wsP~@8M?_o^Rr5!xPy1^;y8u z5TxCwh?x*;YCOcUO@!nuw+gxDKID;dh=rW0(^Uv(Rze?1$RkvR`Y4WgXQyUVEFwU- zW3rJvB2xNAvMy-E(tkl6X{}t?Kn^Uz2T|LwC@S9WLQ!k=Ip?#^lXdeofunAc>KwOa z$C+vUVvP2*^`c(n`X%GEI8C)OT*WoU@bZb*=C!JD2Y4X? z={*2Cd|)o}4soj`GHtqURAH=8N?|rn2^m{DT4KQ`1RkP*&mUo3Tq1ZHPt z?SI=ywllBay{z`=Md8089eM{MjdO-ATwIzUps;lj@#aR!d^u5l!NDRsJ-yZN+_$hs z*O8tqcB(K(E+#@dLQM*h$b~5Ob4RdM-UE=yEnA!?KsV$>>++{U=MYxa#GZwD`|V?` zbqPv}8`wA6vHZ;%obFB-sSPRYF8?jLfPFyjh-7C78J6I805IDs;CX>sp;moT4rt1tjpbtP7}%Y*IFe|nTCEwq%0GQ zLJW`TcwBPB`|>^=LJG>?L<)W)1wWC3pH2#_vP3FZSLz9ve06J*$N%w%mYnMq`c{97)W>cML;(kdk2l}e@G^GBDhfk`D@ zOJHx`QP4HmK;^s;=U2e!m6Wt6XkfXVLhO80L=z$FC+KK(@)|m6hI9XbK486@OAU*! zL~>^jf_+iz&q!d_`q3PVQam)+j-m`{#YhZmGd*aa5pG262I-l^SZ+bA*U+vUYCxZo zk0;*QwTWj|=>cnGCg}o}d7{|MePUU$1Q{-aV6>YB*}83J4_EiY*2cxw(!dx|39(Z? zOdN+CvnQ))t6#kdS;@r&c~r2*3Nl9NedyQeGTZ#hB;>zPCXF5+$U5a3cq5}^kwU6{HHw~H6U?E*1UYBmnkNKS+r={Da+B46XZkQMr~nglTtZ&~a< zDAzS0s2152^ne1c=BM8SKIx!bT8fLfJkp_SnOWBvI0{U3vX{c;tN%G8*_X&a1J=s+ ziiZMU22Z`qf3eqHl@0_7KIns3bj0f`{y1Qk=p%l3K2yq2& z<2$fZX+7E+Dt!VyT?wwkMI#HaRJhLK@=Rhme9uk%F?JPDA<7lZ-s~ z1P%beSyJcW005kV035j6>4lpp!|>P{^r#)%NRQ;$26>#02cK{I8sv?UU+)VG*RG@6 z?3fnE11X2_+BlLCeI}0kfG7|));T*wI#%`m5+rWCX`l{dR1C;65nW^m`4pbVB59%_ z?5-te)4@nFiH&<+p!F)qMj+K7?;mCUyUXc{SnLj4$NJDg9KVOyo|+~z_$ zgy)aBkWPVo+=WaR$UYY`Lm;=ikeLGcgbSG^kULz+Y=L~rg>(tzP8TvqAa}Wtv_S50 zA#(-tX&2Hhkk7b~j6go?Lb3w6*M;;5WPuCm704nNGEX22T}Yom4sap!1(I_i{Q?_Qd@WT^{TD3E0?WRXCYyO09}vciSr1hUeF3<~5R7qVC&2fL6Vfvj>NO9Zmo zg&Zi5hg`@~fjsO&mI>q$7qVO+ShuqEaD_nde$Lgdgg`J_+4VY7AYXSO zM+xLv7jm>fo^v6`2;_Mea;!jJa3RMD;=L+P%dk-+7-4(f$rzTiHEB z`#aaT5bf_wb|Ko|dA|$M{?1evqWzuSE=2n~(_D!5cRE~%_IIYc5ZwqfU5IXl87@RO z!z>q~+uy861JLp~&;e+9Jm>(lJWg`}S{|=C04}}$8k7TmS7VxPT?L2@0M@WRh96D9@qHEXPgKM| ziFg57#jW>IR_SXf_FRdYI;l zfdTLxvQEI174O=Pqls{^ND_-!$Ojea!HNgRjg<8PKQ3 zq{aiAo3X<5j(J=Y;xm-E%q7H|uF}}_iWX zHH^hw0OPG6@f%r;eXxkop~duGum}jfaS!o?z30gE2e~5I`l!%Yr!Y;=LKcgh+l(d6 zcxDQ{k{GX#m6J87IF?~?K2gQ+iuDJP#h;?uyGJCg>yce@DTW6u%9F|o4qt>PhO8Ta z#Vsly!h=?04O-Cc$q1#~V8L{L+tNwh6_99$wP}sDgDfG1x*EZFfa*#CDu=4jV}oPKuz? z9|5Gt0In_*mz@KU`76=I??ukRB6UeRO+`n*#<+hnNY@@K-0>{#J|~0*^}wXiUK+gf zT)YxlgsfkYOzC89$oe^g*n&~Xq(_!@x$A2w(Jw$b_%~K5+G#lW8xFok&>uNL(TE0x zF18+=41Nt5TsTe%xcx@ZYij5Oun0C}eGBRL2La?y1%#|`B5r*YNr}0?BSObOTXImE z0KMTFA2|pa;!FVx7w-s~h@J#(9}zap819$X7%mepWDG7BzjIPX|6 z5z9{3_#5l$dH_%xat;W~o9fS-}R1ia{{QVKNv_ z&X;OZ*n*Xfs}}TiY&TRmS-b2?ywc0<4cQuy-7Kw{JtbsO+h^CwQyPAe-SF81t6Lo} zv&vc>gO^XM=VrTQEzxBysLl+>ioKb&lxa@xN6;wYYw*Vt>gLx;H!Br?WGA>1?Vmt7 z?}UWjRypgna@PAPX9FpxU_?1_b&x1$y(lMMGJBM>5izZtf-h4}8!O6L?@`YB3gslq z36;}JOv+hK>{U)LB`K#vS*4s_N>a{pO0RO-m`dfWca)PT-bOk9iwv(QdT(v^3{^{4F`hyXz6e#WpGwlbQ5Z9D{S>#RdsP(Z`hWb+#p2bybr&mC zrte>%)><40tzl7glk=##bfMZ{jL1GKo&e%PljsN$lVC;E_%Z-+J zb4Y34nVN^iF$k?bws<89j&L`WtTFaD14Mmn@ULt`s2>jg4*C$YK2618P>W|z@IFe`&aQJm}Egti}!M%)Zh&|+W+S&LS zmfI@GQv*}0$z9V@2zd-vcLi;QaXG|GJJpZYH7}FBg;rRY5c?9^$|NPQQhmRt|Sc~-l z8q%zN?JyMFg-@M4NKst$BwiIw+1RV%J6RqAPSl@uYYRqf)9cm#_yv2|<+&vF@L z8x9h_wj|Cu5{?IB=_SCWzd*dQFTG4M`#(#`wgrqQH^|Oe$kNDekip*0?P21h z8-fT~9L#d?i-1Z4!jwAvapPlJ#)Q3Ynm&>vfN18-{POJIE+gv^PNEwP?$jX0p#Zkm z<}gbOU=MA>ICi-;@_{Yd%w{38ct_I)v`7=&LqJBTg1j$w3@m_@j$&C~qHO8p%@L?Y za{h4U2s*OVXdR9y-5i;1$7udDM6x)a16L@$3}MpNcxEQB9poLo0Z@8~63`)@WS*9QSmCX~ zcaAonSr{P;SfIT*lYqsWK@2X$HM=Wu1LO3Qh{;-t5jYi{cMvpQ7b4!#huO097|$g? zwH*r*nMoS#-n|*>$Z2a@-(rSi**px9IBd8Z0yYLaBQYHm>(kRqSJtQ23qF#>Veux~ z@`tO2Iai_gMuut=+Jj-BEqI0lt@FVu^7_d%j**jHF+vh2IDY9(=; zm#j2dYb4))I}^5Eu>Q1QfVRSZ5ArQTWC65YLL+{Ksui-Jokl0!)xLp$2TtBk33)P4 z%SCxyNxr>fmW6pY-BO!~TF)TFmcy*uPXFlyDLdqh`!wl$33h#5Cfp0Q%`>3{%nUfl=wD3*{-DH0_C0_R%`2NSlTE zDY_^r#lPE3|KFN;$P(7?Rk$o45H`Jowxr5 zS>x>z)`!UpA^|(@4v%xcr1CL#VVM>oxcAmC=m`P*Cq1z!on4}ERbF!bLVA#+jj;x! z+)R=d4PuipYC**?h@}ri63eFW0;KLgcVLE z1ThxI#?v{@0_)Ip9*pSve+*aM$o6Dr8F*75CuE`5@gPO{oU9%i^(zDL*8ggYnZO4o zAwZlUS$-KBG-7=VF*@DFV?^gH%ure)$vu!x4PCZvy+UYm7*BxzcR58cDC83@f$c)8 z>MMycFl6U?X=K=;vs?AL@Qpl>@@N7VYI-a?O%`@i5My$_ir}g=8s!VGQh5p7SXrA* z3PocC_YMH_AA;$RWlc^~PiYF@7U&csi#u-%-;r2@u8xkYMEA5%`n$lsg0s)z&%58HAyw@W0whVmXLiB%wFHA<#=u}ZuXoNY7 zT0ey%=}FgmfU{AXS$R z*D!*(rM-KFoQ3Nw;(UevaOrbeH|@*(s7L|n-Tzp_bl4cX5ZLaQ1(I#I>Ap{FU?3s7 zko5z^suO)#(DIRC$_6iB1;iEPoP$L|$T>OM_X~OOpPiJxL(EC1zFkthP~>bJMWu@m zi8U#kD^mcE&YO4aUB(_Q*}KdE04}xx8~^~juY`gF@chxw4|n*(o5C>3gPzv$P@jC%g-Yumm{HG zIKz@Y5l;+fc*F~M-E54peDFPKK}hsSJ6gaxgC;>9L_*_ZFtEy&lO*HYhsJiII@rZe zHPLTAiC=vmAXZcDV95>cLQv+FL09%(R&?rrH*0$Le=vAm3E5AIrxKZLxMMC~s~VAO zu(7QO-b?X^5JzYr5iMevkGJBk`4&DqViWWI*7Lt2dl61Veg|<|+UV>WJH&*^`GY8F zwhi+uiF&wS5zn)HLxE0|0lu-L!J|)W(5TQ9f#7v6?~MorPPJ)6g;SxYn6LBx2Al9W zGJRoCXM@z+D77K)-+YZ<}umbW=JM9lzk7MR`s^p3pJ)y5YhG$+ID7q|ZT_!SvZwatp@gaNrSmB3M|( zxt(BVd!=p74VoY}zXyfqeK8-mJ zJ~%{-*$H-2f|J0sS^N3|7+1piUcHSy0CpNKWH}Nd4VG|2yf+00v=hN7tAZJQy${tQ zA7SxSOsX-e>ThntdxV=%CvHNthqO?V;L9`JSah$U#tuxW=@_b65P`gcfx830xCGEQHfB5F5EZhWn?}z%i`xDlNQFgZoLxnhjsnmO8E3btIm;Dbs#XBp z#zWpfh^O=-Ha{(RG{iIL5Kpaxg*xm3FyNH_ zG^B^EfXyPBiqz6ZNjd|EqkBZ4&%jCdlqQjv6=4|9W_{X?2&Gp(7s;ZTTDYuQtY8KL zYu&QY(>0ub`&3jL$uwFwsUl{9i7b4DRy)UN2l7+sT9Thoc0nL1uVYMTQe*yTSrs9V!li+@i%$=Hy$invw@|GS3h;dzXNDCwO}@0SNtD@2X7>4N_Zk^V&?9ZChj<8 zp(_l9sIw}UrYe`FDwn1zm!>M0CT_ygIkPnBA{eX#D^<9G#EIK<3K+KNKAIRCP8_bc zP=zl01!I{kmp*}4Dv=~hI(vu?Xn!lGqgs(aNK(b(Z&aDZ|3!wZw`CFXx{WSenXqt@ z?b1X#ZM5=lv~fTRLCtgCbxASrWlW~&C{wl=QM4OE{gY6|MJENRy# zs-_fi!-n06Q;Kj9BmzFC6oZHr8tD4NsfbUt&I8*>Z&CW`sND`V8O-9UT`vpsuLP$b zaf+~o|6^Cgn$!J&w3^d$pLa31Z`Kt)ZfPqQe#g!b7)q~3$=RaPu~mq5B8T7rHturk z*chVYPg0qknT0mcCa)Ptp-P?{Wjkyq-Sk1cui@(H45ziO=LW)!wy#rr>Kkp#nrLxV z6LrL-jSRL;646!vGZ;>p)7f8*Xgb`8%XR+{HPnnr=FCavEN=;@WKCec!iBbBNI^b; zVb07nI)FyJgED8T89QrjDd8qI9iwWSFRPbHQZ>DO5(o~gu|+mZ1U1`^&*V66W$Q=l zXL=y2St6jW3W_eFNaQmqM{4#ZySidr=cSV%pGZfen3#ec5_!x&BW2FX9AN|19G1i= ztZfZ0aEtKyDKc2COeRo$w0 z*xP)4M>tsn6Js)A}Z!rh-pfca{^`-@C0qTD*)n9u30{)m2V6Rrf@aB-d)M*Z= zd8%IxSgXi*EL00o42Jf&)ky6l;m*J=DD2=O;t>?Q-;80WGgYJ`A>C&Z@vbAf!C;rr zeHP_@&OJ-$)FP?ZhpAo%r~`J!s2AAXJ-qw@(ArS4461MW#~D;9ZY>W=Iq@kI;cOPAU9FEp0b1&ud{rm z919~Jmy4aaz=z7Q+X;J;lNOA5bH0GESbLS6Gh?1Y^MpFVL2YtTYA}txgmJE4X02M7 zbW6_+xn%P7522c+7Pd&JT`huoFH(!jRA8w=TR>v`kzc5eI5vGC>{ETJXPLr9IjS4c zZq@DWgG;CVgN)ErF+x*P=3;yYb}{G{tHm~0ti>-Q9Zf=YO3EC{thS?Sh_!5V92GT} z*c>b=La*t32i`Wtc|WIlpgK@3DK=5Pr;yGMp_CytWalHLd`iV+bq{-Q!x8x=Cwi4? zwR@U|F?DKQB7Y`f*aoG{rAc#Hi@ChTT+w2#OqmBI&4cr|0iRYV)LfM^SLf*3aGix( zE#1Q;#*yfdvaFXrhjzB@&rkzeC9m1cPR4e$gUO$HiIux=o zmrjspX3+%cdivisowqU7($?W6w8g+SeM`{*uZs6+Xqa>=D|N$mvYVJY))wc0DpdD9$`+kzya zbN=SxxS>yq%{F-pfUFKzhpWTZ(^&+1%6GulW+rj0ry->Q>Tn`Wnn$R^%p-G~!HLdB z9Vrd`5o+M0)KNu~)Zl1!bWf`~3~3HiYk8{=X5B$^ZElv-p#?;;r|oE5O?fnu!Q95p zW1`}`+7aqV{2qyl zAHM8*ItbG4uOQf9@8|^Yme$IWT8_nwfZcd0^F%0`E%Q*)JTYYs=br^z=6Wr^q!#Z< z>LgpdC##dom_JbstM%$+Vg3}gHmOc=ct1sW&*6=YtFLFd5v@f@^VHl*f}V=no#x>w ztWGsg%YPFbsng_QpcZYJyo-Jx;ouvZ)4KjP#Zy1Ykh6WLgny-=-jH~!&U+iN_Z^gBtd1b=I+ z1F5fUQCoCM+JP%Owa|UQ(XX_Zf2UJ_VWrR?MI(U$`U~69F!E2KC zV0s9dt8*hAGB~P(Wcd9k8M*V+>aFU$VjJq0Mz_>cqHVaI^V`s0dlgPReb7xNIgnF7 zdOf8+KdR1m7Z=QJQIKGy))voF=c;YW0&FV5mciI`se{nqG|Z}H2rSdXj&%$fB8)w1 zC13QGyOIr`%vm75!lw!E$Z!i{=61E6cS2^pN8;CUV}Vz zcAIXu#ScDLmyb?Me}g-|)vEL~REvEN^MXi+UQ9R`HY{c4Tg;J^Ihs2dYG|X>s0=Q* zlH@LobWGxrQ{@GUHO3mVpz^6<8^eQUU#lG@l!tUgjbyvjs2VADfQy02DsK~Vx_JiC zn|U>27NY9HVyDw87wLih!Xn1$c;V|L%Zt^;#YT)x{H<5>lBBsKGaDd9!_^LLt%~Xr zwL=wkZ``MLAp0GFNXq{G2*95Yn>*D`yT^{{=2eB_3@SfQWtCJ(U8KgWWz-QaRhNRf znS{KCn)p(6iFF$VE>M?h`H!NUQ8j8yk!!c~E?R2q&`VO^gtDZ}UAYJGw1H}u$m(Mx z*2~moJ7=K`XC~9aUkw{o6 zGpMOs---GIM4ib0n!1+Ef_I^IyVNe5ZK>S>aL7cjE{}Ayh=X+*>(v!4=9MY)s@yTG zSAtSkNmchzRj-b8G>cwcA<)|i`d)Pf>6Ks{y|M{yy<*ZU)s@hztJIanIjBGM>WZ>- z&?{tRUZJiu-y2m|7tgbGO>9-lwjyR+4({QG1Z!Iyb={u$&>< zKalmlT zk|^c1kjAy@TD$HdjR>zj)r&U1ikhX&y}3tN8Uj>%SsJ6CAdUN=wlW_qW+Cpvi%@*? zdiBAbJzm`iW6z7TQ+rXvy=rfyLs*v@4sZ=IP(eWl&&9G-%Dgdm6rPa|0je9Ny<4b;AC9UIyI)f` zZ)!1bZZXHzhuu%M+iQivzIJ=jaDiSl9QqTwbM7@rPXI=*#@uEvFVaUkoHgy{>vDu^1 z#}LQfhHzJl`Ehk~cB;Bb-K;)_=?~^;27%PaEqs!~$en<$uEqzCqQ^~YoN3^B63saK z&*J%N>NLp8OaB_t?o;~)d9`Vu`;}^&liSrk`9e4A=qHlq9q<$P$@jVM7&kwuKFMEK zx5@8BoMr@czChhs;keT`OAYf*^$B$cvZzHP;B3_$so_izfIIa^t`RryV1gJ3`maaz zN7PMa?(j_>ceoeTz8&)T1cdV}Wq+5>x+kXY!kP}d#W$c5y=G;M=`Dt~x`t(j5YLn5&4s)k@mePKP3FdhSIP4O9R()1|hPG80>K<@YmB8e}{8+X{f_2c6d9+b0HJ6#N&E z$>JwD%qJbQv($YYFz;9Q%UgPNuZ5j+*d+iT*67OJAjITaV|n2|cDBvWtIt;uW0Nyp zRBp{*L>0ojK)leBj5!fcJswaGR3zdO@Ipy)b+dm^Jy=00*(bc&-;1WX7jlL!#~1I$ zAdRgkugH$ySJkW7%JZt;@!L@+I&}r4nKB>Bor7mY1Jy&)B%dVB`9h2Nu=>Kz`Ld00K%Q^M_KVnQ*|P$rtg}??tE&diWR9 zFQD|1K~f;fxZRua!%&?cddeyGgJ<*6r1_YnH@~dDtRA&(Ya>U@$JC<`*`t|LI7;TV zL#Qsrh__+dCW_$A`sqXntNSQ625~0K2;I?Y<2#x*olXesVVEW%Y+v9+Fwu{yN7ZA) z=Hp583H6xliTINGl6r!3`lv9Vo&Xz9fbjADpQ>gt!O1OG^(UWvqa0 zK!*8L^ONjL+JkAsfp|HgN6m2GGCi>NwW*hrL%gQ=GBsY!(780g((yQZ#QiLFCPs#b zb|Rcx;~=AbQ}r-B^dl$p=n!W3fgI*0I$3N8gMbdlhwxfMhZ~0Qk~)yX-Ubbr+;@~@ z*J9_ZhJN5+TZV{}-180~HB@xM?L(h;!jp&YB*J|TV5Xg2=MWYWbzw7yE+ZziLuUr|q47{l$=4kwxVicHoM zl;zW6OrBBCOho;RP_xq`UZ0bb+%|p?iqna6U!Zo7d4r~sDZ^LQS8WouPcb_tD+coJ zTG-88hzxdN+^i^u2i>o!uj!PTqtpcz6g0OpzplP+Q}n3ID=3x|#k1;Jn<9}PnCzIp zhoI#TsfTP!BZJ%(|4WmJv>2poAu&6m2RYUO{U+lwFkOZ=pW}A3Vd^;<_3k9w^?X!4 zUtEB$#M7`AL>3~_Qn-hP;uqBOJMnTbp2jd7&#o}69G>Qj>P63YR*(Y2c3gjL1*kFd z)4rizP%o-)Sc`$O?m}KU+WBE;XTK8c8aO1<(Iu6_dJ-xn5u1*OR$edq6!H3!dMSIc zdLgP_Dh^S;2cabElgJ$Yk7m@)dFH^u_U@YODm!Eox!Lmmr2T3?h7e@C-`%Ao*_Ju| zu_6gpy`blO&tb9tIrUs>$sM%l7&E{5dc*73himBbA}!tOo81BXJ4{!11On7Vq3HqH z$cW0q0tBYwf8Lxx;BfrM5vD)de~z=Pv|k+igVQ7EI&p00T!I?&e&qXsI|8^+76=deQZa6d!Br!jhlJ>V!K*Vo$#I2s$~Oocj` zzW}PLZ@j04Y4OjaAxUb}?gC%*1t&00EG|0(V?j67%?K<&MbSlYpZwXV+F%dFB0=jU z(z&6exg^J~?~qzD&KJ_br}!}+1RNh&kbw1b6qUxhJ1qv{ z&6irL4!m|t|2GWlkCc3wBwt3!$u>bO-vTX?OTZ`jGD-ellzcfQ$J#C>2me9rc)bpA zS@FT%p-3=&1fI$Hb2A-8kUpHC^qH(CZ1ST|Wno%sSO-Fhg0v0Q==rYfOV{_wy+rm$ z{7D=>$Bhpl8Qn>pjxH*btWU}ug{%>E@}7)*e+Ei?=ajxvqYGJzLNc>ZbSJ~!pn`|+ zW0HG<9hGus8SA-pml^YN7Xq|~E6dLF;>kT*AYhz*YXE|@0-Yc*na*y*f2p0~u$r*X z)#J!veU4w0<4ZNFe%v~M%ER$(`WDhY0ZApcXEsqKn8>NeOuHWKlwzDuox1f|PW8#6 zKIw}3V1J@RjgK)&y;e}YID4+z2UR`V4@U?Um|b=%x&a*-{{#9kh?>UbcAaT%Sp0A2 z^18)OIFx^f9OzKu;AaS=P<6art1+9CrkT5rKy)BH0s?c~qVtlYgvMo16qmZ6l^LeG zGsLMo4j)PzPNEFP-D*RXKj5*53eDwax`tx~a+Y6S~P)jH04G)m}w5-+P5Q}|U3 z8F(s&*@J(@+7uJE*%ezw72`~G#adA@>_=4KQ7SeG0DL8jD>jcS)}Bj3SJ9sB<6^Y1 zHE-~rRM*L<7}is;12Ab$&5_fprr;B)_@cdlvej-Y4<0iZO*>#gKugVuwytvDu0VXWJEnlOLC2bj4<& z9{hECTLbh)JC6o**%j*=xBg0f7W?3&VslWjSyZuf(wr+5!$(q4u{mn)_{egYJZmXx zw~yoHeU>O(yX)z*$iP#vwG&mWn?A=njvaBj#SWv2ai+Rrb5Swc(gz+zwYvboS74or z1=;=KWd;fAcZ6mp2j>izH1S2Vc$^0>iC?|RDXi<@r16D@l1gR7TR)0wdppGA1HQu_ z{3(E~!@)Q<`el8RO_1B2lWzP-&%rf2Dytdl*H>7FDr6T?h)0%1u)*g{CyLVzGapkZdhHcSYMfC>nRs8JBY zq9VAkt0)M(E{GZhM3h|w7Z6cFK#l+BIj8RJdwV(|sPFrJ|KIlorn=6ls#B*j2qrW!nL@XWK>Y+;eQx^x-?r zt)JPtFU$YF?96TMxj^*oM{pK4@6BQdiaqt0u9qhmH=WP1ss~{+URC`f8ju1Olwm-y z#g32T;0$bbmmK^D!ks)wnw{7o4yC7=*jkwgU+XO*yQ!T?Ce$?pm*CRqtl$NaW~D&La|YufjxGfwLuXg+Tn=Dc>rqhL zQMQtqy0RF{v!9OKr7UCbU}PPI^G0P^$WOPfl4OA>IzY!MzgB+i@>!u)%vjrdNcIQM z`+;XYK*h5=IE}>fJQ!l>0U$x}<*(5$j^Gk)k9XzSP%xS7%9(l_re}s82X#H0gnftx zd3f~Yz?dF_THJlG4tHUUM;W7L0(e>Bml0R6BCGZWM~-pe z2qu|#Rm1h1Fy(&`;yZGX*pH#fTuFxW&?+BkSV+LL^Dqbo8#Qfx$pmmF-)0|h`a=v-N;rfrw!Ax4JacPp=U0Fb5-4w z@RO95Kbr$13A2g_LJ_@}NR3E2T4>@b3q3YBZb(uw7fG=e-FJYgArUZBhO00uqV>Wf zf0$huGbC1kdVckgP$9jjB3en`3Q?=>j~WsyKw2dxZBC+zzAyLzXQiFV&){^fN}O+@ zd;id&4F4P;@DpL%gv!YK#v3t$jJ3~IQmbL2v@;+UPaii*u-*GB>L{P+rP!U%T zuo3OZ2@sVY`;67dYe}>V&*tUiJSFXL3_7fN&Mr}TuX$=FQQeY^ZZ34s!G*}AZA9NR zpwGlnY7G5@AjW|{$ornEAQ0@j&i4Ko;^$oiygT2O6L`IU25QVDlo-zr;_`uyAaPE& zJo73KMKt*cJdns3%{~IBjENZTBXE5J5hHvA-j*g}q>mV-h*3Ubv?50Ph}I}dmdU%l zr;j!#XG9&UqQwiqbs^V#18@rPf?ij;tg7CdfgKSn2!+BDB;hn;_!t|WfFqQvQ#^qF zVMGWBkvx__?=c6?&75SFEXpI|yhi~a-|0b08U91*Cn@;ZWehT3_zxqVG~vey{d5th z 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≪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 b^d!)eKu-ca3G^h;lR!@bJqi53l)!%jl4nw% diff --git a/prebuild.xml b/prebuild.xml index 49063daa57..33a6b5005b 100644 --- a/prebuild.xml +++ b/prebuild.xml @@ -2163,7 +2163,6 @@ - @@ -2201,8 +2200,6 @@ - - @@ -2235,8 +2232,6 @@ - - @@ -2274,8 +2269,6 @@ - - @@ -2304,7 +2297,6 @@ - @@ -2346,7 +2338,6 @@ - @@ -2385,7 +2376,6 @@ - From 3a38bd8854a93bb23308e4f09b02fd7b0aa672ce Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Mon, 25 Apr 2011 08:53:48 -0700 Subject: [PATCH 22/45] Addresses mantis #5449 -- crashing exception in SendInventoryFolderDetails. --- .../Connectors/Inventory/XInventoryConnector.cs | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/OpenSim/Services/Connectors/Inventory/XInventoryConnector.cs b/OpenSim/Services/Connectors/Inventory/XInventoryConnector.cs index cd9f2bfcff..a662abb867 100644 --- a/OpenSim/Services/Connectors/Inventory/XInventoryConnector.cs +++ b/OpenSim/Services/Connectors/Inventory/XInventoryConnector.cs @@ -158,7 +158,10 @@ namespace OpenSim.Services.Connectors public InventoryCollection GetFolderContent(UUID principalID, UUID folderID) { InventoryCollection inventory = new InventoryCollection(); - + inventory.Folders = new List(); + inventory.Items = new List(); + inventory.UserID = principalID; + try { Dictionary ret = MakeRequest("GETFOLDERCONTENT", @@ -172,11 +175,6 @@ namespace OpenSim.Services.Connectors if (ret.Count == 0) return null; - - inventory.Folders = new List(); - inventory.Items = new List(); - inventory.UserID = principalID; - Dictionary folders = (Dictionary)ret["FOLDERS"]; Dictionary items = From 024c12abc3aa42432e55e322141cad1edeb5bad1 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Mon, 25 Apr 2011 10:44:41 -0700 Subject: [PATCH 23/45] Cleaned up various configuration options. Removed the category throttle limits because the only ones used now are the defaults (which are overwritten by the client throttles anyway). Updated the default rates to correspond to about 350kbps. Also added a configuration to disable adaptive throttle. The default is the previous behavior (no adaptation). --- .../ClientStack/LindenUDP/LLUDPClient.cs | 13 ++- .../ClientStack/LindenUDP/ThrottleRates.cs | 80 +++---------------- .../ClientStack/LindenUDP/TokenBucket.cs | 19 ++++- .../Agent/UDP/Linden/LindenUDPInfoModule.cs | 18 ++--- bin/OpenSimDefaults.ini | 38 ++++----- 5 files changed, 59 insertions(+), 109 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs index 103ec66590..494cb0c63f 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs @@ -181,9 +181,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_maxRTO = maxRTO; // Create a token bucket throttle for this client that has the scene token bucket as a parent - m_throttleClient = new AdaptiveTokenBucket(parentThrottle, rates.TotalLimit); + m_throttleClient = new AdaptiveTokenBucket(parentThrottle, rates.Total, rates.AdaptiveThrottlesEnabled); // Create a token bucket throttle for the total categary with the client bucket as a throttle - m_throttleCategory = new TokenBucket(m_throttleClient, rates.TotalLimit); + m_throttleCategory = new TokenBucket(m_throttleClient, rates.Total); // Create an array of token buckets for this clients different throttle categories m_throttleCategories = new TokenBucket[THROTTLE_CATEGORY_COUNT]; @@ -194,7 +194,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Initialize the packet outboxes, where packets sit while they are waiting for tokens m_packetOutboxes[i] = new OpenSim.Framework.LocklessQueue(); // Initialize the token buckets that control the throttling for each category - m_throttleCategories[i] = new TokenBucket(m_throttleCategory, rates.GetLimit(type)); + m_throttleCategories[i] = new TokenBucket(m_throttleCategory, rates.GetRate(type)); } // Default the retransmission timeout to three seconds @@ -341,12 +341,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP task = Math.Max(task, LLUDPServer.MTU); texture = Math.Max(texture, LLUDPServer.MTU); asset = Math.Max(asset, LLUDPServer.MTU); - state = Math.Max(state, LLUDPServer.MTU); - int total = resend + land + wind + cloud + task + texture + asset + state; + int total = resend + land + wind + cloud + task + texture + asset; - //m_log.DebugFormat("[LLUDPCLIENT]: {0} is setting throttles. Resend={1}, Land={2}, Wind={3}, Cloud={4}, Task={5}, Texture={6}, Asset={7}, State={8}, Total={9}", - // AgentID, resend, land, wind, cloud, task, texture, asset, state, total); + //m_log.DebugFormat("[LLUDPCLIENT]: {0} is setting throttles. Resend={1}, Land={2}, Wind={3}, Cloud={4}, Task={5}, Texture={6}, Asset={7}, Total={8}", + // AgentID, resend, land, wind, cloud, task, texture, asset, total); // Update the token buckets with new throttle values TokenBucket bucket; diff --git a/OpenSim/Region/ClientStack/LindenUDP/ThrottleRates.cs b/OpenSim/Region/ClientStack/LindenUDP/ThrottleRates.cs index aaf6e26d58..c9aac0ba09 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/ThrottleRates.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/ThrottleRates.cs @@ -52,30 +52,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP public int Texture; /// Drip rate for asset packets public int Asset; - /// Drip rate for state packets - public int State; + /// Drip rate for the parent token bucket public int Total; - /// Maximum burst rate for resent packets - public int ResendLimit; - /// Maximum burst rate for land packets - public int LandLimit; - /// Maximum burst rate for wind packets - public int WindLimit; - /// Maximum burst rate for cloud packets - public int CloudLimit; - /// Maximum burst rate for task (state and transaction) packets - public int TaskLimit; - /// Maximum burst rate for texture packets - public int TextureLimit; - /// Maximum burst rate for asset packets - public int AssetLimit; - /// Maximum burst rate for state packets - public int StateLimit; - /// Burst rate for the parent token bucket - public int TotalLimit; - + /// Flag used to enable adaptive throttles + public bool AdaptiveThrottlesEnabled; + /// /// Default constructor /// @@ -86,26 +69,17 @@ namespace OpenSim.Region.ClientStack.LindenUDP { IConfig throttleConfig = config.Configs["ClientStack.LindenUDP"]; - Resend = throttleConfig.GetInt("resend_default", 12500); - Land = throttleConfig.GetInt("land_default", 1000); - Wind = throttleConfig.GetInt("wind_default", 1000); - Cloud = throttleConfig.GetInt("cloud_default", 1000); - Task = throttleConfig.GetInt("task_default", 1000); - Texture = throttleConfig.GetInt("texture_default", 1000); - Asset = throttleConfig.GetInt("asset_default", 1000); - State = throttleConfig.GetInt("state_default", 1000); - - ResendLimit = throttleConfig.GetInt("resend_limit", 18750); - LandLimit = throttleConfig.GetInt("land_limit", 29750); - WindLimit = throttleConfig.GetInt("wind_limit", 18750); - CloudLimit = throttleConfig.GetInt("cloud_limit", 18750); - TaskLimit = throttleConfig.GetInt("task_limit", 18750); - TextureLimit = throttleConfig.GetInt("texture_limit", 55750); - AssetLimit = throttleConfig.GetInt("asset_limit", 27500); - StateLimit = throttleConfig.GetInt("state_limit", 37000); + Resend = throttleConfig.GetInt("resend_default", 6625); + Land = throttleConfig.GetInt("land_default", 9125); + Wind = throttleConfig.GetInt("wind_default", 1750); + Cloud = throttleConfig.GetInt("cloud_default", 1750); + Task = throttleConfig.GetInt("task_default", 18500); + Texture = throttleConfig.GetInt("texture_default", 18500); + Asset = throttleConfig.GetInt("asset_default", 10500); Total = throttleConfig.GetInt("client_throttle_max_bps", 0); - TotalLimit = Total; + + AdaptiveThrottlesEnabled = throttleConfig.GetBoolean("enable_adaptive_throttles", false); } catch (Exception) { } } @@ -128,34 +102,6 @@ namespace OpenSim.Region.ClientStack.LindenUDP return Texture; case ThrottleOutPacketType.Asset: return Asset; - case ThrottleOutPacketType.State: - return State; - case ThrottleOutPacketType.Unknown: - default: - return 0; - } - } - - public int GetLimit(ThrottleOutPacketType type) - { - switch (type) - { - case ThrottleOutPacketType.Resend: - return ResendLimit; - case ThrottleOutPacketType.Land: - return LandLimit; - case ThrottleOutPacketType.Wind: - return WindLimit; - case ThrottleOutPacketType.Cloud: - return CloudLimit; - case ThrottleOutPacketType.Task: - return TaskLimit; - case ThrottleOutPacketType.Texture: - return TextureLimit; - case ThrottleOutPacketType.Asset: - return AssetLimit; - case ThrottleOutPacketType.State: - return StateLimit; case ThrottleOutPacketType.Unknown: default: return 0; diff --git a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs index 677d3d1713..2ec79abd16 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs @@ -340,6 +340,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP set { m_maxDripRate = (value == 0 ? 0 : Math.Max(value,m_minimumFlow)); } } + private bool m_enabled = false; + // // // @@ -357,9 +359,16 @@ namespace OpenSim.Region.ClientStack.LindenUDP // // // - public AdaptiveTokenBucket(TokenBucket parent, Int64 maxDripRate) : base(parent,m_minimumFlow) + public AdaptiveTokenBucket(TokenBucket parent, Int64 maxDripRate, bool enabled) : base(parent,maxDripRate) { - MaxDripRate = maxDripRate; + m_enabled = enabled; + + if (m_enabled) + { + m_log.WarnFormat("[TOKENBUCKET] Adaptive throttle enabled"); + MaxDripRate = maxDripRate; + AdjustedDripRate = m_minimumFlow; + } } // @@ -368,7 +377,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP public void ExpirePackets(Int32 count) { // m_log.WarnFormat("[ADAPTIVEBUCKET] drop {0} by {1} expired packets",AdjustedDripRate,count); - AdjustedDripRate = (Int64) (AdjustedDripRate / Math.Pow(2,count)); + if (m_enabled) + AdjustedDripRate = (Int64) (AdjustedDripRate / Math.Pow(2,count)); } // @@ -376,7 +386,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP // public void AcknowledgePackets(Int32 count) { - AdjustedDripRate = AdjustedDripRate + count; + if (m_enabled) + AdjustedDripRate = AdjustedDripRate + count; } } } diff --git a/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs b/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs index ddbc079579..db17d8faf7 100644 --- a/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs +++ b/OpenSim/Region/OptionalModules/Agent/UDP/Linden/LindenUDPInfoModule.cs @@ -398,7 +398,7 @@ namespace OpenSim.Region.CoreModules.UDP.Linden if (client is LLClientView) { LLClientView llClient = client as LLClientView; - + if (firstClient) { report.AppendLine(GetServerThrottlesReport(llClient.UDPServer)); @@ -451,7 +451,7 @@ namespace OpenSim.Region.CoreModules.UDP.Linden int maxRegionNameLength = 14; int maxTypeLength = 4; - string name = "SERVER AGENT LIMITS"; + string name = "SERVER AGENT RATES"; report.Append(GetColumnEntry(name, maxNameLength, columnPadding)); report.Append(GetColumnEntry("-", maxRegionNameLength, columnPadding)); @@ -461,13 +461,13 @@ namespace OpenSim.Region.CoreModules.UDP.Linden report.AppendFormat( "{0,7} {1,8} {2,7} {3,7} {4,7} {5,7} {6,9} {7,7}", (throttleRates.Total * 8) / 1000, - (throttleRates.ResendLimit * 8) / 1000, - (throttleRates.LandLimit * 8) / 1000, - (throttleRates.WindLimit * 8) / 1000, - (throttleRates.CloudLimit * 8) / 1000, - (throttleRates.TaskLimit * 8) / 1000, - (throttleRates.TextureLimit * 8) / 1000, - (throttleRates.AssetLimit * 8) / 1000); + (throttleRates.Resend * 8) / 1000, + (throttleRates.Land * 8) / 1000, + (throttleRates.Wind * 8) / 1000, + (throttleRates.Cloud * 8) / 1000, + (throttleRates.Task * 8) / 1000, + (throttleRates.Texture * 8) / 1000, + (throttleRates.Asset * 8) / 1000); return report.ToString(); } diff --git a/bin/OpenSimDefaults.ini b/bin/OpenSimDefaults.ini index 7ea98d49e8..35e8079cf5 100644 --- a/bin/OpenSimDefaults.ini +++ b/bin/OpenSimDefaults.ini @@ -362,30 +362,24 @@ ; ;client_throttle_max_bps = 196608 - ; Per-client bytes per second rates for the various throttle categories. - ; These are default values that will be overriden by clients + ; Adaptive throttling attempts to limit network overload when multiple + ; clients login by starting each connection more slowly. Disabled by + ; default ; - ;resend_default = 12500 - ;land_default = 1000 - ;wind_default = 1000 - ;cloud_default = 1000 - ;task_default = 1000 - ;texture_default = 1000 - ;asset_default = 1000 - ;state_default = 1000 + ;enable_adaptive_throttles = true - ; Per-client maximum burst rates in bytes per second for the various - ; throttle categories. These are default values that will be overriden by - ; clients - ; - ;resend_limit = 18750 - ;land_limit = 29750 - ;wind_limit = 18750 - ;cloud_limit = 18750 - ;task_limit = 18750 - ;texture_limit = 55750 - ;asset_limit = 27500 - ;state_limit = 37000 + ; Per-client bytes per second rates for the various throttle categories. + ; These are default values that will be overriden by clients. These + ; defaults are approximately equivalent to the throttles set by the Imprudence + ; viewer when maximum bandwidth is set to 350kbps + + ;resend_default = 6625 + ;land_default = 9125 + ;wind_default = 1750 + ;cloud_default = 1750 + ;task_default = 18500 + ;texture_default = 18500 + ;asset_default = 10500 ; Configures how ObjectUpdates are aggregated. These numbers ; do not literally mean how many updates will be put in each From ef4f1fc4baaa00c6465c875c2ee02323644d0ed1 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Mon, 25 Apr 2011 11:00:48 -0700 Subject: [PATCH 24/45] Made things consistent between LocalInventoryServiceConnector and RemoteXInventoryServiceConnector on GetFolderContent. --- .../LocalInventoryServiceConnector.cs | 9 +++-- .../RemoteXInventoryServiceConnector.cs | 34 ++++++++++++++----- 2 files changed, 32 insertions(+), 11 deletions(-) diff --git a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/LocalInventoryServiceConnector.cs b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/LocalInventoryServiceConnector.cs index c7244c88ef..6dd871faa3 100644 --- a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/LocalInventoryServiceConnector.cs +++ b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/LocalInventoryServiceConnector.cs @@ -182,9 +182,12 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Inventory public InventoryCollection GetFolderContent(UUID userID, UUID folderID) { InventoryCollection invCol = m_InventoryService.GetFolderContent(userID, folderID); - if (UserManager != null) - foreach (InventoryItemBase item in invCol.Items) - UserManager.AddUser(item.CreatorIdAsUuid, item.CreatorData); + Util.FireAndForget(delegate + { + if (UserManager != null) + foreach (InventoryItemBase item in invCol.Items) + UserManager.AddUser(item.CreatorIdAsUuid, item.CreatorData); + }); return invCol; } diff --git a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/RemoteXInventoryServiceConnector.cs b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/RemoteXInventoryServiceConnector.cs index bd01bb944f..8f1f25774a 100644 --- a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/RemoteXInventoryServiceConnector.cs +++ b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Inventory/RemoteXInventoryServiceConnector.cs @@ -46,10 +46,22 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Inventory LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private bool m_Enabled = false; - private bool m_Initialized = false; -// private Scene m_Scene; + private Scene m_Scene; private XInventoryServicesConnector m_RemoteConnector; + private IUserManagement m_UserManager; + private IUserManagement UserManager + { + get + { + if (m_UserManager == null) + { + m_UserManager = m_Scene.RequestModuleInterface(); + } + return m_UserManager; + } + } + public Type ReplaceableInterface { get { return null; } @@ -114,12 +126,10 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Inventory if (!m_Enabled) return; - if (!m_Initialized) - { - m_Initialized = true; - } - scene.RegisterModuleInterface(this); + + if (m_Scene == null) + m_Scene = scene; } public void RemoveRegion(Scene scene) @@ -173,7 +183,15 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Inventory public InventoryCollection GetFolderContent(UUID userID, UUID folderID) { - return m_RemoteConnector.GetFolderContent(userID, folderID); + InventoryCollection invCol = m_RemoteConnector.GetFolderContent(userID, folderID); + Util.FireAndForget(delegate + { + if (UserManager != null) + foreach (InventoryItemBase item in invCol.Items) + UserManager.AddUser(item.CreatorIdAsUuid, item.CreatorData); + }); + + return invCol; } public List GetFolderItems(UUID userID, UUID folderID) From 77ab7ce084d32c45273d79e9ec4f52c43e3a1d97 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Mon, 25 Apr 2011 15:11:29 -0700 Subject: [PATCH 25/45] Fixed the transmission of throttles from root agent to child agents. Child throttles are based on the number of child agents known to the root and at least 1/4 of the throttle given to the root. --- .../ClientStack/LindenUDP/LLClientView.cs | 2 +- .../ClientStack/LindenUDP/LLUDPClient.cs | 40 ++++++++++++------- .../Region/Framework/Scenes/ScenePresence.cs | 37 ++++++++--------- 3 files changed, 43 insertions(+), 36 deletions(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 32a075a79c..cdd4224346 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -11420,7 +11420,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP /// public byte[] GetThrottlesPacked(float multiplier) { - return m_udpClient.GetThrottlesPacked(); + return m_udpClient.GetThrottlesPacked(multiplier); } /// diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs index 494cb0c63f..50f1e2c87e 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs @@ -329,8 +329,6 @@ namespace OpenSim.Region.ClientStack.LindenUDP int asset = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f); // State is a subcategory of task that we allocate a percentage to int state = 0; - // int state = (int)((float)task * STATE_TASK_PERCENTAGE); - // task -= state; // Make sure none of the throttles are set below our packet MTU, // otherwise a throttle could become permanently clogged @@ -342,17 +340,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP texture = Math.Max(texture, LLUDPServer.MTU); asset = Math.Max(asset, LLUDPServer.MTU); - int total = resend + land + wind + cloud + task + texture + asset; - + //int total = resend + land + wind + cloud + task + texture + asset; //m_log.DebugFormat("[LLUDPCLIENT]: {0} is setting throttles. Resend={1}, Land={2}, Wind={3}, Cloud={4}, Task={5}, Texture={6}, Asset={7}, Total={8}", // AgentID, resend, land, wind, cloud, task, texture, asset, total); // Update the token buckets with new throttle values TokenBucket bucket; - bucket = m_throttleCategory; - bucket.RequestedDripRate = total; - bucket = m_throttleCategories[(int)ThrottleOutPacketType.Resend]; bucket.RequestedDripRate = resend; @@ -381,22 +375,38 @@ namespace OpenSim.Region.ClientStack.LindenUDP m_packedThrottles = null; } - public byte[] GetThrottlesPacked() + public byte[] GetThrottlesPacked(float multiplier) { byte[] data = m_packedThrottles; if (data == null) { + float rate; + data = new byte[7 * 4]; int i = 0; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Resend].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Land].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Wind].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Cloud].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Task].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Texture].RequestedDripRate), 0, data, i, 4); i += 4; - Buffer.BlockCopy(Utils.FloatToBytes((float)m_throttleCategories[(int)ThrottleOutPacketType.Asset].RequestedDripRate), 0, data, i, 4); i += 4; + // multiply by 8 to convert bytes back to bits + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Resend].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Land].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Wind].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Cloud].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Task].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Texture].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; + + rate = (float)m_throttleCategories[(int)ThrottleOutPacketType.Asset].RequestedDripRate * 8 * multiplier; + Buffer.BlockCopy(Utils.FloatToBytes(rate), 0, data, i, 4); i += 4; m_packedThrottles = data; } diff --git a/OpenSim/Region/Framework/Scenes/ScenePresence.cs b/OpenSim/Region/Framework/Scenes/ScenePresence.cs index 00a1487ab6..ef0eb89fc9 100644 --- a/OpenSim/Region/Framework/Scenes/ScenePresence.cs +++ b/OpenSim/Region/Framework/Scenes/ScenePresence.cs @@ -2632,18 +2632,17 @@ namespace OpenSim.Region.Framework.Scenes cadu.GroupAccess = 0; cadu.Position = AbsolutePosition; cadu.regionHandle = m_rootRegionHandle; - float multiplier = 1; - int innacurateNeighbors = m_scene.GetInaccurateNeighborCount(); - if (innacurateNeighbors != 0) - { - multiplier = 1f / (float)innacurateNeighbors; - } - if (multiplier <= 0f) - { - multiplier = 0.25f; - } - //m_log.Info("[NeighborThrottle]: " + m_scene.GetInaccurateNeighborCount().ToString() + " - m: " + multiplier.ToString()); + // Throttles + float multiplier = 1; + int childRegions = m_knownChildRegions.Count; + if (childRegions != 0) + multiplier = 1f / childRegions; + + // Minimum throttle for a child region is 1/4 of the root region throttle + if (multiplier <= 0.25f) + multiplier = 0.25f; + cadu.throttles = ControllingClient.GetThrottlesPacked(multiplier); cadu.Velocity = Velocity; @@ -3039,16 +3038,14 @@ namespace OpenSim.Region.Framework.Scenes // Throttles float multiplier = 1; - int innacurateNeighbors = m_scene.GetInaccurateNeighborCount(); - if (innacurateNeighbors != 0) - { - multiplier = 1f / innacurateNeighbors; - } - if (multiplier <= 0f) - { + int childRegions = m_knownChildRegions.Count; + if (childRegions != 0) + multiplier = 1f / childRegions; + + // Minimum throttle for a child region is 1/4 of the root region throttle + if (multiplier <= 0.25f) multiplier = 0.25f; - } - //m_log.Info("[NeighborThrottle]: " + m_scene.GetInaccurateNeighborCount().ToString() + " - m: " + multiplier.ToString()); + cAgent.Throttles = ControllingClient.GetThrottlesPacked(multiplier); cAgent.HeadRotation = m_headrotation; From 13f141a4d521fc15618c2983c989805d9174b969 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Mon, 25 Apr 2011 15:36:59 -0700 Subject: [PATCH 26/45] Fix the totals shown by show throttle --- OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs index 50f1e2c87e..ca5501d875 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLUDPClient.cs @@ -183,7 +183,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP // Create a token bucket throttle for this client that has the scene token bucket as a parent m_throttleClient = new AdaptiveTokenBucket(parentThrottle, rates.Total, rates.AdaptiveThrottlesEnabled); // Create a token bucket throttle for the total categary with the client bucket as a throttle - m_throttleCategory = new TokenBucket(m_throttleClient, rates.Total); + m_throttleCategory = new TokenBucket(m_throttleClient, 0); // Create an array of token buckets for this clients different throttle categories m_throttleCategories = new TokenBucket[THROTTLE_CATEGORY_COUNT]; From b9bca893efaeea3dd76dd3cb2082c181e5510b59 Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Mon, 25 Apr 2011 16:13:16 -0700 Subject: [PATCH 27/45] Removed debug message in the token bucket code --- OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs index 2ec79abd16..29fd1a409a 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/TokenBucket.cs @@ -365,7 +365,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP if (m_enabled) { - m_log.WarnFormat("[TOKENBUCKET] Adaptive throttle enabled"); + // m_log.DebugFormat("[TOKENBUCKET] Adaptive throttle enabled"); MaxDripRate = maxDripRate; AdjustedDripRate = m_minimumFlow; } From 037373b825235235e89112abe81f985efa052c2a Mon Sep 17 00:00:00 2001 From: dahlia Date: Mon, 25 Apr 2011 17:11:54 -0700 Subject: [PATCH 28/45] recover from unhandled exception from bad rotation data while processing entity updates in LLClientView.cs --- OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 1f7e66dabb..0f7f666c60 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -4764,7 +4764,15 @@ namespace OpenSim.Region.ClientStack.LindenUDP data.RelativePosition.ToBytes(objectData, 0); data.Velocity.ToBytes(objectData, 12); data.Acceleration.ToBytes(objectData, 24); - data.RotationOffset.ToBytes(objectData, 36); + try + { + data.RotationOffset.ToBytes(objectData, 36); + } + catch (Exception e) + { + m_log.Warn("[LLClientView]: exception converting quaternion to bytes, using Quaternion.Identity. Exception: " + e.ToString()); + OpenMetaverse.Quaternion.Identity.ToBytes(objectData, 36); + } data.AngularVelocity.ToBytes(objectData, 48); ObjectUpdatePacket.ObjectDataBlock update = new ObjectUpdatePacket.ObjectDataBlock(); From e579a990b4466fcecc11a9b8c6ad74cc70530e5c Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Mon, 25 Apr 2011 18:59:01 -0700 Subject: [PATCH 29/45] Removed stale client components: MXP and VWoHTTP. --- .../Client/MXP/ClientStack/MXPClientView.cs | 1728 ----------------- OpenSim/Client/MXP/MXPModule.cs | 131 -- OpenSim/Client/MXP/MXPUtil.cs | 42 - .../MXP/PacketHandler/MXPPacketServer.cs | 561 ------ .../VWoHTTP/ClientStack/VWHClientView.cs | 1231 ------------ OpenSim/Client/VWoHTTP/VWoHTTPModule.cs | 133 -- bin/MXP.dll | Bin 114688 -> 0 bytes bin/protobuf-net.dll | Bin 107008 -> 0 bytes prebuild.xml | 63 - 9 files changed, 3889 deletions(-) delete mode 100644 OpenSim/Client/MXP/ClientStack/MXPClientView.cs delete mode 100644 OpenSim/Client/MXP/MXPModule.cs delete mode 100644 OpenSim/Client/MXP/MXPUtil.cs delete mode 100644 OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs delete mode 100644 OpenSim/Client/VWoHTTP/ClientStack/VWHClientView.cs delete mode 100644 OpenSim/Client/VWoHTTP/VWoHTTPModule.cs delete mode 100644 bin/MXP.dll delete mode 100644 bin/protobuf-net.dll diff --git a/OpenSim/Client/MXP/ClientStack/MXPClientView.cs b/OpenSim/Client/MXP/ClientStack/MXPClientView.cs deleted file mode 100644 index a604a2ee63..0000000000 --- a/OpenSim/Client/MXP/ClientStack/MXPClientView.cs +++ /dev/null @@ -1,1728 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -using System; -using System.Collections.Generic; -using System.Net; -using System.Reflection; -using System.Text; -using log4net; -using MXP; -using MXP.Messages; -using OpenMetaverse; -using OpenMetaverse.Packets; -using OpenSim.Framework; -using OpenSim.Framework.Client; -using Packet=OpenMetaverse.Packets.Packet; -using MXP.Extentions.OpenMetaverseFragments.Proto; -using MXP.Util; -using MXP.Fragments; -using MXP.Common.Proto; -using OpenSim.Region.Framework.Scenes; - -namespace OpenSim.Client.MXP.ClientStack -{ - public class MXPClientView : IClientAPI, IClientCore - { - internal static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - - #region Constants - private Vector3 FORWARD = new Vector3(1, 0, 0); - private Vector3 BACKWARD = new Vector3(-1, 0, 0); - private Vector3 LEFT = new Vector3(0, 1, 0); - private Vector3 RIGHT = new Vector3(0, -1, 0); - private Vector3 UP = new Vector3(0, 0, 1); - private Vector3 DOWN = new Vector3(0, 0, -1); - #endregion - - #region Fields - private readonly Session m_session; - private readonly UUID m_sessionID; - private readonly UUID m_userID; - private readonly IScene m_scene; - private readonly string m_firstName; - private readonly string m_lastName; -// private int m_objectsToSynchronize = 0; -// private int m_objectsSynchronized = -1; - - private Vector3 m_startPosition=new Vector3(128f, 128f, 128f); - #endregion - - #region Properties - - public Session Session - { - get { return m_session; } - } - - public Vector3 StartPos - { - get { return m_startPosition; } - set { m_startPosition = value; } - } - - public UUID AgentId - { - get { return m_userID; } - } - - public UUID SessionId - { - get { return m_sessionID; } - } - - public UUID SecureSessionId - { - get { return m_sessionID; } - } - - public UUID ActiveGroupId - { - get { return UUID.Zero; } - } - - public string ActiveGroupName - { - get { return ""; } - } - - public ulong ActiveGroupPowers - { - get { return 0; } - } - - public ulong GetGroupPowers(UUID groupID) - { - return 0; - } - - public bool IsGroupMember(UUID GroupID) - { - return false; - } - - public string FirstName - { - get { return m_firstName; } - } - - public string LastName - { - get { return m_lastName; } - } - - public IScene Scene - { - get { return m_scene; } - } - - public int NextAnimationSequenceNumber - { - get { return 0; } - } - - public string Name - { - get { return FirstName; } - } - - public bool IsActive - { - get { return Session.SessionState == SessionState.Connected; } - set - { - if (!value) - Stop(); - } - } - - public bool IsLoggingOut - { - get { return false ; } - set { } - } - - #endregion - - #region Constructors - public MXPClientView(Session mxpSession, UUID mxpSessionID, UUID userID, IScene mxpHostBubble, string mxpFirstName, string mxpLastName) - { - this.m_session = mxpSession; - this.m_userID = userID; - this.m_firstName = mxpFirstName; - this.m_lastName = mxpLastName; - this.m_scene = mxpHostBubble; - this.m_sessionID = mxpSessionID; - } - #endregion - - #region MXP Incoming Message Processing - - public void MXPPRocessMessage(Message message) - { - if (message.GetType() == typeof(ModifyRequestMessage)) - { - MXPProcessModifyRequest((ModifyRequestMessage)message); - } - else - { - m_log.Warn("[MXP ClientStack] Received messaged unhandled: " + message); - } - } - - private void MXPProcessModifyRequest(ModifyRequestMessage modifyRequest) - { - ObjectFragment objectFragment=modifyRequest.ObjectFragment; - if (objectFragment.ObjectId == m_userID.Guid) - { - OmAvatarExt avatarExt = modifyRequest.GetExtension(); - - AgentUpdateArgs agentUpdate = new AgentUpdateArgs(); - agentUpdate.AgentID = new UUID(objectFragment.ObjectId); - agentUpdate.SessionID = m_sessionID; - agentUpdate.State = (byte)avatarExt.State; - - Quaternion avatarOrientation = FromOmQuaternion(objectFragment.Orientation); - if (avatarOrientation.X == 0 && avatarOrientation.Y == 0 && avatarOrientation.Z == 0 && avatarOrientation.W == 0) - { - avatarOrientation = Quaternion.Identity; - } - Vector3 avatarLocation=FromOmVector(objectFragment.Location); - - if (avatarExt.MovementDirection != null) - { - Vector3 direction = FromOmVector(avatarExt.MovementDirection); - - direction = direction * Quaternion.Inverse(avatarOrientation); - - if ((direction - FORWARD).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_AT_POS; - } - if ((direction - BACKWARD).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_AT_NEG; - } - if ((direction - LEFT).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_LEFT_POS; - } - if ((direction - RIGHT).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_LEFT_NEG; - } - if ((direction - UP).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_UP_POS; - } - if ((direction - DOWN).Length() < 0.5) - { - agentUpdate.ControlFlags += (uint)AgentManager.ControlFlags.AGENT_CONTROL_UP_NEG; - } - - } - if (avatarExt.TargetOrientation != null) - { - agentUpdate.BodyRotation = FromOmQuaternion(avatarExt.TargetOrientation); - } - else - { - agentUpdate.BodyRotation = FromOmQuaternion(objectFragment.Orientation); - } - - if (avatarExt.Body != null) - { - foreach (OmBipedBoneOrientation boneOrientation in avatarExt.Body.BipedBoneOrientations) - { - if (boneOrientation.Bone == OmBipedBones.Head) - { - agentUpdate.HeadRotation = FromOmQuaternion(boneOrientation.Orientation); - } - } - } - else - { - agentUpdate.HeadRotation = Quaternion.Identity; - } - - if (avatarExt.Camera != null) - { - Quaternion cameraOrientation = FromOmQuaternion(avatarExt.Camera.Orientation); - agentUpdate.CameraCenter = FromOmVector(avatarExt.Camera.Location); - agentUpdate.CameraAtAxis = FORWARD * cameraOrientation; - agentUpdate.CameraLeftAxis = LEFT * cameraOrientation; - agentUpdate.CameraUpAxis = UP * cameraOrientation; - } - else - { - agentUpdate.CameraCenter = avatarLocation; - agentUpdate.CameraAtAxis = FORWARD * avatarOrientation; - agentUpdate.CameraLeftAxis = LEFT * avatarOrientation; - agentUpdate.CameraUpAxis = UP * avatarOrientation; - } - - OnAgentUpdate(this, agentUpdate); - - ModifyResponseMessage modifyResponse = new ModifyResponseMessage(); - modifyResponse.FailureCode = MxpResponseCodes.SUCCESS; - modifyResponse.RequestMessageId = modifyRequest.MessageId; - m_session.Send(modifyResponse); - } - else - { - ModifyResponseMessage modifyResponse = new ModifyResponseMessage(); - modifyResponse.FailureCode = MxpResponseCodes.UNAUTHORIZED_OPERATION; - modifyResponse.RequestMessageId = modifyRequest.MessageId; - m_session.Send(modifyResponse); - } - } - - #endregion - - #region MXP Outgoing Message Processing - -// private void MXPSendPrimitive(uint localID, UUID ownerID, Vector3 acc, Vector3 rvel, PrimitiveBaseShape primShape, Vector3 pos, UUID objectID, Vector3 vel, Quaternion rotation, uint flags, string text, byte[] textColor, uint parentID, byte[] particleSystem, byte clickAction, byte material, byte[] textureanim) -// { -// String typeName = ToOmType(primShape.PCode); -// m_log.Info("[MXP ClientStack] Transmitting Primitive" + typeName); -// -// PerceptionEventMessage pe = new PerceptionEventMessage(); -// pe.ObjectFragment.ObjectId = objectID.Guid; -// -// pe.ObjectFragment.ParentObjectId = Guid.Empty; -// -// // Resolving parent UUID. -// OpenSim.Region.Framework.Scenes.Scene scene = (OpenSim.Region.Framework.Scenes.Scene)Scene; -// if (scene.Entities.ContainsKey(parentID)) -// { -// pe.ObjectFragment.ParentObjectId = scene.Entities[parentID].UUID.Guid; -// } -// -// pe.ObjectFragment.ObjectIndex = localID; -// pe.ObjectFragment.ObjectName = typeName + " Object"; -// pe.ObjectFragment.OwnerId = ownerID.Guid; -// pe.ObjectFragment.TypeId = Guid.Empty; -// pe.ObjectFragment.TypeName = typeName; -// pe.ObjectFragment.Acceleration = ToOmVector(acc); -// pe.ObjectFragment.AngularAcceleration=new MsdQuaternion4f(); -// pe.ObjectFragment.AngularVelocity = ToOmQuaternion(rvel); -// pe.ObjectFragment.BoundingSphereRadius = primShape.Scale.Length(); -// -// pe.ObjectFragment.Location = ToOmVector(pos); -// -// pe.ObjectFragment.Mass = 1.0f; -// pe.ObjectFragment.Orientation = ToOmQuaternion(rotation); -// pe.ObjectFragment.Velocity =ToOmVector(vel); -// -// OmSlPrimitiveExt ext = new OmSlPrimitiveExt(); -// -// if (!((primShape.PCode == (byte)PCode.NewTree) || (primShape.PCode == (byte)PCode.Tree) || (primShape.PCode == (byte)PCode.Grass))) -// { -// -// ext.PathBegin = primShape.PathBegin; -// ext.PathEnd = primShape.PathEnd; -// ext.PathScaleX = primShape.PathScaleX; -// ext.PathScaleY = primShape.PathScaleY; -// ext.PathShearX = primShape.PathShearX; -// ext.PathShearY = primShape.PathShearY; -// ext.PathSkew = primShape.PathSkew; -// ext.ProfileBegin = primShape.ProfileBegin; -// ext.ProfileEnd = primShape.ProfileEnd; -// ext.PathCurve = primShape.PathCurve; -// ext.ProfileCurve = primShape.ProfileCurve; -// ext.ProfileHollow = primShape.ProfileHollow; -// ext.PathRadiusOffset = primShape.PathRadiusOffset; -// ext.PathRevolutions = primShape.PathRevolutions; -// ext.PathTaperX = primShape.PathTaperX; -// ext.PathTaperY = primShape.PathTaperY; -// ext.PathTwist = primShape.PathTwist; -// ext.PathTwistBegin = primShape.PathTwistBegin; -// -// -// } -// -// ext.UpdateFlags = flags; -// ext.ExtraParams = primShape.ExtraParams; -// ext.State = primShape.State; -// ext.TextureEntry = primShape.TextureEntry; -// ext.TextureAnim = textureanim; -// ext.Scale = ToOmVector(primShape.Scale); -// ext.Text = text; -// ext.TextColor = ToOmColor(textColor); -// ext.PSBlock = particleSystem; -// ext.ClickAction = clickAction; -// ext.Material = material; -// -// pe.SetExtension(ext); -// -// Session.Send(pe); -// -// if (m_objectsSynchronized != -1) -// { -// m_objectsSynchronized++; -// -// if (m_objectsToSynchronize >= m_objectsSynchronized) -// { -// SynchronizationEndEventMessage synchronizationEndEventMessage = new SynchronizationEndEventMessage(); -// Session.Send(synchronizationEndEventMessage); -// m_objectsSynchronized = -1; -// } -// } -// } - - public void MXPSendAvatarData(string participantName, UUID ownerID, UUID parentId, UUID avatarID, uint avatarLocalID, Vector3 position, Quaternion rotation) - { - m_log.Info("[MXP ClientStack] Transmitting Avatar Data " + participantName); - - PerceptionEventMessage pe = new PerceptionEventMessage(); - - pe.ObjectFragment.ObjectId = avatarID.Guid; - pe.ObjectFragment.ParentObjectId = parentId.Guid; - pe.ObjectFragment.ObjectIndex = avatarLocalID; - pe.ObjectFragment.ObjectName = participantName; - pe.ObjectFragment.OwnerId = ownerID.Guid; - pe.ObjectFragment.TypeId = Guid.Empty; - pe.ObjectFragment.TypeName = "Avatar"; - pe.ObjectFragment.Acceleration = new MsdVector3f(); - pe.ObjectFragment.AngularAcceleration = new MsdQuaternion4f(); - pe.ObjectFragment.AngularVelocity = new MsdQuaternion4f(); - - pe.ObjectFragment.BoundingSphereRadius = 1.0f; // TODO Fill in appropriate value - - pe.ObjectFragment.Location = ToOmVector(position); - - pe.ObjectFragment.Mass = 1.0f; // TODO Fill in appropriate value - pe.ObjectFragment.Orientation = ToOmQuaternion(rotation); - pe.ObjectFragment.Velocity = new MsdVector3f(); - - Session.Send(pe); - } - - public void MXPSendTerrain(float[] map) - { - m_log.Info("[MXP ClientStack] Transmitting terrain for " + m_scene.RegionInfo.RegionName); - - PerceptionEventMessage pe = new PerceptionEventMessage(); - - // Hacking terrain object uuid to zero and index to hashcode of regionuuid - pe.ObjectFragment.ObjectId = m_scene.RegionInfo.RegionSettings.RegionUUID.Guid; - pe.ObjectFragment.ObjectIndex = (uint)(m_scene.RegionInfo.RegionSettings.RegionUUID.GetHashCode() + ((long)int.MaxValue) / 2); - pe.ObjectFragment.ParentObjectId = UUID.Zero.Guid; - pe.ObjectFragment.ObjectName = "Terrain of " + m_scene.RegionInfo.RegionName; - pe.ObjectFragment.OwnerId = m_scene.RegionInfo.EstateSettings.EstateOwner.Guid; - pe.ObjectFragment.TypeId = Guid.Empty; - pe.ObjectFragment.TypeName = "Terrain"; - pe.ObjectFragment.Acceleration = new MsdVector3f(); - pe.ObjectFragment.AngularAcceleration = new MsdQuaternion4f(); - pe.ObjectFragment.AngularVelocity = new MsdQuaternion4f(); - pe.ObjectFragment.BoundingSphereRadius = 128f; - - pe.ObjectFragment.Location = new MsdVector3f(); - - pe.ObjectFragment.Mass = 1.0f; - pe.ObjectFragment.Orientation = new MsdQuaternion4f(); - pe.ObjectFragment.Velocity = new MsdVector3f(); - - OmBitmapTerrainExt terrainExt = new OmBitmapTerrainExt(); - terrainExt.Width = 256; - terrainExt.Height = 256; - terrainExt.WaterLevel = (float) m_scene.RegionInfo.RegionSettings.WaterHeight; - terrainExt.Offset = 0; - terrainExt.Scale = 10; - terrainExt.HeightMap = CompressUtil.CompressHeightMap(map, 0, 10); - - pe.SetExtension(terrainExt); - - Session.Send(pe); - } - - public void MXPSendSynchronizationBegin(int objectCount) - { -// m_objectsToSynchronize = objectCount; -// m_objectsSynchronized = 0; - SynchronizationBeginEventMessage synchronizationBeginEventMessage = new SynchronizationBeginEventMessage(); - synchronizationBeginEventMessage.ObjectCount = (uint)objectCount; - Session.Send(synchronizationBeginEventMessage); - } - - #endregion - - #region MXP Conversions - - private MsdVector3f ToOmVector(Vector3 value) - { - MsdVector3f encodedValue = new MsdVector3f(); - encodedValue.X = value.X; - encodedValue.Y = value.Y; - encodedValue.Z = value.Z; - return encodedValue; - } - - private MsdQuaternion4f ToOmQuaternion(Vector3 value) - { - Quaternion quaternion=Quaternion.CreateFromEulers(value); - MsdQuaternion4f encodedValue = new MsdQuaternion4f(); - encodedValue.X = quaternion.X; - encodedValue.Y = quaternion.Y; - encodedValue.Z = quaternion.Z; - encodedValue.W = quaternion.W; - return encodedValue; - } - - private MsdQuaternion4f ToOmQuaternion(Quaternion value) - { - MsdQuaternion4f encodedValue = new MsdQuaternion4f(); - encodedValue.X = value.X; - encodedValue.Y = value.Y; - encodedValue.Z = value.Z; - encodedValue.W = value.W; - return encodedValue; - } - - private Vector3 FromOmVector(MsdVector3f vector) - { - return new Vector3(vector.X, vector.Y, vector.Z); - } - -// private Vector3 FromOmVector(float[] vector) -// { -// return new Vector3(vector[0], vector[1], vector[2]); -// } - - private Quaternion FromOmQuaternion(MsdQuaternion4f quaternion) - { - return new Quaternion(quaternion.X, quaternion.Y, quaternion.Z, quaternion.W); - } - -// private Quaternion FromOmQuaternion(float[] quaternion) -// { -// return new Quaternion(quaternion[0], quaternion[1], quaternion[2], quaternion[3]); -// } - - private MsdColor4f ToOmColor(byte[] value) - { - MsdColor4f encodedValue = new MsdColor4f(); - encodedValue.R = value[0]; - encodedValue.G = value[1]; - encodedValue.B = value[2]; - encodedValue.A = value[3]; - return encodedValue; - } - - private string ToOmType(byte value) - { - if (value == (byte)PCodeEnum.Avatar) - { - return "Avatar"; - } - if (value == (byte)PCodeEnum.Grass) - { - return "Grass"; - } - if (value == (byte)PCodeEnum.NewTree) - { - return "NewTree"; - } - if (value == (byte)PCodeEnum.ParticleSystem) - { - return "ParticleSystem"; - } - if (value == (byte)PCodeEnum.Primitive) - { - return "Primitive"; - } - if (value == (byte)PCodeEnum.Tree) - { - return "Tree"; - } - throw new Exception("Unsupported PCode value: " + value); - } - - #endregion - - #region OpenSim Event Handlers - - #pragma warning disable 67 - public event GenericMessage OnGenericMessage; - public event ImprovedInstantMessage OnInstantMessage; - public event ChatMessage OnChatFromClient; - public event TextureRequest OnRequestTexture; - public event RezObject OnRezObject; - public event ModifyTerrain OnModifyTerrain; - public event BakeTerrain OnBakeTerrain; - public event EstateChangeInfo OnEstateChangeInfo; - public event SetAppearance OnSetAppearance; - public event AvatarNowWearing OnAvatarNowWearing; - public event RezSingleAttachmentFromInv OnRezSingleAttachmentFromInv; - public event RezMultipleAttachmentsFromInv OnRezMultipleAttachmentsFromInv; - public event UUIDNameRequest OnDetachAttachmentIntoInv; - public event ObjectAttach OnObjectAttach; - public event ObjectDeselect OnObjectDetach; - public event ObjectDrop OnObjectDrop; - public event StartAnim OnStartAnim; - public event StopAnim OnStopAnim; - public event LinkObjects OnLinkObjects; - public event DelinkObjects OnDelinkObjects; - public event RequestMapBlocks OnRequestMapBlocks; - public event RequestMapName OnMapNameRequest; - public event TeleportLocationRequest OnTeleportLocationRequest; - public event DisconnectUser OnDisconnectUser; - public event RequestAvatarProperties OnRequestAvatarProperties; - public event SetAlwaysRun OnSetAlwaysRun; - public event TeleportLandmarkRequest OnTeleportLandmarkRequest; - public event DeRezObject OnDeRezObject; - public event Action OnRegionHandShakeReply; - public event GenericCall1 OnRequestWearables; - public event GenericCall1 OnCompleteMovementToRegion; - public event UpdateAgent OnPreAgentUpdate; - public event UpdateAgent OnAgentUpdate; - public event AgentRequestSit OnAgentRequestSit; - public event AgentSit OnAgentSit; - public event AvatarPickerRequest OnAvatarPickerRequest; - public event Action OnRequestAvatarsData; - public event AddNewPrim OnAddPrim; - public event FetchInventory OnAgentDataUpdateRequest; - public event TeleportLocationRequest OnSetStartLocationRequest; - public event RequestGodlikePowers OnRequestGodlikePowers; - public event GodKickUser OnGodKickUser; - public event ObjectDuplicate OnObjectDuplicate; - public event ObjectDuplicateOnRay OnObjectDuplicateOnRay; - public event GrabObject OnGrabObject; - public event DeGrabObject OnDeGrabObject; - public event MoveObject OnGrabUpdate; - public event SpinStart OnSpinStart; - public event SpinObject OnSpinUpdate; - public event SpinStop OnSpinStop; - public event UpdateShape OnUpdatePrimShape; - public event ObjectExtraParams OnUpdateExtraParams; - public event ObjectRequest OnObjectRequest; - public event ObjectSelect OnObjectSelect; - public event ObjectDeselect OnObjectDeselect; - public event GenericCall7 OnObjectDescription; - public event GenericCall7 OnObjectName; - public event GenericCall7 OnObjectClickAction; - public event GenericCall7 OnObjectMaterial; - public event RequestObjectPropertiesFamily OnRequestObjectPropertiesFamily; - public event UpdatePrimFlags OnUpdatePrimFlags; - public event UpdatePrimTexture OnUpdatePrimTexture; - public event UpdateVector OnUpdatePrimGroupPosition; - public event UpdateVector OnUpdatePrimSinglePosition; - public event UpdatePrimRotation OnUpdatePrimGroupRotation; - public event UpdatePrimSingleRotationPosition OnUpdatePrimSingleRotationPosition; - public event UpdatePrimSingleRotation OnUpdatePrimSingleRotation; - public event UpdatePrimGroupRotation OnUpdatePrimGroupMouseRotation; - public event UpdateVector OnUpdatePrimScale; - public event UpdateVector OnUpdatePrimGroupScale; - public event StatusChange OnChildAgentStatus; - public event GenericCall2 OnStopMovement; - public event Action OnRemoveAvatar; - public event ObjectPermissions OnObjectPermissions; - public event CreateNewInventoryItem OnCreateNewInventoryItem; - public event LinkInventoryItem OnLinkInventoryItem; - public event CreateInventoryFolder OnCreateNewInventoryFolder; - public event UpdateInventoryFolder OnUpdateInventoryFolder; - public event MoveInventoryFolder OnMoveInventoryFolder; - public event FetchInventoryDescendents OnFetchInventoryDescendents; - public event PurgeInventoryDescendents OnPurgeInventoryDescendents; - public event FetchInventory OnFetchInventory; - public event RequestTaskInventory OnRequestTaskInventory; - public event UpdateInventoryItem OnUpdateInventoryItem; - public event CopyInventoryItem OnCopyInventoryItem; - public event MoveInventoryItem OnMoveInventoryItem; - public event RemoveInventoryFolder OnRemoveInventoryFolder; - public event RemoveInventoryItem OnRemoveInventoryItem; - public event UDPAssetUploadRequest OnAssetUploadRequest; - public event XferReceive OnXferReceive; - public event RequestXfer OnRequestXfer; - public event ConfirmXfer OnConfirmXfer; - public event AbortXfer OnAbortXfer; - public event RezScript OnRezScript; - public event UpdateTaskInventory OnUpdateTaskInventory; - public event MoveTaskInventory OnMoveTaskItem; - public event RemoveTaskInventory OnRemoveTaskItem; - public event RequestAsset OnRequestAsset; - public event UUIDNameRequest OnNameFromUUIDRequest; - public event ParcelAccessListRequest OnParcelAccessListRequest; - public event ParcelAccessListUpdateRequest OnParcelAccessListUpdateRequest; - public event ParcelPropertiesRequest OnParcelPropertiesRequest; - public event ParcelDivideRequest OnParcelDivideRequest; - public event ParcelJoinRequest OnParcelJoinRequest; - public event ParcelPropertiesUpdateRequest OnParcelPropertiesUpdateRequest; - public event ParcelSelectObjects OnParcelSelectObjects; - public event ParcelObjectOwnerRequest OnParcelObjectOwnerRequest; - public event ParcelAbandonRequest OnParcelAbandonRequest; - public event ParcelGodForceOwner OnParcelGodForceOwner; - public event ParcelReclaim OnParcelReclaim; - public event ParcelReturnObjectsRequest OnParcelReturnObjectsRequest; - public event ParcelDeedToGroup OnParcelDeedToGroup; - public event RegionInfoRequest OnRegionInfoRequest; - public event EstateCovenantRequest OnEstateCovenantRequest; - public event FriendActionDelegate OnApproveFriendRequest; - public event FriendActionDelegate OnDenyFriendRequest; - public event FriendshipTermination OnTerminateFriendship; - public event GrantUserFriendRights OnGrantUserRights; - public event MoneyTransferRequest OnMoneyTransferRequest; - public event EconomyDataRequest OnEconomyDataRequest; - public event MoneyBalanceRequest OnMoneyBalanceRequest; - public event UpdateAvatarProperties OnUpdateAvatarProperties; - public event ParcelBuy OnParcelBuy; - public event RequestPayPrice OnRequestPayPrice; - public event ObjectSaleInfo OnObjectSaleInfo; - public event ObjectBuy OnObjectBuy; - public event BuyObjectInventory OnBuyObjectInventory; - public event RequestTerrain OnRequestTerrain; - public event RequestTerrain OnUploadTerrain; - public event ObjectIncludeInSearch OnObjectIncludeInSearch; - public event UUIDNameRequest OnTeleportHomeRequest; - public event ScriptAnswer OnScriptAnswer; - public event AgentSit OnUndo; - public event AgentSit OnRedo; - public event LandUndo OnLandUndo; - public event ForceReleaseControls OnForceReleaseControls; - public event GodLandStatRequest OnLandStatRequest; - public event DetailedEstateDataRequest OnDetailedEstateDataRequest; - public event SetEstateFlagsRequest OnSetEstateFlagsRequest; - public event SetEstateTerrainBaseTexture OnSetEstateTerrainBaseTexture; - public event SetEstateTerrainDetailTexture OnSetEstateTerrainDetailTexture; - public event SetEstateTerrainTextureHeights OnSetEstateTerrainTextureHeights; - public event CommitEstateTerrainTextureRequest OnCommitEstateTerrainTextureRequest; - public event SetRegionTerrainSettings OnSetRegionTerrainSettings; - public event EstateRestartSimRequest OnEstateRestartSimRequest; - public event EstateChangeCovenantRequest OnEstateChangeCovenantRequest; - public event UpdateEstateAccessDeltaRequest OnUpdateEstateAccessDeltaRequest; - public event SimulatorBlueBoxMessageRequest OnSimulatorBlueBoxMessageRequest; - public event EstateBlueBoxMessageRequest OnEstateBlueBoxMessageRequest; - public event EstateDebugRegionRequest OnEstateDebugRegionRequest; - public event EstateTeleportOneUserHomeRequest OnEstateTeleportOneUserHomeRequest; - public event EstateTeleportAllUsersHomeRequest OnEstateTeleportAllUsersHomeRequest; - public event UUIDNameRequest OnUUIDGroupNameRequest; - public event RegionHandleRequest OnRegionHandleRequest; - public event ParcelInfoRequest OnParcelInfoRequest; - public event RequestObjectPropertiesFamily OnObjectGroupRequest; - public event ScriptReset OnScriptReset; - public event GetScriptRunning OnGetScriptRunning; - public event SetScriptRunning OnSetScriptRunning; - public event UpdateVector OnAutoPilotGo; - public event TerrainUnacked OnUnackedTerrain; - public event ActivateGesture OnActivateGesture; - public event DeactivateGesture OnDeactivateGesture; - public event ObjectOwner OnObjectOwner; - public event DirPlacesQuery OnDirPlacesQuery; - public event DirFindQuery OnDirFindQuery; - public event DirLandQuery OnDirLandQuery; - public event DirPopularQuery OnDirPopularQuery; - public event DirClassifiedQuery OnDirClassifiedQuery; - public event EventInfoRequest OnEventInfoRequest; - public event ParcelSetOtherCleanTime OnParcelSetOtherCleanTime; - public event MapItemRequest OnMapItemRequest; - public event OfferCallingCard OnOfferCallingCard; - public event AcceptCallingCard OnAcceptCallingCard; - public event DeclineCallingCard OnDeclineCallingCard; - public event SoundTrigger OnSoundTrigger; - public event StartLure OnStartLure; - public event TeleportLureRequest OnTeleportLureRequest; - public event NetworkStats OnNetworkStatsUpdate; - public event ClassifiedInfoRequest OnClassifiedInfoRequest; - public event ClassifiedInfoUpdate OnClassifiedInfoUpdate; - public event ClassifiedDelete OnClassifiedDelete; - public event ClassifiedDelete OnClassifiedGodDelete; - public event EventNotificationAddRequest OnEventNotificationAddRequest; - public event EventNotificationRemoveRequest OnEventNotificationRemoveRequest; - public event EventGodDelete OnEventGodDelete; - public event ParcelDwellRequest OnParcelDwellRequest; - public event UserInfoRequest OnUserInfoRequest; - public event UpdateUserInfo OnUpdateUserInfo; - public event ViewerEffectEventHandler OnViewerEffect; - public event Action OnLogout; - public event Action OnConnectionClosed; - public event RetrieveInstantMessages OnRetrieveInstantMessages; - public event PickDelete OnPickDelete; - public event PickGodDelete OnPickGodDelete; - public event PickInfoUpdate OnPickInfoUpdate; - public event AvatarNotesUpdate OnAvatarNotesUpdate; - public event MuteListRequest OnMuteListRequest; - public event AvatarInterestUpdate OnAvatarInterestUpdate; - public event FindAgentUpdate OnFindAgent; - public event TrackAgentUpdate OnTrackAgent; - public event NewUserReport OnUserReport; - public event SaveStateHandler OnSaveState; - public event GroupAccountSummaryRequest OnGroupAccountSummaryRequest; - public event GroupAccountDetailsRequest OnGroupAccountDetailsRequest; - public event GroupAccountTransactionsRequest OnGroupAccountTransactionsRequest; - public event FreezeUserUpdate OnParcelFreezeUser; - public event EjectUserUpdate OnParcelEjectUser; - public event ParcelBuyPass OnParcelBuyPass; - public event ParcelGodMark OnParcelGodMark; - public event GroupActiveProposalsRequest OnGroupActiveProposalsRequest; - public event GroupVoteHistoryRequest OnGroupVoteHistoryRequest; - public event SimWideDeletesDelegate OnSimWideDeletes; - public event SendPostcard OnSendPostcard; - public event MuteListEntryUpdate OnUpdateMuteListEntry; - public event MuteListEntryRemove OnRemoveMuteListEntry; - public event GodlikeMessage onGodlikeMessage; - public event GodUpdateRegionInfoUpdate OnGodUpdateRegionInfoUpdate; - - public event PlacesQuery OnPlacesQuery; - - #pragma warning restore 67 - - #endregion - - #region OpenSim ClientView Public Methods - // Do we need this? - public bool SendLogoutPacketWhenClosing - { - set { } - } - - public uint CircuitCode - { - get { return m_sessionID.CRC(); } - } - - public IPEndPoint RemoteEndPoint - { - get { return Session.RemoteEndPoint; } - } - - public void SetDebugPacketLevel(int newDebug) - { - //m_debugLevel = newDebug; - } - - public void InPacket(object NewPack) - { - //throw new System.NotImplementedException(); - } - - public void ProcessInPacket(Packet NewPack) - { - //throw new System.NotImplementedException(); - } - - public void OnClean() - { - if (OnLogout != null) - OnLogout(this); - - if (OnConnectionClosed != null) - OnConnectionClosed(this); - } - - public void Close() - { - m_log.Info("[MXP ClientStack] Close Called"); - - // Tell the client to go - SendLogoutPacket(); - - // Let MXPPacketServer clean it up - if (Session.SessionState != SessionState.Disconnected) - { - Session.SetStateDisconnected(); - } - - } - - public void Kick(string message) - { - Close(); - } - - public void Start() - { - Scene.AddNewClient(this); - - // Mimicking LLClientView which gets always set appearance from client. - OpenSim.Region.Framework.Scenes.Scene scene=(OpenSim.Region.Framework.Scenes.Scene)Scene; - AvatarAppearance appearance; - scene.GetAvatarAppearance(this,out appearance); - OnSetAppearance(this, appearance.Texture, (byte[])appearance.VisualParams.Clone()); - } - - public void Stop() - { - // Nor this - } - - public void SendWearables(AvatarWearable[] wearables, int serial) - { - // Need to translate to MXP somehow - } - - public void SendAppearance(UUID agentID, byte[] visualParams, byte[] textureEntry) - { - // Need to translate to MXP somehow - } - - public void SendStartPingCheck(byte seq) - { - // Need to translate to MXP somehow - } - - public void SendKillObject(ulong regionHandle, uint localID) - { - DisappearanceEventMessage de = new DisappearanceEventMessage(); - de.ObjectIndex = localID; - - Session.Send(de); - } - - public void SendAnimations(UUID[] animID, int[] seqs, UUID sourceAgentId, UUID[] objectIDs) - { - // Need to translate to MXP somehow - } - - public void SendRegionHandshake(RegionInfo regionInfo, RegionHandshakeArgs args) - { - m_log.Info("[MXP ClientStack] Completing Handshake to Region"); - - if (OnRegionHandShakeReply != null) - { - OnRegionHandShakeReply(this); - } - - if (OnCompleteMovementToRegion != null) - { - OnCompleteMovementToRegion(this); - } - - // Need to translate to MXP somehow - } - - public void SendChatMessage(string message, byte type, Vector3 fromPos, string fromName, UUID fromAgentID, byte source, byte audible) - { - ActionEventMessage chatActionEvent = new ActionEventMessage(); - chatActionEvent.ActionFragment.ActionName = "Chat"; - chatActionEvent.ActionFragment.SourceObjectId = fromAgentID.Guid; - chatActionEvent.ActionFragment.ObservationRadius = 180.0f; - chatActionEvent.ActionFragment.ExtensionDialect = "TEXT"; - chatActionEvent.SetPayloadData(Util.UTF8.GetBytes(message)); - - Session.Send(chatActionEvent); - } - - public void SendInstantMessage(GridInstantMessage im) - { - // Need to translate to MXP somehow - } - - public void SendGenericMessage(string method, List message) - { - } - - public void SendGenericMessage(string method, List message) - { - // Need to translate to MXP somehow - } - - public void SendLayerData(float[] map) - { - MXPSendTerrain(map); - } - - public void SendLayerData(int px, int py, float[] map) - { - } - - public void SendWindData(Vector2[] windSpeeds) - { - // Need to translate to MXP somehow - } - - public void SendCloudData(float[] cloudCover) - { - // Need to translate to MXP somehow - } - - public void MoveAgentIntoRegion(RegionInfo regInfo, Vector3 pos, Vector3 look) - { - //throw new System.NotImplementedException(); - } - - public void InformClientOfNeighbour(ulong neighbourHandle, IPEndPoint neighbourExternalEndPoint) - { - //throw new System.NotImplementedException(); - } - - public AgentCircuitData RequestClientInfo() - { - AgentCircuitData clientinfo = new AgentCircuitData(); - clientinfo.AgentID = AgentId; - clientinfo.Appearance = new AvatarAppearance(); - clientinfo.BaseFolder = UUID.Zero; - clientinfo.CapsPath = ""; - clientinfo.child = false; - clientinfo.ChildrenCapSeeds = new Dictionary(); - clientinfo.circuitcode = CircuitCode; - clientinfo.firstname = FirstName; - clientinfo.InventoryFolder = UUID.Zero; - clientinfo.lastname = LastName; - clientinfo.SecureSessionID = SecureSessionId; - clientinfo.SessionID = SessionId; - clientinfo.startpos = StartPos; - - return clientinfo; - } - - public void CrossRegion(ulong newRegionHandle, Vector3 pos, Vector3 lookAt, IPEndPoint newRegionExternalEndPoint, string capsURL) - { - // TODO: We'll want to get this one working. - // Need to translate to MXP somehow - } - - public void SendMapBlock(List mapBlocks, uint flag) - { - // Need to translate to MXP somehow - } - - public void SendLocalTeleport(Vector3 position, Vector3 lookAt, uint flags) - { - //throw new System.NotImplementedException(); - } - - public void SendRegionTeleport(ulong regionHandle, byte simAccess, IPEndPoint regionExternalEndPoint, uint locationID, uint flags, string capsURL) - { - // Need to translate to MXP somehow - } - - public void SendTeleportFailed(string reason) - { - // Need to translate to MXP somehow - } - - public void SendTeleportStart(uint flags) - { - // Need to translate to MXP somehow - } - - public void SendTeleportProgress(uint flags, string message) - { - } - - public void SendMoneyBalance(UUID transaction, bool success, byte[] description, int balance) - { - // Need to translate to MXP somehow - } - - public void SendPayPrice(UUID objectID, int[] payPrice) - { - // Need to translate to MXP somehow - } - - public void SendCoarseLocationUpdate(List users, List CoarseLocations) - { - // Minimap function, not used. - } - - public void SetChildAgentThrottle(byte[] throttle) - { - // Need to translate to MXP somehow - } - - public void SendAvatarDataImmediate(ISceneEntity avatar) - { - //ScenePresence presence=((Scene)this.Scene).GetScenePresence(avatarID); - ScenePresence presence = (ScenePresence)avatar; - UUID ownerID = presence.UUID; - MXPSendAvatarData(presence.Firstname + " " + presence.Lastname, ownerID, UUID.Zero, presence.UUID, presence.LocalId, presence.AbsolutePosition, presence.Rotation); - } - - public void SendPrimUpdate(ISceneEntity entity, PrimUpdateFlags updateFlags) - { - //MovementEventMessage me = new MovementEventMessage(); - //me.ObjectIndex = data.LocalID; - //me.Location = ToOmVector(data.Position); - //me.Orientation = ToOmQuaternion(data.Rotation); - - //MXPSendPrimitive(data.localID, data.ownerID, data.acc, data.rvel, data.primShape, data.pos, data.objectID, data.vel, - // data.rotation, (uint)data.flags, data.text, data.color, data.parentID, data.particleSystem, data.clickAction, - // data.material, data.textureanim); - - //Session.Send(me); - - throw new System.NotImplementedException(); - } - - public void ReprioritizeUpdates() - { - } - - public void FlushPrimUpdates() - { - } - - public void SendInventoryFolderDetails(UUID ownerID, UUID folderID, List items, List folders, int version, bool fetchFolders, bool fetchItems) - { - // Need to translate to MXP somehow - } - - public void SendInventoryItemDetails(UUID ownerID, InventoryItemBase item) - { - // Need to translate to MXP somehow - } - - public void SendInventoryItemCreateUpdate(InventoryItemBase Item, uint callbackId) - { - // Need to translate to MXP somehow - } - - public void SendRemoveInventoryItem(UUID itemID) - { - // Need to translate to MXP somehow - } - - public void SendTakeControls(int controls, bool passToAgent, bool TakeControls) - { - // Need to translate to MXP somehow - } - - public void SendTaskInventory(UUID taskID, short serial, byte[] fileName) - { - // Need to translate to MXP somehow - } - - public void SendBulkUpdateInventory(InventoryNodeBase node) - { - // Need to translate to MXP somehow - } - - public void SendXferPacket(ulong xferID, uint packet, byte[] data) - { - // SL Specific, Ignore. (Remove from IClient) - } - - public void SendAbortXferPacket(ulong xferID) - { - - } - - - public void SendEconomyData(float EnergyEfficiency, int ObjectCapacity, int ObjectCount, int PriceEnergyUnit, int PriceGroupCreate, int PriceObjectClaim, float PriceObjectRent, float PriceObjectScaleFactor, int PriceParcelClaim, float PriceParcelClaimFactor, int PriceParcelRent, int PricePublicObjectDecay, int PricePublicObjectDelete, int PriceRentLight, int PriceUpload, int TeleportMinPrice, float TeleportPriceExponent) - { - // SL Specific, Ignore. (Remove from IClient) - } - - public void SendAvatarPickerReply(AvatarPickerReplyAgentDataArgs AgentData, List Data) - { - // Need to translate to MXP somehow - } - - public void SendAgentDataUpdate(UUID agentid, UUID activegroupid, string firstname, string lastname, ulong grouppowers, string groupname, string grouptitle) - { - // Need to translate to MXP somehow - // TODO: This may need doing - involves displaying the users avatar name - } - - public void SendPreLoadSound(UUID objectID, UUID ownerID, UUID soundID) - { - // Need to translate to MXP somehow - } - - public void SendPlayAttachedSound(UUID soundID, UUID objectID, UUID ownerID, float gain, byte flags) - { - // Need to translate to MXP somehow - } - - public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) - { - // Need to translate to MXP somehow - } - - public void SendAttachedSoundGainChange(UUID objectID, float gain) - { - // Need to translate to MXP somehow - } - - public void SendNameReply(UUID profileId, string firstname, string lastname) - { - // SL Specific - } - - public void SendAlertMessage(string message) - { - SendChatMessage(message, 0, Vector3.Zero, "System", UUID.Zero, 0, 0); - } - - public void SendAgentAlertMessage(string message, bool modal) - { - SendChatMessage(message, 0, Vector3.Zero, "System" + (modal ? " Notice" : ""), UUID.Zero, 0, 0); - } - - public void SendLoadURL(string objectname, UUID objectID, UUID ownerID, bool groupOwned, string message, string url) - { - // TODO: Probably can do this better - SendChatMessage("Please visit: " + url, 0, Vector3.Zero, objectname, UUID.Zero, 0, 0); - } - - public void SendDialog(string objectname, UUID objectID, string ownerFirstName, string ownerLastName, string msg, UUID textureID, int ch, string[] buttonlabels) - { - // TODO: Probably can do this better - SendChatMessage("Dialog: " + msg, 0, Vector3.Zero, objectname, UUID.Zero, 0, 0); - } - - public bool AddMoney(int debit) - { - SendChatMessage("You were paid: " + debit, 0, Vector3.Zero, "System", UUID.Zero, 0, 0); - return true; - } - - public void SendSunPos(Vector3 sunPos, Vector3 sunVel, ulong CurrentTime, uint SecondsPerSunCycle, uint SecondsPerYear, float OrbitalPosition) - { - // Need to translate to MXP somehow - // Send a light object? - } - - public void SendViewerEffect(ViewerEffectPacket.EffectBlock[] effectBlocks) - { - // Need to translate to MXP somehow - } - - public void SendViewerTime(int phase) - { - // Need to translate to MXP somehow - } - - public UUID GetDefaultAnimation(string name) - { - return UUID.Zero; - } - - public void SendAvatarProperties(UUID avatarID, string aboutText, string bornOn, byte[] charterMember, string flAbout, uint flags, UUID flImageID, UUID imageID, string profileURL, UUID partnerID) - { - // Need to translate to MXP somehow - } - - public void SendScriptQuestion(UUID taskID, string taskName, string ownerName, UUID itemID, int question) - { - // Need to translate to MXP somehow - } - - public void SendHealth(float health) - { - // Need to translate to MXP somehow - } - - public void SendEstateList(UUID invoice, int code, UUID[] Data, uint estateID) - { - // Need to translate to MXP somehow - } - - public void SendBannedUserList(UUID invoice, EstateBan[] banlist, uint estateID) - { - // Need to translate to MXP somehow - } - - public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) - { - // Need to translate to MXP somehow - } - - public void SendEstateCovenantInformation(UUID covenant) - { - // Need to translate to MXP somehow - } - - public void SendDetailedEstateData(UUID invoice, string estateName, uint estateID, uint parentEstate, uint estateFlags, uint sunPosition, UUID covenant, string abuseEmail, UUID estateOwner) - { - // Need to translate to MXP somehow - } - - public void SendLandProperties(int sequence_id, bool snap_selection, int request_result, ILandObject lo, float simObjectBonusFactor, int parcelObjectCapacity, int simObjectCapacity, uint regionFlags) - { - // Need to translate to MXP somehow - } - - public void SendLandAccessListData(List avatars, uint accessFlag, int localLandID) - { - // Need to translate to MXP somehow - } - - public void SendForceClientSelectObjects(List objectIDs) - { - // Need to translate to MXP somehow - } - - public void SendLandObjectOwners(LandData land, List groups, Dictionary ownersAndCount) - { - // Need to translate to MXP somehow - } - - public void SendCameraConstraint(Vector4 ConstraintPlane) - { - - } - - public void SendLandParcelOverlay(byte[] data, int sequence_id) - { - // Need to translate to MXP somehow - } - - public void SendParcelMediaCommand(uint flags, ParcelMediaCommandEnum command, float time) - { - // Need to translate to MXP somehow - } - - public void SendParcelMediaUpdate(string mediaUrl, UUID mediaTextureID, byte autoScale, string mediaType, string mediaDesc, int mediaWidth, int mediaHeight, byte mediaLoop) - { - // Need to translate to MXP somehow - } - - public void SendAssetUploadCompleteMessage(sbyte AssetType, bool Success, UUID AssetFullID) - { - // Need to translate to MXP somehow - } - - public void SendConfirmXfer(ulong xferID, uint PacketID) - { - // Need to translate to MXP somehow - } - - public void SendXferRequest(ulong XferID, short AssetType, UUID vFileID, byte FilePath, byte[] FileName) - { - // Need to translate to MXP somehow - } - - public void SendInitiateDownload(string simFileName, string clientFileName) - { - // Need to translate to MXP somehow - } - - public void SendImageFirstPart(ushort numParts, UUID ImageUUID, uint ImageSize, byte[] ImageData, byte imageCodec) - { - // Need to translate to MXP somehow - } - - public void SendImageNextPart(ushort partNumber, UUID imageUuid, byte[] imageData) - { - // Need to translate to MXP somehow - } - - public void SendImageNotFound(UUID imageid) - { - // Need to translate to MXP somehow - } - - public void SendShutdownConnectionNotice() - { - // Need to translate to MXP somehow - } - - public void SendSimStats(SimStats stats) - { - // Need to translate to MXP somehow - } - - public void SendObjectPropertiesFamilyData(ISceneEntity Entity, uint RequestFlags) - { - //throw new System.NotImplementedException(); - } - - public void SendObjectPropertiesReply(ISceneEntity entity) - { - //throw new System.NotImplementedException(); - } - - public void SendAgentOffline(UUID[] agentIDs) - { - // Need to translate to MXP somehow (Friends List) - } - - public void SendAgentOnline(UUID[] agentIDs) - { - // Need to translate to MXP somehow (Friends List) - } - - public void SendSitResponse(UUID TargetID, Vector3 OffsetPos, Quaternion SitOrientation, bool autopilot, Vector3 CameraAtOffset, Vector3 CameraEyeOffset, bool ForceMouseLook) - { - // Need to translate to MXP somehow - } - - public void SendAdminResponse(UUID Token, uint AdminLevel) - { - // Need to translate to MXP somehow - } - - public void SendGroupMembership(GroupMembershipData[] GroupMembership) - { - // Need to translate to MXP somehow - } - - public void SendGroupNameReply(UUID groupLLUID, string GroupName) - { - // Need to translate to MXP somehow - } - - public void SendJoinGroupReply(UUID groupID, bool success) - { - // Need to translate to MXP somehow - } - - public void SendEjectGroupMemberReply(UUID agentID, UUID groupID, bool success) - { - // Need to translate to MXP somehow - } - - public void SendLeaveGroupReply(UUID groupID, bool success) - { - // Need to translate to MXP somehow - } - - public void SendLandStatReply(uint reportType, uint requestFlags, uint resultCount, LandStatReportItem[] lsrpia) - { - // Need to translate to MXP somehow - } - - public void SendScriptRunningReply(UUID objectID, UUID itemID, bool running) - { - // Need to translate to MXP somehow - } - - public void SendAsset(AssetRequestToClient req) - { - // Need to translate to MXP somehow - } - - public void SendTexture(AssetBase TextureAsset) - { - // Need to translate to MXP somehow - } - - public byte[] GetThrottlesPacked(float multiplier) - { - // LL Specific, get out of IClientAPI - - const int singlefloat = 4; - float tResend = multiplier; - float tLand = multiplier; - float tWind = multiplier; - float tCloud = multiplier; - float tTask = multiplier; - float tTexture = multiplier; - float tAsset = multiplier; - - byte[] throttles = new byte[singlefloat * 7]; - int i = 0; - Buffer.BlockCopy(BitConverter.GetBytes(tResend), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tLand), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tWind), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tCloud), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tTask), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tTexture), 0, throttles, singlefloat * i, singlefloat); - i++; - Buffer.BlockCopy(BitConverter.GetBytes(tAsset), 0, throttles, singlefloat * i, singlefloat); - - return throttles; - } - - public void SendBlueBoxMessage(UUID FromAvatarID, string FromAvatarName, string Message) - { - SendChatMessage(Message, 0, Vector3.Zero, FromAvatarName, UUID.Zero, 0, 0); - } - - public void SendLogoutPacket() - { - LeaveRequestMessage lrm = new LeaveRequestMessage(); - Session.Send(lrm); - } - - public EndPoint GetClientEP() - { - return null; - } - - public ClientInfo GetClientInfo() - { - return null; - //throw new System.NotImplementedException(); - } - - public void SetClientInfo(ClientInfo info) - { - //throw new System.NotImplementedException(); - } - - public void SetClientOption(string option, string value) - { - // Need to translate to MXP somehow - } - - public string GetClientOption(string option) - { - // Need to translate to MXP somehow - return ""; - } - - public void Terminate() - { - Close(); - } - - public void SendSetFollowCamProperties(UUID objectID, SortedDictionary parameters) - { - // Need to translate to MXP somehow - } - - public void SendClearFollowCamProperties(UUID objectID) - { - // Need to translate to MXP somehow - } - - public void SendRegionHandle(UUID regoinID, ulong handle) - { - // Need to translate to MXP somehow - } - - public void SendParcelInfo(RegionInfo info, LandData land, UUID parcelID, uint x, uint y) - { - // Need to translate to MXP somehow - } - - public void SendScriptTeleportRequest(string objName, string simName, Vector3 pos, Vector3 lookAt) - { - // Need to translate to MXP somehow - } - - public void SendDirPlacesReply(UUID queryID, DirPlacesReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirPeopleReply(UUID queryID, DirPeopleReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirEventsReply(UUID queryID, DirEventsReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirGroupsReply(UUID queryID, DirGroupsReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirClassifiedReply(UUID queryID, DirClassifiedReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirLandReply(UUID queryID, DirLandReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendDirPopularReply(UUID queryID, DirPopularReplyData[] data) - { - // Need to translate to MXP somehow - } - - public void SendEventInfoReply(EventData info) - { - // Need to translate to MXP somehow - } - - public void SendMapItemReply(mapItemReply[] replies, uint mapitemtype, uint flags) - { - // Need to translate to MXP somehow - } - - public void SendAvatarGroupsReply(UUID avatarID, GroupMembershipData[] data) - { - // Need to translate to MXP somehow - } - - public void SendOfferCallingCard(UUID srcID, UUID transactionID) - { - // Need to translate to MXP somehow - } - - public void SendAcceptCallingCard(UUID transactionID) - { - // Need to translate to MXP somehow - } - - public void SendDeclineCallingCard(UUID transactionID) - { - // Need to translate to MXP somehow - } - - public void SendTerminateFriend(UUID exFriendID) - { - // Need to translate to MXP somehow - } - - public void SendAvatarClassifiedReply(UUID targetID, UUID[] classifiedID, string[] name) - { - // Need to translate to MXP somehow - } - - public void SendClassifiedInfoReply(UUID classifiedID, UUID creatorID, uint creationDate, uint expirationDate, uint category, string name, string description, UUID parcelID, uint parentEstate, UUID snapshotID, string simName, Vector3 globalPos, string parcelName, byte classifiedFlags, int price) - { - // Need to translate to MXP somehow - } - - public void SendAgentDropGroup(UUID groupID) - { - // Need to translate to MXP somehow - } - - public void SendAvatarNotesReply(UUID targetID, string text) - { - // Need to translate to MXP somehow - } - - public void SendAvatarPicksReply(UUID targetID, Dictionary picks) - { - // Need to translate to MXP somehow - } - - public void SendAvatarClassifiedReply(UUID targetID, Dictionary classifieds) - { - // Need to translate to MXP somehow - } - - public void SendParcelDwellReply(int localID, UUID parcelID, float dwell) - { - // Need to translate to MXP somehow - } - - public void SendUserInfoReply(bool imViaEmail, bool visible, string email) - { - // Need to translate to MXP somehow - } - - public void KillEndDone() - { - Stop(); - } - - public bool AddGenericPacketHandler(string MethodName, GenericMessage handler) - { - // Need to translate to MXP somehow - return true; - } - - #endregion - - #region IClientCore - - public bool TryGet(out T iface) - { - iface = default(T); - return false; - } - - public T Get() - { - return default(T); - } - - public void Disconnect(string reason) - { - Kick(reason); - Close(); - } - - public void Disconnect() - { - Close(); - } - - #endregion - - public void SendCreateGroupReply(UUID groupID, bool success, string message) - { - } - - public void RefreshGroupMembership() - { - } - - public void SendUseCachedMuteList() - { - } - - public void SendMuteListUpdate(string filename) - { - } - - public void SendPickInfoReply(UUID pickID,UUID creatorID, bool topPick, UUID parcelID, string name, string desc, UUID snapshotID, string user, string originalName, string simName, Vector3 posGlobal, int sortOrder, bool enabled) - { - } - - public void SendRebakeAvatarTextures(UUID textureID) - { - } - - public void SendAvatarInterestsReply(UUID avatarID, uint wantMask, string wantText, uint skillsMask, string skillsText, string languages) - { - } - - public void SendGroupAccountingDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID, int amt) - { - } - - public void SendGroupAccountingSummary(IClientAPI sender,UUID groupID, uint moneyAmt, int totalTier, int usedTier) - { - } - - public void SendGroupTransactionsSummaryDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID,int amt) - { - } - - public void SendGroupVoteHistory(UUID groupID, UUID transactionID, GroupVoteHistory[] Votes) - { - } - - public void SendGroupActiveProposals(UUID groupID, UUID transactionID, GroupActiveProposals[] Proposals) - { - } - - public void SendChangeUserRights(UUID agentID, UUID friendID, int rights) - { - } - - public void SendTextBoxRequest(string message, int chatChannel, string objectname, string ownerFirstName, string ownerLastName, UUID objectId) - { - } - - public void StopFlying(ISceneEntity presence) - { - } - - public void SendPlacesReply(UUID queryID, UUID transactionID, PlacesReplyData[] data) - { - } - } -} diff --git a/OpenSim/Client/MXP/MXPModule.cs b/OpenSim/Client/MXP/MXPModule.cs deleted file mode 100644 index 0b442cca15..0000000000 --- a/OpenSim/Client/MXP/MXPModule.cs +++ /dev/null @@ -1,131 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -using System; -using System.Collections.Generic; -using System.Reflection; -using System.Text; -using System.Timers; -using log4net; -using MXP; -using Nini.Config; -using OpenMetaverse; -using OpenSim.Client.MXP.PacketHandler; -using OpenSim.Region.Framework.Interfaces; -using OpenSim.Region.Framework.Scenes; - -namespace OpenSim.Client.MXP -{ - - /** - * MXP Client Module which adds MXP support to client / region communication. - */ - public class MXPModule : IRegionModule - { - private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - - private MXPPacketServer m_server; - - private IConfigSource m_config; - private int m_port = 1253; - private Timer m_ticker; - - private readonly Dictionary m_scenes = new Dictionary(); - private bool m_shutdown; - - public void Initialise(Scene scene, IConfigSource source) - { - if (!m_scenes.ContainsKey(scene.RegionInfo.RegionID)) - m_scenes.Add(scene.RegionInfo.RegionID, scene); - - m_config = source; - } - - public void PostInitialise() - { - if (m_config.Configs["MXP"] != null) - { - IConfig con = m_config.Configs["MXP"]; - - if (!con.GetBoolean("Enabled", false)) - return; - - m_port = con.GetInt("Port", m_port); - - m_server = new MXPPacketServer(m_port, m_scenes,m_config.Configs["StandAlone"].GetBoolean("accounts_authenticate",true)); - - m_ticker = new Timer(100); - m_ticker.AutoReset = false; - m_ticker.Elapsed += ticker_Elapsed; - - lock (m_ticker) - m_ticker.Start(); - - m_log.Info("[MXP ClientStack] MXP Enabled and Listening"); - } - } - - void ticker_Elapsed(object sender, ElapsedEventArgs e) - { - try - { - m_server.Process(); - } - catch (Exception ex) - { - m_log.Error("[MXP ClientStack]: Unhandled exception in process loop: " + ex.ToString() + " :" + ex.StackTrace.ToString()); - } - - if (!m_shutdown) - { - lock (m_ticker) - m_ticker.Start(); - } - } - - public void Close() - { - m_shutdown = true; - if (m_ticker != null) - { - lock (m_ticker) - m_ticker.Stop(); - } - } - - public string Name - { - get { return "MXP ClientStack Module"; } - } - - public bool IsSharedModule - { - get { return true; } - } - - } -} diff --git a/OpenSim/Client/MXP/MXPUtil.cs b/OpenSim/Client/MXP/MXPUtil.cs deleted file mode 100644 index 3387145c22..0000000000 --- a/OpenSim/Client/MXP/MXPUtil.cs +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -using System; -using System.Collections.Generic; -using System.Text; -using OpenMetaverse; - -namespace OpenSim.Client.MXP -{ - public static class MXPUtil - { - public static string GenerateMXPURL(string server, int port, UUID bubbleID, Vector3 location) - { - return string.Format("mxp://{0}:{1}/{2}/{3}", server, port, bubbleID.Guid, location); - } - } -} diff --git a/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs b/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs deleted file mode 100644 index dcecb8b384..0000000000 --- a/OpenSim/Client/MXP/PacketHandler/MXPPacketServer.cs +++ /dev/null @@ -1,561 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -/* This file borrows heavily from MXPServer.cs - the reference MXPServer - * See http://www.bubblecloud.org for a copy of the original file and - * implementation details. */ - -using System; -using System.Collections.Generic; -using System.Reflection; -using System.Threading; -using log4net; -using MXP; -using MXP.Messages; -using OpenMetaverse; -using OpenSim.Client.MXP.ClientStack; -using OpenSim.Framework; -using OpenSim.Region.Framework.Scenes; -using OpenSim.Framework.Communications; -using OpenSim.Services.Interfaces; -using System.Security.Cryptography; - -namespace OpenSim.Client.MXP.PacketHandler -{ - public class MXPPacketServer - { - internal static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - - #region Fields - - private readonly List m_clients = new List(); - private readonly Dictionary m_scenes; - private readonly Transmitter m_transmitter; - -// private readonly Thread m_clientThread; - - private readonly IList m_sessions = new List(); - private readonly IList m_sessionsToClient = new List(); - private readonly IList m_sessionsToRemove = new List(); - - private readonly int m_port; -// private readonly bool m_accountsAuthenticate; - - private readonly String m_programName; - private readonly byte m_programMajorVersion; - private readonly byte m_programMinorVersion; - - #endregion - - #region Constructors - - public MXPPacketServer(int port, Dictionary scenes, bool accountsAuthenticate) - { - m_port = port; -// m_accountsAuthenticate = accountsAuthenticate; - - m_scenes = scenes; - - m_programMinorVersion = 63; - m_programMajorVersion = 0; - m_programName = "OpenSimulator"; - - m_transmitter = new Transmitter(port); - - StartListener(); - } - - public void StartListener() - { - m_log.Info("[MXP ClientStack] Transmitter starting on UDP server port: " + m_port); - m_transmitter.Startup(); - m_log.Info("[MXP ClientStack] Transmitter started. MXP version: "+MxpConstants.ProtocolMajorVersion+"."+MxpConstants.ProtocolMinorVersion+" Source Revision: "+MxpConstants.ProtocolSourceRevision); - } - - #endregion - - #region Properties - - /// - /// Number of sessions pending. (Process() accepts pending sessions). - /// - public int PendingSessionCount - { - get - { - return m_transmitter.PendingSessionCount; - } - } - /// - /// Number of connected sessions. - /// - public int SessionCount - { - get - { - return m_sessions.Count; - } - } - /// - /// Property reflecting whether client transmitter threads are alive. - /// - public bool IsTransmitterAlive - { - get - { - return m_transmitter != null && m_transmitter.IsAlive; - } - } - /// - /// Number of packets sent. - /// - public ulong PacketsSent - { - get - { - return m_transmitter != null ? m_transmitter.PacketsSent : 0; - } - } - /// - /// Number of packets received. - /// - public ulong PacketsReceived - { - get - { - return m_transmitter != null ? m_transmitter.PacketsReceived : 0; - } - } - /// - /// Bytes client has received so far. - /// - public ulong BytesReceived - { - get - { - return m_transmitter != null ? m_transmitter.BytesReceived : 0; - } - } - /// - /// Bytes client has sent so far. - /// - public ulong BytesSent - { - get - { - return m_transmitter != null ? m_transmitter.BytesSent : 0; - } - } - /// - /// Number of bytes received (bytes per second) during past second. - /// - public double ReceiveRate - { - get - { - return m_transmitter != null ? m_transmitter.ReceiveRate : 0; - } - } - /// - /// Number of bytes sent (bytes per second) during past second. - /// - public double SendRate - { - get - { - return m_transmitter != null ? m_transmitter.SendRate : 0; - } - } - - #endregion - - #region Session Management - - public void Disconnect(Session session) - { - if (session.IsConnected) - { - Message message = MessageFactory.Current.ReserveMessage(typeof(LeaveRequestMessage)); - session.Send(message); - MessageFactory.Current.ReleaseMessage(message); - } - else - { - throw new Exception("Not connected."); - } - } - - #endregion - - #region Processing - - public void Process() - { - ProcessMessages(); - Clean(); - } - - public void Clean() - { - foreach (MXPClientView clientView in m_clients) - { - if (clientView.Session.SessionState == SessionState.Disconnected) - { - m_sessionsToRemove.Add(clientView); - } - } - - foreach (MXPClientView clientView in m_sessionsToRemove) - { - clientView.Scene.RemoveClient(clientView.AgentId); - clientView.OnClean(); - m_clients.Remove(clientView); - m_sessions.Remove(clientView.Session); - } - - m_sessionsToRemove.Clear(); - } - - public void ProcessMessages() - { - if (m_transmitter.PendingSessionCount > 0) - { - Session tmp = m_transmitter.AcceptPendingSession(); - m_sessions.Add(tmp); - m_sessionsToClient.Add(tmp); - - } - - List tmpRemove = new List(); - - foreach (Session session in m_sessionsToClient) - { - while (session.AvailableMessages > 0) - { - Message message = session.Receive(); - - if (message.GetType() == typeof (JoinRequestMessage)) - { - - JoinRequestMessage joinRequestMessage = (JoinRequestMessage) message; - - m_log.Info("[MXP ClientStack]: Session join request: " + session.SessionId + " (" + - (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + - session.RemoteEndPoint.Port + ")"); - - try - { - - if (joinRequestMessage.BubbleId == Guid.Empty) - { - foreach (Scene scene in m_scenes.Values) - { - if (scene.RegionInfo.RegionName == joinRequestMessage.BubbleName) - { - m_log.Info("[MXP ClientStack]: Resolved region by name: " + joinRequestMessage.BubbleName + " (" + scene.RegionInfo.RegionID + ")"); - joinRequestMessage.BubbleId = scene.RegionInfo.RegionID.Guid; - } - } - } - - if (joinRequestMessage.BubbleId == Guid.Empty) - { - m_log.Warn("[MXP ClientStack]: Failed to resolve region by name: " + joinRequestMessage.BubbleName); - } - - UUID sceneId = new UUID(joinRequestMessage.BubbleId); - - bool regionExists = true; - if (!m_scenes.ContainsKey(sceneId)) - { - m_log.Info("[MXP ClientStack]: No such region: " + sceneId); - regionExists = false; - } - - UUID userId = UUID.Zero; - UserAccount account = null; - bool authorized = regionExists ? AuthoriseUser(joinRequestMessage.ParticipantName, - joinRequestMessage.ParticipantPassphrase, - new UUID(joinRequestMessage.BubbleId), out account) - : false; - - if (authorized) - { - Scene scene = m_scenes[sceneId]; - UUID mxpSessionID = UUID.Random(); - - string reason; - - m_log.Debug("[MXP ClientStack]: Session join request success: " + session.SessionId + " (" + - (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + - session.RemoteEndPoint.Port + ")"); - - m_log.Debug("[MXP ClientStack]: Attaching UserAgent to UserProfile..."); - UUID secureSession = UUID.Zero; - AttachUserAgentToUserProfile(account, session, mxpSessionID, sceneId, out secureSession); - m_log.Debug("[MXP ClientStack]: Attached UserAgent to UserProfile."); - m_log.Debug("[MXP ClientStack]: Preparing Scene to Connection..."); - if (!PrepareSceneForConnection(mxpSessionID, secureSession, sceneId, account, out reason)) - { - m_log.DebugFormat("[MXP ClientStack]: Scene refused connection: {0}", reason); - DeclineConnection(session, joinRequestMessage); - tmpRemove.Add(session); - continue; - } - m_log.Debug("[MXP ClientStack]: Prepared Scene to Connection."); - m_log.Debug("[MXP ClientStack]: Accepting connection..."); - AcceptConnection(session, joinRequestMessage, mxpSessionID, userId); - m_log.Info("[MXP ClientStack]: Accepted connection."); - - m_log.Debug("[MXP ClientStack]: Creating ClientView...."); - MXPClientView client = new MXPClientView(session, mxpSessionID, userId, scene, account.FirstName, account.LastName); - m_clients.Add(client); - m_log.Debug("[MXP ClientStack]: Created ClientView."); - - client.MXPSendSynchronizationBegin(m_scenes[new UUID(joinRequestMessage.BubbleId)].SceneContents.GetTotalObjectsCount()); - - m_log.Debug("[MXP ClientStack]: Starting ClientView..."); - try - { - client.Start(); - m_log.Debug("[MXP ClientStack]: Started ClientView."); - } - catch (Exception e) - { - m_log.Error(e); - } - - m_log.Debug("[MXP ClientStack]: Connected"); - } - else - { - m_log.Info("[MXP ClientStack]: Session join request failure: " + session.SessionId + " (" + - (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + - session.RemoteEndPoint.Port + ")"); - - DeclineConnection(session, joinRequestMessage); - } - } - catch (Exception e) - { - m_log.Error("[MXP ClientStack]: Session join request failure: " + session.SessionId + " (" + - (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + - session.RemoteEndPoint.Port + "): "+e.ToString()+" :"+e.StackTrace.ToString()); - } - tmpRemove.Add(session); - } - } - } - - foreach (Session session in tmpRemove) - { - m_sessionsToClient.Remove(session); - } - - foreach (MXPClientView clientView in m_clients) - { - int messagesProcessedCount = 0; - Session session = clientView.Session; - - while (session.AvailableMessages > 0) - { - Message message = session.Receive(); - - if (message.GetType() == typeof(LeaveRequestMessage)) - { - LeaveResponseMessage leaveResponseMessage = (LeaveResponseMessage)MessageFactory.Current.ReserveMessage( - typeof(LeaveResponseMessage)); - - m_log.Debug("[MXP ClientStack]: Session leave request: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); - - leaveResponseMessage.RequestMessageId = message.MessageId; - leaveResponseMessage.FailureCode = 0; - session.Send(leaveResponseMessage); - - if (session.SessionState != SessionState.Disconnected) - { - session.SetStateDisconnected(); - } - - m_log.Debug("[MXP ClientStack]: Removing Client from Scene"); - //clientView.Scene.RemoveClient(clientView.AgentId); - } - if (message.GetType() == typeof(LeaveResponseMessage)) - { - LeaveResponseMessage leaveResponseMessage = (LeaveResponseMessage)message; - - m_log.Debug("[MXP ClientStack]: Session leave response: " + session.SessionId + " (" + (session.IsIncoming ? "from" : "to") + " " + session.RemoteEndPoint.Address + ":" + session.RemoteEndPoint.Port + ")"); - - if (leaveResponseMessage.FailureCode == 0) - { - session.SetStateDisconnected(); - } - - m_log.Debug("[MXP ClientStack]: Removing Client from Scene"); - //clientView.Scene.RemoveClient(clientView.AgentId); - } - else - { - clientView.MXPPRocessMessage(message); - } - - MessageFactory.Current.ReleaseMessage(message); - messagesProcessedCount++; - if (messagesProcessedCount > 1000) - { - break; - } - } - } - } - - private void AcceptConnection(Session session, JoinRequestMessage joinRequestMessage, UUID mxpSessionID, UUID userId) - { - JoinResponseMessage joinResponseMessage = (JoinResponseMessage)MessageFactory.Current.ReserveMessage( - typeof(JoinResponseMessage)); - - joinResponseMessage.RequestMessageId = joinRequestMessage.MessageId; - joinResponseMessage.FailureCode = MxpResponseCodes.SUCCESS; - - joinResponseMessage.BubbleId = joinRequestMessage.BubbleId; - joinResponseMessage.ParticipantId = userId.Guid; - joinResponseMessage.AvatarId = userId.Guid; - joinResponseMessage.BubbleAssetCacheUrl = "http://" + - NetworkUtil.GetHostFor(session.RemoteEndPoint.Address, - m_scenes[ - new UUID(joinRequestMessage.BubbleId)]. - RegionInfo. - ExternalHostName) + ":" + - m_scenes[new UUID(joinRequestMessage.BubbleId)].RegionInfo. - HttpPort + "/assets/"; - - joinResponseMessage.BubbleName = m_scenes[new UUID(joinRequestMessage.BubbleId)].RegionInfo.RegionName; - - joinResponseMessage.BubbleRange = 128; - joinResponseMessage.BubblePerceptionRange = 128 + 256; - joinResponseMessage.BubbleRealTime = 0; - joinResponseMessage.ProgramName = m_programName; - joinResponseMessage.ProgramMajorVersion = m_programMajorVersion; - joinResponseMessage.ProgramMinorVersion = m_programMinorVersion; - joinResponseMessage.ProtocolMajorVersion = MxpConstants.ProtocolMajorVersion; - joinResponseMessage.ProtocolMinorVersion = MxpConstants.ProtocolMinorVersion; - joinResponseMessage.ProtocolSourceRevision = MxpConstants.ProtocolSourceRevision; - - session.Send(joinResponseMessage); - - session.SetStateConnected(); - } - - private void DeclineConnection(Session session, Message joinRequestMessage) - { - JoinResponseMessage joinResponseMessage = (JoinResponseMessage)MessageFactory.Current.ReserveMessage(typeof(JoinResponseMessage)); - - joinResponseMessage.RequestMessageId = joinRequestMessage.MessageId; - joinResponseMessage.FailureCode = MxpResponseCodes.UNAUTHORIZED_OPERATION; - - joinResponseMessage.ProgramName = m_programName; - joinResponseMessage.ProgramMajorVersion = m_programMajorVersion; - joinResponseMessage.ProgramMinorVersion = m_programMinorVersion; - joinResponseMessage.ProtocolMajorVersion = MxpConstants.ProtocolMajorVersion; - joinResponseMessage.ProtocolMinorVersion = MxpConstants.ProtocolMinorVersion; - joinResponseMessage.ProtocolSourceRevision = MxpConstants.ProtocolSourceRevision; - - session.Send(joinResponseMessage); - - session.SetStateDisconnected(); - } - - public bool AuthoriseUser(string participantName, string password, UUID sceneId, out UserAccount account) - { - string firstName = ""; - string lastName = ""; - account = null; - - string[] nameParts = participantName.Split(' '); - if (nameParts.Length != 2) - { - m_log.Error("[MXP ClientStack]: Login failed as user name is not formed of first and last name separated by space: " + participantName); - return false; - } - firstName = nameParts[0]; - lastName = nameParts[1]; - - account = m_scenes[sceneId].UserAccountService.GetUserAccount(m_scenes[sceneId].RegionInfo.ScopeID, firstName, lastName); - if (account != null) - return (m_scenes[sceneId].AuthenticationService.Authenticate(account.PrincipalID, password, 1) != string.Empty); - - return false; - } - - private void AttachUserAgentToUserProfile(UserAccount account, Session session, UUID sessionId, UUID sceneId, out UUID secureSessionId) - { - secureSessionId = UUID.Random(); - Scene scene = m_scenes[sceneId]; - scene.PresenceService.LoginAgent(account.PrincipalID.ToString(), sessionId, secureSessionId); - } - - private bool PrepareSceneForConnection(UUID sessionId, UUID secureSessionId, UUID sceneId, UserAccount account, out string reason) - { - Scene scene = m_scenes[sceneId]; - - AgentCircuitData agent = new AgentCircuitData(); - agent.AgentID = account.PrincipalID; - agent.firstname = account.FirstName; - agent.lastname = account.LastName; - agent.SessionID = sessionId; - agent.SecureSessionID = secureSessionId; - agent.circuitcode = sessionId.CRC(); - agent.BaseFolder = UUID.Zero; - agent.InventoryFolder = UUID.Zero; - agent.startpos = new Vector3(0, 0, 0); // TODO Fill in region start position - agent.CapsPath = "http://localhost/"; - agent.Appearance = scene.AvatarService.GetAppearance(account.PrincipalID); - - if (agent.Appearance == null) - { - m_log.WarnFormat("[INTER]: Appearance not found for {0} {1}. Creating default.", agent.firstname, agent.lastname); - agent.Appearance = new AvatarAppearance(); - } - - return scene.NewUserConnection(agent, 0, out reason); - } - - public void PrintDebugInformation() - { - m_log.Info("[MXP ClientStack]: Statistics report"); - m_log.Info("Pending Sessions: " + PendingSessionCount); - m_log.Info("Sessions: " + SessionCount + " (Clients: " + m_clients.Count + " )"); - m_log.Info("Transmitter Alive?: " + IsTransmitterAlive); - m_log.Info("Packets Sent/Received: " + PacketsSent + " / " + PacketsReceived); - m_log.Info("Bytes Sent/Received: " + BytesSent + " / " + BytesReceived); - m_log.Info("Send/Receive Rate (bps): " + SendRate + " / " + ReceiveRate); - } - - #endregion - - } -} diff --git a/OpenSim/Client/VWoHTTP/ClientStack/VWHClientView.cs b/OpenSim/Client/VWoHTTP/ClientStack/VWHClientView.cs deleted file mode 100644 index d8cd0acbd9..0000000000 --- a/OpenSim/Client/VWoHTTP/ClientStack/VWHClientView.cs +++ /dev/null @@ -1,1231 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -using System; -using System.Collections.Generic; -using System.Drawing; -using System.Drawing.Imaging; -using System.IO; -using System.Net; -using System.Text; -using OpenMetaverse; -using OpenMetaverse.Imaging; -using OpenMetaverse.Packets; -using OpenSim.Framework; -using OpenSim.Framework.Servers; -using OpenSim.Framework.Servers.HttpServer; -using OpenSim.Region.Framework.Scenes; - -namespace OpenSim.Client.VWoHTTP.ClientStack -{ - class VWHClientView : IClientAPI - { - private Scene m_scene; - - - public bool ProcessInMsg(OSHttpRequest req, OSHttpResponse resp) - { - // 0 1 2 3 - // http://simulator.com:9000/vwohttp/sessionid/methodname/param - string[] urlparts = req.Url.AbsolutePath.Split(new char[] {'/'}, StringSplitOptions.RemoveEmptyEntries); - - UUID sessionID; - // Check for session - if (!UUID.TryParse(urlparts[1], out sessionID)) - return false; - // Check we match session - if (sessionID != SessionId) - return false; - - string method = urlparts[2]; - - string param = String.Empty; - if (urlparts.Length > 3) - param = urlparts[3]; - - bool found; - - switch (method.ToLower()) - { - case "textures": - found = ProcessTextureRequest(param, resp); - break; - default: - found = false; - break; - } - - return found; - } - - private bool ProcessTextureRequest(string param, OSHttpResponse resp) - { - UUID assetID; - if (!UUID.TryParse(param, out assetID)) - return false; - - AssetBase asset = m_scene.AssetService.Get(assetID.ToString()); - - if (asset == null) - return false; - - ManagedImage tmp; - Image imgData; - byte[] jpegdata; - - OpenJPEG.DecodeToImage(asset.Data, out tmp, out imgData); - - using (MemoryStream ms = new MemoryStream()) - { - imgData.Save(ms, ImageFormat.Jpeg); - jpegdata = ms.GetBuffer(); - } - - resp.ContentType = "image/jpeg"; - resp.ContentLength = jpegdata.Length; - resp.StatusCode = 200; - resp.Body.Write(jpegdata, 0, jpegdata.Length); - - return true; - } - - public VWHClientView(UUID sessionID, UUID agentID, string agentName, Scene scene) - { - m_scene = scene; - } - - #region Implementation of IClientAPI - - public Vector3 StartPos - { - get { throw new System.NotImplementedException(); } - set { throw new System.NotImplementedException(); } - } - - public UUID AgentId - { - get { throw new System.NotImplementedException(); } - } - - public UUID SessionId - { - get { throw new System.NotImplementedException(); } - } - - public UUID SecureSessionId - { - get { throw new System.NotImplementedException(); } - } - - public UUID ActiveGroupId - { - get { throw new System.NotImplementedException(); } - } - - public string ActiveGroupName - { - get { throw new System.NotImplementedException(); } - } - - public ulong ActiveGroupPowers - { - get { throw new System.NotImplementedException(); } - } - - public ulong GetGroupPowers(UUID groupID) - { - throw new System.NotImplementedException(); - } - - public bool IsGroupMember(UUID GroupID) - { - throw new System.NotImplementedException(); - } - - public string FirstName - { - get { throw new System.NotImplementedException(); } - } - - public string LastName - { - get { throw new System.NotImplementedException(); } - } - - public IScene Scene - { - get { throw new System.NotImplementedException(); } - } - - public int NextAnimationSequenceNumber - { - get { throw new System.NotImplementedException(); } - } - - public string Name - { - get { throw new System.NotImplementedException(); } - } - - public bool IsActive - { - get { throw new System.NotImplementedException(); } - set { throw new System.NotImplementedException(); } - } - public bool IsLoggingOut - { - get { throw new System.NotImplementedException(); } - set { throw new System.NotImplementedException(); } - } - public bool SendLogoutPacketWhenClosing - { - set { throw new System.NotImplementedException(); } - } - - public uint CircuitCode - { - get { throw new System.NotImplementedException(); } - } - - public IPEndPoint RemoteEndPoint - { - get { throw new System.NotImplementedException(); } - } - - public event GenericMessage OnGenericMessage = delegate { }; - public event ImprovedInstantMessage OnInstantMessage = delegate { }; - public event ChatMessage OnChatFromClient = delegate { }; - public event TextureRequest OnRequestTexture = delegate { }; - public event RezObject OnRezObject = delegate { }; - public event ModifyTerrain OnModifyTerrain = delegate { }; - public event BakeTerrain OnBakeTerrain = delegate { }; - public event EstateChangeInfo OnEstateChangeInfo = delegate { }; - public event SetAppearance OnSetAppearance = delegate { }; - public event AvatarNowWearing OnAvatarNowWearing = delegate { }; - public event RezSingleAttachmentFromInv OnRezSingleAttachmentFromInv = delegate { return new UUID(); }; - public event RezMultipleAttachmentsFromInv OnRezMultipleAttachmentsFromInv = delegate { }; - public event UUIDNameRequest OnDetachAttachmentIntoInv = delegate { }; - public event ObjectAttach OnObjectAttach = delegate { }; - public event ObjectDeselect OnObjectDetach = delegate { }; - public event ObjectDrop OnObjectDrop = delegate { }; - public event StartAnim OnStartAnim = delegate { }; - public event StopAnim OnStopAnim = delegate { }; - public event LinkObjects OnLinkObjects = delegate { }; - public event DelinkObjects OnDelinkObjects = delegate { }; - public event RequestMapBlocks OnRequestMapBlocks = delegate { }; - public event RequestMapName OnMapNameRequest = delegate { }; - public event TeleportLocationRequest OnTeleportLocationRequest = delegate { }; - public event DisconnectUser OnDisconnectUser = delegate { }; - public event RequestAvatarProperties OnRequestAvatarProperties = delegate { }; - public event SetAlwaysRun OnSetAlwaysRun = delegate { }; - public event TeleportLandmarkRequest OnTeleportLandmarkRequest = delegate { }; - public event DeRezObject OnDeRezObject = delegate { }; - public event Action OnRegionHandShakeReply = delegate { }; - public event GenericCall1 OnRequestWearables = delegate { }; - public event GenericCall1 OnCompleteMovementToRegion = delegate { }; - public event UpdateAgent OnPreAgentUpdate; - public event UpdateAgent OnAgentUpdate = delegate { }; - public event AgentRequestSit OnAgentRequestSit = delegate { }; - public event AgentSit OnAgentSit = delegate { }; - public event AvatarPickerRequest OnAvatarPickerRequest = delegate { }; - public event Action OnRequestAvatarsData = delegate { }; - public event AddNewPrim OnAddPrim = delegate { }; - public event FetchInventory OnAgentDataUpdateRequest = delegate { }; - public event TeleportLocationRequest OnSetStartLocationRequest = delegate { }; - public event RequestGodlikePowers OnRequestGodlikePowers = delegate { }; - public event GodKickUser OnGodKickUser = delegate { }; - public event ObjectDuplicate OnObjectDuplicate = delegate { }; - public event ObjectDuplicateOnRay OnObjectDuplicateOnRay = delegate { }; - public event GrabObject OnGrabObject = delegate { }; - public event DeGrabObject OnDeGrabObject = delegate { }; - public event MoveObject OnGrabUpdate = delegate { }; - public event SpinStart OnSpinStart = delegate { }; - public event SpinObject OnSpinUpdate = delegate { }; - public event SpinStop OnSpinStop = delegate { }; - public event UpdateShape OnUpdatePrimShape = delegate { }; - public event ObjectExtraParams OnUpdateExtraParams = delegate { }; - public event ObjectRequest OnObjectRequest = delegate { }; - public event ObjectSelect OnObjectSelect = delegate { }; - public event ObjectDeselect OnObjectDeselect = delegate { }; - public event GenericCall7 OnObjectDescription = delegate { }; - public event GenericCall7 OnObjectName = delegate { }; - public event GenericCall7 OnObjectClickAction = delegate { }; - public event GenericCall7 OnObjectMaterial = delegate { }; - public event RequestObjectPropertiesFamily OnRequestObjectPropertiesFamily = delegate { }; - public event UpdatePrimFlags OnUpdatePrimFlags = delegate { }; - public event UpdatePrimTexture OnUpdatePrimTexture = delegate { }; - public event UpdateVector OnUpdatePrimGroupPosition = delegate { }; - public event UpdateVector OnUpdatePrimSinglePosition = delegate { }; - public event UpdatePrimRotation OnUpdatePrimGroupRotation = delegate { }; - public event UpdatePrimSingleRotation OnUpdatePrimSingleRotation = delegate { }; - public event UpdatePrimSingleRotationPosition OnUpdatePrimSingleRotationPosition = delegate { }; - public event UpdatePrimGroupRotation OnUpdatePrimGroupMouseRotation = delegate { }; - public event UpdateVector OnUpdatePrimScale = delegate { }; - public event UpdateVector OnUpdatePrimGroupScale = delegate { }; - public event StatusChange OnChildAgentStatus = delegate { }; - public event GenericCall2 OnStopMovement = delegate { }; - public event Action OnRemoveAvatar = delegate { }; - public event ObjectPermissions OnObjectPermissions = delegate { }; - public event CreateNewInventoryItem OnCreateNewInventoryItem = delegate { }; - public event LinkInventoryItem OnLinkInventoryItem = delegate { }; - public event CreateInventoryFolder OnCreateNewInventoryFolder = delegate { }; - public event UpdateInventoryFolder OnUpdateInventoryFolder = delegate { }; - public event MoveInventoryFolder OnMoveInventoryFolder = delegate { }; - public event FetchInventoryDescendents OnFetchInventoryDescendents = delegate { }; - public event PurgeInventoryDescendents OnPurgeInventoryDescendents = delegate { }; - public event FetchInventory OnFetchInventory = delegate { }; - public event RequestTaskInventory OnRequestTaskInventory = delegate { }; - public event UpdateInventoryItem OnUpdateInventoryItem = delegate { }; - public event CopyInventoryItem OnCopyInventoryItem = delegate { }; - public event MoveInventoryItem OnMoveInventoryItem = delegate { }; - public event RemoveInventoryFolder OnRemoveInventoryFolder = delegate { }; - public event RemoveInventoryItem OnRemoveInventoryItem = delegate { }; - public event UDPAssetUploadRequest OnAssetUploadRequest = delegate { }; - public event XferReceive OnXferReceive = delegate { }; - public event RequestXfer OnRequestXfer = delegate { }; - public event ConfirmXfer OnConfirmXfer = delegate { }; - public event AbortXfer OnAbortXfer = delegate { }; - public event RezScript OnRezScript = delegate { }; - public event UpdateTaskInventory OnUpdateTaskInventory = delegate { }; - public event MoveTaskInventory OnMoveTaskItem = delegate { }; - public event RemoveTaskInventory OnRemoveTaskItem = delegate { }; - public event RequestAsset OnRequestAsset = delegate { }; - public event UUIDNameRequest OnNameFromUUIDRequest = delegate { }; - public event ParcelAccessListRequest OnParcelAccessListRequest = delegate { }; - public event ParcelAccessListUpdateRequest OnParcelAccessListUpdateRequest = delegate { }; - public event ParcelPropertiesRequest OnParcelPropertiesRequest = delegate { }; - public event ParcelDivideRequest OnParcelDivideRequest = delegate { }; - public event ParcelJoinRequest OnParcelJoinRequest = delegate { }; - public event ParcelPropertiesUpdateRequest OnParcelPropertiesUpdateRequest = delegate { }; - public event ParcelSelectObjects OnParcelSelectObjects = delegate { }; - public event ParcelObjectOwnerRequest OnParcelObjectOwnerRequest = delegate { }; - public event ParcelAbandonRequest OnParcelAbandonRequest = delegate { }; - public event ParcelGodForceOwner OnParcelGodForceOwner = delegate { }; - public event ParcelReclaim OnParcelReclaim = delegate { }; - public event ParcelReturnObjectsRequest OnParcelReturnObjectsRequest = delegate { }; - public event ParcelDeedToGroup OnParcelDeedToGroup = delegate { }; - public event RegionInfoRequest OnRegionInfoRequest = delegate { }; - public event EstateCovenantRequest OnEstateCovenantRequest = delegate { }; - public event FriendActionDelegate OnApproveFriendRequest = delegate { }; - public event FriendActionDelegate OnDenyFriendRequest = delegate { }; - public event FriendshipTermination OnTerminateFriendship = delegate { }; - public event GrantUserFriendRights OnGrantUserRights = delegate { }; - public event MoneyTransferRequest OnMoneyTransferRequest = delegate { }; - public event EconomyDataRequest OnEconomyDataRequest = delegate { }; - public event MoneyBalanceRequest OnMoneyBalanceRequest = delegate { }; - public event UpdateAvatarProperties OnUpdateAvatarProperties = delegate { }; - public event ParcelBuy OnParcelBuy = delegate { }; - public event RequestPayPrice OnRequestPayPrice = delegate { }; - public event ObjectSaleInfo OnObjectSaleInfo = delegate { }; - public event ObjectBuy OnObjectBuy = delegate { }; - public event BuyObjectInventory OnBuyObjectInventory = delegate { }; - public event RequestTerrain OnRequestTerrain = delegate { }; - public event RequestTerrain OnUploadTerrain = delegate { }; - public event ObjectIncludeInSearch OnObjectIncludeInSearch = delegate { }; - public event UUIDNameRequest OnTeleportHomeRequest = delegate { }; - public event ScriptAnswer OnScriptAnswer = delegate { }; - public event AgentSit OnUndo = delegate { }; - public event AgentSit OnRedo = delegate { }; - public event LandUndo OnLandUndo = delegate { }; - public event ForceReleaseControls OnForceReleaseControls = delegate { }; - public event GodLandStatRequest OnLandStatRequest = delegate { }; - public event DetailedEstateDataRequest OnDetailedEstateDataRequest = delegate { }; - public event SetEstateFlagsRequest OnSetEstateFlagsRequest = delegate { }; - public event SetEstateTerrainBaseTexture OnSetEstateTerrainBaseTexture = delegate { }; - public event SetEstateTerrainDetailTexture OnSetEstateTerrainDetailTexture = delegate { }; - public event SetEstateTerrainTextureHeights OnSetEstateTerrainTextureHeights = delegate { }; - public event CommitEstateTerrainTextureRequest OnCommitEstateTerrainTextureRequest = delegate { }; - public event SetRegionTerrainSettings OnSetRegionTerrainSettings = delegate { }; - public event EstateRestartSimRequest OnEstateRestartSimRequest = delegate { }; - public event EstateChangeCovenantRequest OnEstateChangeCovenantRequest = delegate { }; - public event UpdateEstateAccessDeltaRequest OnUpdateEstateAccessDeltaRequest = delegate { }; - public event SimulatorBlueBoxMessageRequest OnSimulatorBlueBoxMessageRequest = delegate { }; - public event EstateBlueBoxMessageRequest OnEstateBlueBoxMessageRequest = delegate { }; - public event EstateDebugRegionRequest OnEstateDebugRegionRequest = delegate { }; - public event EstateTeleportOneUserHomeRequest OnEstateTeleportOneUserHomeRequest = delegate { }; - public event EstateTeleportAllUsersHomeRequest OnEstateTeleportAllUsersHomeRequest = delegate { }; - public event UUIDNameRequest OnUUIDGroupNameRequest = delegate { }; - public event RegionHandleRequest OnRegionHandleRequest = delegate { }; - public event ParcelInfoRequest OnParcelInfoRequest = delegate { }; - public event RequestObjectPropertiesFamily OnObjectGroupRequest = delegate { }; - public event ScriptReset OnScriptReset = delegate { }; - public event GetScriptRunning OnGetScriptRunning = delegate { }; - public event SetScriptRunning OnSetScriptRunning = delegate { }; - public event UpdateVector OnAutoPilotGo = delegate { }; - public event TerrainUnacked OnUnackedTerrain = delegate { }; - public event ActivateGesture OnActivateGesture = delegate { }; - public event DeactivateGesture OnDeactivateGesture = delegate { }; - public event ObjectOwner OnObjectOwner = delegate { }; - public event DirPlacesQuery OnDirPlacesQuery = delegate { }; - public event DirFindQuery OnDirFindQuery = delegate { }; - public event DirLandQuery OnDirLandQuery = delegate { }; - public event DirPopularQuery OnDirPopularQuery = delegate { }; - public event DirClassifiedQuery OnDirClassifiedQuery = delegate { }; - public event EventInfoRequest OnEventInfoRequest = delegate { }; - public event ParcelSetOtherCleanTime OnParcelSetOtherCleanTime = delegate { }; - public event MapItemRequest OnMapItemRequest = delegate { }; - public event OfferCallingCard OnOfferCallingCard = delegate { }; - public event AcceptCallingCard OnAcceptCallingCard = delegate { }; - public event DeclineCallingCard OnDeclineCallingCard = delegate { }; - public event SoundTrigger OnSoundTrigger = delegate { }; - public event StartLure OnStartLure = delegate { }; - public event TeleportLureRequest OnTeleportLureRequest = delegate { }; - public event NetworkStats OnNetworkStatsUpdate = delegate { }; - public event ClassifiedInfoRequest OnClassifiedInfoRequest = delegate { }; - public event ClassifiedInfoUpdate OnClassifiedInfoUpdate = delegate { }; - public event ClassifiedDelete OnClassifiedDelete = delegate { }; - public event ClassifiedDelete OnClassifiedGodDelete = delegate { }; - public event EventNotificationAddRequest OnEventNotificationAddRequest = delegate { }; - public event EventNotificationRemoveRequest OnEventNotificationRemoveRequest = delegate { }; - public event EventGodDelete OnEventGodDelete = delegate { }; - public event ParcelDwellRequest OnParcelDwellRequest = delegate { }; - public event UserInfoRequest OnUserInfoRequest = delegate { }; - public event UpdateUserInfo OnUpdateUserInfo = delegate { }; - public event RetrieveInstantMessages OnRetrieveInstantMessages = delegate { }; - public event PickDelete OnPickDelete = delegate { }; - public event PickGodDelete OnPickGodDelete = delegate { }; - public event PickInfoUpdate OnPickInfoUpdate = delegate { }; - public event AvatarNotesUpdate OnAvatarNotesUpdate = delegate { }; - public event MuteListRequest OnMuteListRequest = delegate { }; - public event AvatarInterestUpdate OnAvatarInterestUpdate = delegate { }; - public event PlacesQuery OnPlacesQuery = delegate { }; - public event FindAgentUpdate OnFindAgent = delegate { }; - public event TrackAgentUpdate OnTrackAgent = delegate { }; - public event NewUserReport OnUserReport = delegate { }; - public event SaveStateHandler OnSaveState = delegate { }; - public event GroupAccountSummaryRequest OnGroupAccountSummaryRequest = delegate { }; - public event GroupAccountDetailsRequest OnGroupAccountDetailsRequest = delegate { }; - public event GroupAccountTransactionsRequest OnGroupAccountTransactionsRequest = delegate { }; - public event FreezeUserUpdate OnParcelFreezeUser = delegate { }; - public event EjectUserUpdate OnParcelEjectUser = delegate { }; - public event ParcelBuyPass OnParcelBuyPass = delegate { }; - public event ParcelGodMark OnParcelGodMark = delegate { }; - public event GroupActiveProposalsRequest OnGroupActiveProposalsRequest = delegate { }; - public event GroupVoteHistoryRequest OnGroupVoteHistoryRequest = delegate { }; - public event SimWideDeletesDelegate OnSimWideDeletes = delegate { }; - public event SendPostcard OnSendPostcard = delegate { }; - public event MuteListEntryUpdate OnUpdateMuteListEntry = delegate { }; - public event MuteListEntryRemove OnRemoveMuteListEntry = delegate { }; - public event GodlikeMessage onGodlikeMessage = delegate { }; - public event GodUpdateRegionInfoUpdate OnGodUpdateRegionInfoUpdate = delegate { }; - - - - public void SetDebugPacketLevel(int newDebug) - { - throw new System.NotImplementedException(); - } - - public void InPacket(object NewPack) - { - throw new System.NotImplementedException(); - } - - public void ProcessInPacket(Packet NewPack) - { - throw new System.NotImplementedException(); - } - - public void Close() - { - throw new System.NotImplementedException(); - } - - public void Kick(string message) - { - throw new System.NotImplementedException(); - } - - public void Start() - { - throw new System.NotImplementedException(); - } - - public void Stop() - { - throw new System.NotImplementedException(); - } - - public void SendWearables(AvatarWearable[] wearables, int serial) - { - throw new System.NotImplementedException(); - } - - public void SendAppearance(UUID agentID, byte[] visualParams, byte[] textureEntry) - { - throw new System.NotImplementedException(); - } - - public void SendStartPingCheck(byte seq) - { - throw new System.NotImplementedException(); - } - - public void SendKillObject(ulong regionHandle, uint localID) - { - throw new System.NotImplementedException(); - } - - public void SendAnimations(UUID[] animID, int[] seqs, UUID sourceAgentId, UUID[] objectIDs) - { - throw new System.NotImplementedException(); - } - - public void SendRegionHandshake(RegionInfo regionInfo, RegionHandshakeArgs args) - { - throw new System.NotImplementedException(); - } - - public void SendChatMessage(string message, byte type, Vector3 fromPos, string fromName, UUID fromAgentID, byte source, byte audible) - { - throw new System.NotImplementedException(); - } - - public void SendInstantMessage(GridInstantMessage im) - { - throw new System.NotImplementedException(); - } - - public void SendGenericMessage(string method, List message) - { - } - - public void SendGenericMessage(string method, List message) - { - throw new System.NotImplementedException(); - } - - public void SendLayerData(float[] map) - { - throw new System.NotImplementedException(); - } - - public void SendLayerData(int px, int py, float[] map) - { - throw new System.NotImplementedException(); - } - - public void SendWindData(Vector2[] windSpeeds) - { - throw new System.NotImplementedException(); - } - - public void SendCloudData(float[] cloudCover) - { - throw new System.NotImplementedException(); - } - - public void MoveAgentIntoRegion(RegionInfo regInfo, Vector3 pos, Vector3 look) - { - throw new System.NotImplementedException(); - } - - public void InformClientOfNeighbour(ulong neighbourHandle, IPEndPoint neighbourExternalEndPoint) - { - throw new System.NotImplementedException(); - } - - public AgentCircuitData RequestClientInfo() - { - throw new System.NotImplementedException(); - } - - public void CrossRegion(ulong newRegionHandle, Vector3 pos, Vector3 lookAt, IPEndPoint newRegionExternalEndPoint, string capsURL) - { - throw new System.NotImplementedException(); - } - - public void SendMapBlock(List mapBlocks, uint flag) - { - throw new System.NotImplementedException(); - } - - public void SendLocalTeleport(Vector3 position, Vector3 lookAt, uint flags) - { - throw new System.NotImplementedException(); - } - - public void SendRegionTeleport(ulong regionHandle, byte simAccess, IPEndPoint regionExternalEndPoint, uint locationID, uint flags, string capsURL) - { - throw new System.NotImplementedException(); - } - - public void SendTeleportFailed(string reason) - { - throw new System.NotImplementedException(); - } - - public void SendTeleportStart(uint flags) - { - throw new System.NotImplementedException(); - } - - public void SendTeleportProgress(uint flags, string message) - { - throw new System.NotImplementedException(); - } - - public void SendMoneyBalance(UUID transaction, bool success, byte[] description, int balance) - { - throw new System.NotImplementedException(); - } - - public void SendPayPrice(UUID objectID, int[] payPrice) - { - throw new System.NotImplementedException(); - } - - public void SendCoarseLocationUpdate(List users, List CoarseLocations) - { - throw new System.NotImplementedException(); - } - - public void SetChildAgentThrottle(byte[] throttle) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarDataImmediate(ISceneEntity avatar) - { - throw new System.NotImplementedException(); - } - - public void SendPrimUpdate(ISceneEntity entity, PrimUpdateFlags updateFlags) - { - throw new System.NotImplementedException(); - } - - public void ReprioritizeUpdates() - { - throw new System.NotImplementedException(); - } - - public void FlushPrimUpdates() - { - throw new System.NotImplementedException(); - } - - public void SendInventoryFolderDetails(UUID ownerID, UUID folderID, List items, List folders, int version, bool fetchFolders, bool fetchItems) - { - throw new System.NotImplementedException(); - } - - public void SendInventoryItemDetails(UUID ownerID, InventoryItemBase item) - { - throw new System.NotImplementedException(); - } - - public void SendInventoryItemCreateUpdate(InventoryItemBase Item, uint callbackId) - { - throw new System.NotImplementedException(); - } - - public void SendRemoveInventoryItem(UUID itemID) - { - throw new System.NotImplementedException(); - } - - public void SendTakeControls(int controls, bool passToAgent, bool TakeControls) - { - throw new System.NotImplementedException(); - } - - public void SendTaskInventory(UUID taskID, short serial, byte[] fileName) - { - throw new System.NotImplementedException(); - } - - public void SendBulkUpdateInventory(InventoryNodeBase node) - { - throw new System.NotImplementedException(); - } - - public void SendXferPacket(ulong xferID, uint packet, byte[] data) - { - throw new System.NotImplementedException(); - } - - public virtual void SendAbortXferPacket(ulong xferID) - { - throw new System.NotImplementedException(); - } - - public void SendEconomyData(float EnergyEfficiency, int ObjectCapacity, int ObjectCount, int PriceEnergyUnit, int PriceGroupCreate, int PriceObjectClaim, float PriceObjectRent, float PriceObjectScaleFactor, int PriceParcelClaim, float PriceParcelClaimFactor, int PriceParcelRent, int PricePublicObjectDecay, int PricePublicObjectDelete, int PriceRentLight, int PriceUpload, int TeleportMinPrice, float TeleportPriceExponent) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarPickerReply(AvatarPickerReplyAgentDataArgs AgentData, List Data) - { - throw new System.NotImplementedException(); - } - - public void SendAgentDataUpdate(UUID agentid, UUID activegroupid, string firstname, string lastname, ulong grouppowers, string groupname, string grouptitle) - { - throw new System.NotImplementedException(); - } - - public void SendPreLoadSound(UUID objectID, UUID ownerID, UUID soundID) - { - throw new System.NotImplementedException(); - } - - public void SendPlayAttachedSound(UUID soundID, UUID objectID, UUID ownerID, float gain, byte flags) - { - throw new System.NotImplementedException(); - } - - public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) - { - throw new System.NotImplementedException(); - } - - public void SendAttachedSoundGainChange(UUID objectID, float gain) - { - throw new System.NotImplementedException(); - } - - public void SendNameReply(UUID profileId, string firstname, string lastname) - { - throw new System.NotImplementedException(); - } - - public void SendAlertMessage(string message) - { - throw new System.NotImplementedException(); - } - - public void SendAgentAlertMessage(string message, bool modal) - { - throw new System.NotImplementedException(); - } - - public void SendLoadURL(string objectname, UUID objectID, UUID ownerID, bool groupOwned, string message, string url) - { - throw new System.NotImplementedException(); - } - - public void SendDialog(string objectname, UUID objectID, string ownerFirstName, string ownerLastName, string msg, UUID textureID, int ch, string[] buttonlabels) - { - throw new System.NotImplementedException(); - } - - public bool AddMoney(int debit) - { - throw new System.NotImplementedException(); - } - - public void SendSunPos(Vector3 sunPos, Vector3 sunVel, ulong CurrentTime, uint SecondsPerSunCycle, uint SecondsPerYear, float OrbitalPosition) - { - throw new System.NotImplementedException(); - } - - public void SendViewerEffect(ViewerEffectPacket.EffectBlock[] effectBlocks) - { - throw new System.NotImplementedException(); - } - - public void SendViewerTime(int phase) - { - throw new System.NotImplementedException(); - } - - public UUID GetDefaultAnimation(string name) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarProperties(UUID avatarID, string aboutText, string bornOn, byte[] charterMember, string flAbout, uint flags, UUID flImageID, UUID imageID, string profileURL, UUID partnerID) - { - throw new System.NotImplementedException(); - } - - public void SendScriptQuestion(UUID taskID, string taskName, string ownerName, UUID itemID, int question) - { - throw new System.NotImplementedException(); - } - - public void SendHealth(float health) - { - throw new System.NotImplementedException(); - } - - public void SendEstateList(UUID invoice, int code, UUID[] Data, uint estateID) - { - throw new System.NotImplementedException(); - } - - public void SendBannedUserList(UUID invoice, EstateBan[] banlist, uint estateID) - { - throw new System.NotImplementedException(); - } - - public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) - { - throw new System.NotImplementedException(); - } - - public void SendEstateCovenantInformation(UUID covenant) - { - throw new System.NotImplementedException(); - } - - public void SendDetailedEstateData(UUID invoice, string estateName, uint estateID, uint parentEstate, uint estateFlags, uint sunPosition, UUID covenant, string abuseEmail, UUID estateOwner) - { - throw new System.NotImplementedException(); - } - - public void SendLandProperties(int sequence_id, bool snap_selection, int request_result, ILandObject lo, float simObjectBonusFactor, int parcelObjectCapacity, int simObjectCapacity, uint regionFlags) - { - throw new System.NotImplementedException(); - } - - public void SendLandAccessListData(List avatars, uint accessFlag, int localLandID) - { - throw new System.NotImplementedException(); - } - - public void SendForceClientSelectObjects(List objectIDs) - { - throw new System.NotImplementedException(); - } - - public void SendCameraConstraint(Vector4 ConstraintPlane) - { - - } - - public void SendLandObjectOwners(LandData land, List groups, Dictionary ownersAndCount) - { - throw new System.NotImplementedException(); - } - - public void SendLandParcelOverlay(byte[] data, int sequence_id) - { - throw new System.NotImplementedException(); - } - - public void SendParcelMediaCommand(uint flags, ParcelMediaCommandEnum command, float time) - { - throw new System.NotImplementedException(); - } - - public void SendParcelMediaUpdate(string mediaUrl, UUID mediaTextureID, byte autoScale, string mediaType, string mediaDesc, int mediaWidth, int mediaHeight, byte mediaLoop) - { - throw new System.NotImplementedException(); - } - - public void SendAssetUploadCompleteMessage(sbyte AssetType, bool Success, UUID AssetFullID) - { - throw new System.NotImplementedException(); - } - - public void SendConfirmXfer(ulong xferID, uint PacketID) - { - throw new System.NotImplementedException(); - } - - public void SendXferRequest(ulong XferID, short AssetType, UUID vFileID, byte FilePath, byte[] FileName) - { - throw new System.NotImplementedException(); - } - - public void SendInitiateDownload(string simFileName, string clientFileName) - { - throw new System.NotImplementedException(); - } - - public void SendImageFirstPart(ushort numParts, UUID ImageUUID, uint ImageSize, byte[] ImageData, byte imageCodec) - { - throw new System.NotImplementedException(); - } - - public void SendImageNextPart(ushort partNumber, UUID imageUuid, byte[] imageData) - { - throw new System.NotImplementedException(); - } - - public void SendImageNotFound(UUID imageid) - { - throw new System.NotImplementedException(); - } - - public void SendShutdownConnectionNotice() - { - throw new System.NotImplementedException(); - } - - public void SendSimStats(SimStats stats) - { - throw new System.NotImplementedException(); - } - - public void SendObjectPropertiesFamilyData(ISceneEntity Entity, uint RequestFlags) - { - throw new System.NotImplementedException(); - } - - public void SendObjectPropertiesReply(ISceneEntity entity) - { - throw new System.NotImplementedException(); - } - - public void SendAgentOffline(UUID[] agentIDs) - { - throw new System.NotImplementedException(); - } - - public void SendAgentOnline(UUID[] agentIDs) - { - throw new System.NotImplementedException(); - } - - public void SendSitResponse(UUID TargetID, Vector3 OffsetPos, Quaternion SitOrientation, bool autopilot, Vector3 CameraAtOffset, Vector3 CameraEyeOffset, bool ForceMouseLook) - { - throw new System.NotImplementedException(); - } - - public void SendAdminResponse(UUID Token, uint AdminLevel) - { - throw new System.NotImplementedException(); - } - - public void SendGroupMembership(GroupMembershipData[] GroupMembership) - { - throw new System.NotImplementedException(); - } - - public void SendGroupNameReply(UUID groupLLUID, string GroupName) - { - throw new System.NotImplementedException(); - } - - public void SendJoinGroupReply(UUID groupID, bool success) - { - throw new System.NotImplementedException(); - } - - public void SendEjectGroupMemberReply(UUID agentID, UUID groupID, bool success) - { - throw new System.NotImplementedException(); - } - - public void SendLeaveGroupReply(UUID groupID, bool success) - { - throw new System.NotImplementedException(); - } - - public void SendCreateGroupReply(UUID groupID, bool success, string message) - { - throw new System.NotImplementedException(); - } - - public void SendLandStatReply(uint reportType, uint requestFlags, uint resultCount, LandStatReportItem[] lsrpia) - { - throw new System.NotImplementedException(); - } - - public void SendScriptRunningReply(UUID objectID, UUID itemID, bool running) - { - throw new System.NotImplementedException(); - } - - public void SendAsset(AssetRequestToClient req) - { - throw new System.NotImplementedException(); - } - - public void SendTexture(AssetBase TextureAsset) - { - throw new System.NotImplementedException(); - } - - public byte[] GetThrottlesPacked(float multiplier) - { - throw new System.NotImplementedException(); - } - - public event ViewerEffectEventHandler OnViewerEffect; - public event Action OnLogout; - public event Action OnConnectionClosed; - public void SendBlueBoxMessage(UUID FromAvatarID, string FromAvatarName, string Message) - { - throw new System.NotImplementedException(); - } - - public void SendLogoutPacket() - { - throw new System.NotImplementedException(); - } - - public EndPoint GetClientEP() - { - return null; - } - - public ClientInfo GetClientInfo() - { - throw new System.NotImplementedException(); - } - - public void SetClientInfo(ClientInfo info) - { - throw new System.NotImplementedException(); - } - - public void SetClientOption(string option, string value) - { - throw new System.NotImplementedException(); - } - - public string GetClientOption(string option) - { - throw new System.NotImplementedException(); - } - - public void Terminate() - { - throw new System.NotImplementedException(); - } - - public void SendSetFollowCamProperties(UUID objectID, SortedDictionary parameters) - { - throw new System.NotImplementedException(); - } - - public void SendClearFollowCamProperties(UUID objectID) - { - throw new System.NotImplementedException(); - } - - public void SendRegionHandle(UUID regoinID, ulong handle) - { - throw new System.NotImplementedException(); - } - - public void SendParcelInfo(RegionInfo info, LandData land, UUID parcelID, uint x, uint y) - { - throw new System.NotImplementedException(); - } - - public void SendScriptTeleportRequest(string objName, string simName, Vector3 pos, Vector3 lookAt) - { - throw new System.NotImplementedException(); - } - - public void SendDirPlacesReply(UUID queryID, DirPlacesReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirPeopleReply(UUID queryID, DirPeopleReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirEventsReply(UUID queryID, DirEventsReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirGroupsReply(UUID queryID, DirGroupsReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirClassifiedReply(UUID queryID, DirClassifiedReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirLandReply(UUID queryID, DirLandReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendDirPopularReply(UUID queryID, DirPopularReplyData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendEventInfoReply(EventData info) - { - throw new System.NotImplementedException(); - } - - public void SendMapItemReply(mapItemReply[] replies, uint mapitemtype, uint flags) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarGroupsReply(UUID avatarID, GroupMembershipData[] data) - { - throw new System.NotImplementedException(); - } - - public void SendOfferCallingCard(UUID srcID, UUID transactionID) - { - throw new System.NotImplementedException(); - } - - public void SendAcceptCallingCard(UUID transactionID) - { - throw new System.NotImplementedException(); - } - - public void SendDeclineCallingCard(UUID transactionID) - { - throw new System.NotImplementedException(); - } - - public void SendTerminateFriend(UUID exFriendID) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarClassifiedReply(UUID targetID, UUID[] classifiedID, string[] name) - { - throw new System.NotImplementedException(); - } - - public void SendClassifiedInfoReply(UUID classifiedID, UUID creatorID, uint creationDate, uint expirationDate, uint category, string name, string description, UUID parcelID, uint parentEstate, UUID snapshotID, string simName, Vector3 globalPos, string parcelName, byte classifiedFlags, int price) - { - throw new System.NotImplementedException(); - } - - public void SendAgentDropGroup(UUID groupID) - { - throw new System.NotImplementedException(); - } - - public void RefreshGroupMembership() - { - throw new System.NotImplementedException(); - } - - public void SendAvatarNotesReply(UUID targetID, string text) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarPicksReply(UUID targetID, Dictionary picks) - { - throw new System.NotImplementedException(); - } - - public void SendPickInfoReply(UUID pickID, UUID creatorID, bool topPick, UUID parcelID, string name, string desc, UUID snapshotID, string user, string originalName, string simName, Vector3 posGlobal, int sortOrder, bool enabled) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarClassifiedReply(UUID targetID, Dictionary classifieds) - { - throw new System.NotImplementedException(); - } - - public void SendAvatarInterestUpdate(IClientAPI client, uint wantmask, string wanttext, uint skillsmask, string skillstext, string languages) - { - throw new System.NotImplementedException(); - } - - public void SendParcelDwellReply(int localID, UUID parcelID, float dwell) - { - throw new System.NotImplementedException(); - } - - public void SendUserInfoReply(bool imViaEmail, bool visible, string email) - { - throw new System.NotImplementedException(); - } - - public void SendUseCachedMuteList() - { - throw new System.NotImplementedException(); - } - - public void SendMuteListUpdate(string filename) - { - throw new System.NotImplementedException(); - } - - public void KillEndDone() - { - throw new System.NotImplementedException(); - } - - public bool AddGenericPacketHandler(string MethodName, GenericMessage handler) - { - throw new System.NotImplementedException(); - } - - #endregion - - public void SendRebakeAvatarTextures(UUID textureID) - { - } - - public void SendAvatarInterestsReply(UUID avatarID, uint wantMask, string wantText, uint skillsMask, string skillsText, string languages) - { - } - - public void SendGroupAccountingDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID, int amt) - { - } - - public void SendGroupAccountingSummary(IClientAPI sender,UUID groupID, uint moneyAmt, int totalTier, int usedTier) - { - } - - public void SendGroupTransactionsSummaryDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID,int amt) - { - } - - public void SendGroupVoteHistory(UUID groupID, UUID transactionID, GroupVoteHistory[] Votes) - { - } - - public void SendGroupActiveProposals(UUID groupID, UUID transactionID, GroupActiveProposals[] Proposals) - { - } - - public void SendChangeUserRights(UUID agentID, UUID friendID, int rights) - { - } - - public void SendTextBoxRequest(string message, int chatChannel, string objectname, string ownerFirstName, string ownerLastName, UUID objectId) - { - } - - public void StopFlying(ISceneEntity presence) - { - } - - public void SendPlacesReply(UUID queryID, UUID transactionID, PlacesReplyData[] data) - { - } - } -} diff --git a/OpenSim/Client/VWoHTTP/VWoHTTPModule.cs b/OpenSim/Client/VWoHTTP/VWoHTTPModule.cs deleted file mode 100644 index 31385ba0b2..0000000000 --- a/OpenSim/Client/VWoHTTP/VWoHTTPModule.cs +++ /dev/null @@ -1,133 +0,0 @@ -/* - * Copyright (c) Contributors, http://opensimulator.org/ - * See CONTRIBUTORS.TXT for a full list of copyright holders. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * * Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * * Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * * Neither the name of the OpenSimulator Project nor the - * names of its contributors may be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -using System; -using System.Collections.Generic; -using System.Text; -using Nini.Config; -using OpenMetaverse; -using OpenSim.Client.VWoHTTP.ClientStack; -using OpenSim.Framework; -using OpenSim.Framework.Servers; -using OpenSim.Framework.Servers.HttpServer; -using OpenSim.Region.Framework.Interfaces; -using OpenSim.Region.Framework.Scenes; - -namespace OpenSim.Client.VWoHTTP -{ - class VWoHTTPModule : IRegionModule, IHttpAgentHandler - { - private bool m_disabled = true; - - private IHttpServer m_httpd; - - private readonly List m_scenes = new List(); - - private Dictionary m_clients = new Dictionary(); - - #region Implementation of IRegionModule - - public void Initialise(Scene scene, IConfigSource source) - { - if (m_disabled) - return; - - m_scenes.Add(scene); - - m_httpd = MainServer.Instance; - } - - public void PostInitialise() - { - if (m_disabled) - return; - - m_httpd.AddAgentHandler("vwohttp", this); - } - - public void Close() - { - if (m_disabled) - return; - - m_httpd.RemoveAgentHandler("vwohttp", this); - } - - public string Name - { - get { return "VWoHTTP ClientStack"; } - } - - public bool IsSharedModule - { - get { return true; } - } - - #endregion - - #region Implementation of IHttpAgentHandler - - public bool Handle(OSHttpRequest req, OSHttpResponse resp) - { - string[] urlparts = req.Url.AbsolutePath.Split(new char[] {'/'}, StringSplitOptions.RemoveEmptyEntries); - - if (urlparts.Length < 2) - return false; - - if (urlparts[1] == "connect") - { - UUID sessID = UUID.Random(); - - VWHClientView client = new VWHClientView(sessID, UUID.Random(), "VWoHTTPClient", m_scenes[0]); - - m_clients.Add(sessID, client); - - return true; - } - else - { - if (urlparts.Length < 3) - return false; - - UUID sessionID; - if (!UUID.TryParse(urlparts[1], out sessionID)) - return false; - - if (!m_clients.ContainsKey(sessionID)) - return false; - - return m_clients[sessionID].ProcessInMsg(req, resp); - } - } - - public bool Match(OSHttpRequest req, OSHttpResponse resp) - { - return req.Url.ToString().Contains("vwohttp"); - } - - #endregion - } -} diff --git a/bin/MXP.dll b/bin/MXP.dll deleted file mode 100644 index 99e90c879c6747a1bdd19c50d472f7a5861590a1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 114688 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 c3l>yr0MA#wfdBvi diff --git a/bin/protobuf-net.dll b/bin/protobuf-net.dll deleted file mode 100644 index 057a41109423e6ac6fc71ad4899ec645ab6c6f0d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 107008 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 Omxl5G5&a+Hz<&d+^P;-| diff --git a/prebuild.xml b/prebuild.xml index 33a6b5005b..1102c11c09 100644 --- a/prebuild.xml +++ b/prebuild.xml @@ -1959,69 +1959,6 @@ - - - - - - ../../../bin/ - - - - - ../../../bin/ - - - - ../../../bin/ - - - - - - - - - - - - - - - - - - - - - - ../../../bin/ - - - - - ../../../bin/ - - - - ../../../bin/ - - - - - - - - - - - - - - - - - From 1505fbb647f72803efd39aa5e38ad39e6811f6fb Mon Sep 17 00:00:00 2001 From: Mic Bowman Date: Tue, 26 Apr 2011 08:54:05 -0700 Subject: [PATCH 30/45] Add back the high prioritization for other avatars in the BestAvatarResponsiveness prioritizer. --- OpenSim/Region/Framework/Scenes/Prioritizer.cs | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/OpenSim/Region/Framework/Scenes/Prioritizer.cs b/OpenSim/Region/Framework/Scenes/Prioritizer.cs index a7637c041c..4595a29c66 100644 --- a/OpenSim/Region/Framework/Scenes/Prioritizer.cs +++ b/OpenSim/Region/Framework/Scenes/Prioritizer.cs @@ -165,18 +165,20 @@ namespace OpenSim.Region.Framework.Scenes { if (!presence.IsChildAgent) { + // All avatars other than our own go into pqueue 1 + if (entity is ScenePresence) + return 1; + if (entity is SceneObjectPart) { + // Attachments are high priority, + if (((SceneObjectPart)entity).ParentGroup.RootPart.IsAttachment) + return 1; + // Non physical prims are lower priority than physical prims PhysicsActor physActor = ((SceneObjectPart)entity).ParentGroup.RootPart.PhysActor; if (physActor == null || !physActor.IsPhysical) pqueue++; - - // Attachments are high priority, - // MIC: shouldn't these already be in the highest priority queue already - // since their root position is same as the avatars? - if (((SceneObjectPart)entity).ParentGroup.RootPart.IsAttachment) - pqueue = 1; } } } From 16f6f55f2d203f9ef40fba85860bb9dbd416bd0f Mon Sep 17 00:00:00 2001 From: dahlia Date: Tue, 26 Apr 2011 16:22:44 -0700 Subject: [PATCH 31/45] network traffic reduction - decrease update frequency for moving avatars when velocity is unchanged --- .../Region/Framework/Scenes/ScenePresence.cs | 34 +++++++++++++------ 1 file changed, 24 insertions(+), 10 deletions(-) diff --git a/OpenSim/Region/Framework/Scenes/ScenePresence.cs b/OpenSim/Region/Framework/Scenes/ScenePresence.cs index ef0eb89fc9..e4413a95c9 100644 --- a/OpenSim/Region/Framework/Scenes/ScenePresence.cs +++ b/OpenSim/Region/Framework/Scenes/ScenePresence.cs @@ -2340,12 +2340,14 @@ namespace OpenSim.Region.Framework.Scenes #region Update Client(s) + /// /// Sends a location update to the client connected to this scenePresence /// /// public void SendTerseUpdateToClient(IClientAPI remoteClient) { + // If the client is inactive, it's getting its updates from another // server. if (remoteClient.IsActive) @@ -2358,8 +2360,8 @@ namespace OpenSim.Region.Framework.Scenes //m_log.DebugFormat("[SCENEPRESENCE]: TerseUpdate: Pos={0} Rot={1} Vel={2}", m_pos, m_bodyRot, m_velocity); remoteClient.SendPrimUpdate( - this, - PrimUpdateFlags.Position | PrimUpdateFlags.Rotation | PrimUpdateFlags.Velocity + this, + PrimUpdateFlags.Position | PrimUpdateFlags.Rotation | PrimUpdateFlags.Velocity | PrimUpdateFlags.Acceleration | PrimUpdateFlags.AngularVelocity); m_scene.StatsReporter.AddAgentTime(Util.EnvironmentTickCountSubtract(m_perfMonMS)); @@ -2367,16 +2369,31 @@ namespace OpenSim.Region.Framework.Scenes } } + + // vars to support reduced update frequency when velocity is unchanged + private Vector3 lastVelocitySentToAllClients = Vector3.Zero; + private int lastTerseUpdateToAllClientsTick = Util.EnvironmentTickCount(); + /// /// Send a location/velocity/accelleration update to all agents in scene /// public void SendTerseUpdateToAllClients() { - m_perfMonMS = Util.EnvironmentTickCount(); - - m_scene.ForEachClient(SendTerseUpdateToClient); + int currentTick = Util.EnvironmentTickCount(); - m_scene.StatsReporter.AddAgentTime(Util.EnvironmentTickCountSubtract(m_perfMonMS)); + // decrease update frequency when avatar is moving but velocity is not changing + if (m_velocity.Length() < 0.01f + || Vector3.Distance(lastVelocitySentToAllClients, m_velocity) > 0.01f + || currentTick - lastTerseUpdateToAllClientsTick > 1500) + { + m_perfMonMS = currentTick; + lastVelocitySentToAllClients = m_velocity; + lastTerseUpdateToAllClientsTick = currentTick; + + m_scene.ForEachClient(SendTerseUpdateToClient); + + m_scene.StatsReporter.AddAgentTime(Util.EnvironmentTickCountSubtract(m_perfMonMS)); + } } public void SendCoarseLocations(List coarseLocations, List avatarUUIDs) @@ -3268,10 +3285,7 @@ namespace OpenSim.Region.Framework.Scenes m_updateflag = true; - // The magic constant 0.95f seems to make walking feel less jerky, - // probably because it hackishly accounts for the overall latency of - // these Velocity updates -- Diva - Velocity = force * .95F; + Velocity = force; m_forceToApply = null; } From d722f22102f1f7ca618cc220e79c5f2097f84a32 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Wed, 27 Apr 2011 01:53:57 +0100 Subject: [PATCH 32/45] adjust freeswitch grid connector to port 8003 to agree with the default 8003 used in OpenSim.ini.example Adjusting to 8003 seems sensible since this service doesn't need to be exposed to viewers. --- bin/Robust.HG.ini.example | 2 +- bin/Robust.ini.example | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/bin/Robust.HG.ini.example b/bin/Robust.HG.ini.example index e2e9624831..daeb068332 100644 --- a/bin/Robust.HG.ini.example +++ b/bin/Robust.HG.ini.example @@ -21,7 +21,7 @@ ; * [[@]/][:] ; * [Startup] -ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8002/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector,8002/OpenSim.Server.Handlers.dll:GatekeeperServiceInConnector,8002/OpenSim.Server.Handlers.dll:UserAgentServerConnector,HGInventoryService@8002/OpenSim.Server.Handlers.dll:XInventoryInConnector,HGAssetService@8002/OpenSim.Server.Handlers.dll:AssetServiceConnector,8002/OpenSim.Server.Handlers.dll:HeloServiceInConnector" +ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8003/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector,8002/OpenSim.Server.Handlers.dll:GatekeeperServiceInConnector,8002/OpenSim.Server.Handlers.dll:UserAgentServerConnector,HGInventoryService@8002/OpenSim.Server.Handlers.dll:XInventoryInConnector,HGAssetService@8002/OpenSim.Server.Handlers.dll:AssetServiceConnector,8002/OpenSim.Server.Handlers.dll:HeloServiceInConnector" ; * This is common for all services, it's the network setup for the entire ; * server instance, if none is specified above diff --git a/bin/Robust.ini.example b/bin/Robust.ini.example index 4d16236044..9c391eaa25 100644 --- a/bin/Robust.ini.example +++ b/bin/Robust.ini.example @@ -13,7 +13,7 @@ ; * [[@]/][:] ; * [Startup] -ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8002/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector" +ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8003/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector" ; * This is common for all services, it's the network setup for the entire ; * server instance, if none is specified above From 1b43c05c3d8444faca0126bf195df0b0d99b6e83 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Wed, 27 Apr 2011 02:05:16 +0100 Subject: [PATCH 33/45] Change default ini.example port used by Freeswitch in grid settings to 8004. This is to facilitate running Freeswitch in a separate ROBUST shell, as on grid setups there can be quite a lot of traffic flowing between FreeSWITCH and ROBUST. However, running in a spearate shell is not mandatory. --- bin/OpenSim.ini.example | 2 +- bin/Robust.HG.ini.example | 2 +- bin/Robust.ini.example | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/bin/OpenSim.ini.example b/bin/OpenSim.ini.example index fbaa590aed..5389b4455a 100644 --- a/bin/OpenSim.ini.example +++ b/bin/OpenSim.ini.example @@ -626,7 +626,7 @@ ; LocalServiceModule = OpenSim.Services.Connectors.dll:RemoteFreeswitchConnector ;; If using a remote connector, specify the server URL - ; FreeswitchServiceURL = http://my.grid.server:8003/fsapi + ; FreeswitchServiceURL = http://my.grid.server:8004/fsapi [FreeswitchService] diff --git a/bin/Robust.HG.ini.example b/bin/Robust.HG.ini.example index daeb068332..f2f2a66992 100644 --- a/bin/Robust.HG.ini.example +++ b/bin/Robust.HG.ini.example @@ -21,7 +21,7 @@ ; * [[@]/][:] ; * [Startup] -ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8003/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector,8002/OpenSim.Server.Handlers.dll:GatekeeperServiceInConnector,8002/OpenSim.Server.Handlers.dll:UserAgentServerConnector,HGInventoryService@8002/OpenSim.Server.Handlers.dll:XInventoryInConnector,HGAssetService@8002/OpenSim.Server.Handlers.dll:AssetServiceConnector,8002/OpenSim.Server.Handlers.dll:HeloServiceInConnector" +ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8004/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector,8002/OpenSim.Server.Handlers.dll:GatekeeperServiceInConnector,8002/OpenSim.Server.Handlers.dll:UserAgentServerConnector,HGInventoryService@8002/OpenSim.Server.Handlers.dll:XInventoryInConnector,HGAssetService@8002/OpenSim.Server.Handlers.dll:AssetServiceConnector,8002/OpenSim.Server.Handlers.dll:HeloServiceInConnector" ; * This is common for all services, it's the network setup for the entire ; * server instance, if none is specified above diff --git a/bin/Robust.ini.example b/bin/Robust.ini.example index 9c391eaa25..aef0596a03 100644 --- a/bin/Robust.ini.example +++ b/bin/Robust.ini.example @@ -13,7 +13,7 @@ ; * [[@]/][:] ; * [Startup] -ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8003/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector" +ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003/OpenSim.Server.Handlers.dll:XInventoryInConnector,8004/OpenSim.Server.Handlers.dll:FreeswitchServerConnector,8003/OpenSim.Server.Handlers.dll:GridServiceConnector,8002/OpenSim.Server.Handlers.dll:GridInfoServerInConnector,8003/OpenSim.Server.Handlers.dll:AuthenticationServiceConnector,8002/OpenSim.Server.Handlers.dll:OpenIdServerConnector,8003/OpenSim.Server.Handlers.dll:AvatarServiceConnector,8002/OpenSim.Server.Handlers.dll:LLLoginServiceInConnector,8003/OpenSim.Server.Handlers.dll:PresenceServiceConnector,8003/OpenSim.Server.Handlers.dll:UserAccountServiceConnector,8003/OpenSim.Server.Handlers.dll:GridUserServiceConnector,8003/OpenSim.Server.Handlers.dll:FriendsServiceConnector" ; * This is common for all services, it's the network setup for the entire ; * server instance, if none is specified above From 12010849b822ad5db564ac9894b84de0c96d72b2 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Wed, 27 Apr 2011 02:25:49 +0100 Subject: [PATCH 34/45] Bump minimum required mono to 2.4.3 from 2.4.2. OpenSim fails at runtime below 2.4.3 due to unimplemented ServicePointManager.get_ServerCertificateValidationCallback() --- README.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.txt b/README.txt index 391a6599d1..72702a231d 100644 --- a/README.txt +++ b/README.txt @@ -37,7 +37,7 @@ Now see the "Configuring OpenSim" section === Running OpenSim on Linux === ================================ -You will need Mono >= 2.4.2 to run OpenSim. On some Linux distributions you +You will need Mono >= 2.4.3 to run OpenSim. On some Linux distributions you may need to install additional packages. See http://opensimulator.org/wiki/Dependencies for more information. From e0576b56d376d6bc7b9c5c3818acbdbcdb0dc56f Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Wed, 27 Apr 2011 07:02:37 -0700 Subject: [PATCH 35/45] Thank you Snoopy for a patch that adds some filtering to client versions allowed at login and HG-login times. NOTE: additional (optional) configuration variables in [LoginService] and [GatekeeperService]. See .examples. --- .../HypergridService/GatekeeperService.cs | 39 +++++++++++++++++- .../Services/LLLoginService/LLLoginService.cs | 40 +++++++++++++++++-- bin/Robust.HG.ini.example | 33 +++++++++++++++ bin/Robust.ini.example | 17 ++++++++ .../StandaloneCommon.ini.example | 34 ++++++++++++++++ 5 files changed, 158 insertions(+), 5 deletions(-) diff --git a/OpenSim/Services/HypergridService/GatekeeperService.cs b/OpenSim/Services/HypergridService/GatekeeperService.cs index b66bfed42f..9385b8df74 100644 --- a/OpenSim/Services/HypergridService/GatekeeperService.cs +++ b/OpenSim/Services/HypergridService/GatekeeperService.cs @@ -29,6 +29,7 @@ using System; using System.Collections.Generic; using System.Net; using System.Reflection; +using System.Text.RegularExpressions; using OpenSim.Framework; using OpenSim.Services.Interfaces; @@ -57,6 +58,9 @@ namespace OpenSim.Services.HypergridService private static IUserAgentService m_UserAgentService; private static ISimulationService m_SimulationService; + protected string m_AllowedClients = string.Empty; + protected string m_DeniedClients = string.Empty; + private static UUID m_ScopeID; private static bool m_AllowTeleportsToAnyRegion; private static string m_ExternalName; @@ -104,6 +108,9 @@ namespace OpenSim.Services.HypergridService else if (simulationService != string.Empty) m_SimulationService = ServerUtils.LoadPlugin(simulationService, args); + m_AllowedClients = serverConfig.GetString("AllowedClients", string.Empty); + m_DeniedClients = serverConfig.GetString("DeniedClients", string.Empty); + if (m_GridService == null || m_PresenceService == null || m_SimulationService == null) throw new Exception("Unable to load a required plugin, Gatekeeper Service cannot function."); @@ -181,8 +188,36 @@ namespace OpenSim.Services.HypergridService string authURL = string.Empty; if (aCircuit.ServiceURLs.ContainsKey("HomeURI")) authURL = aCircuit.ServiceURLs["HomeURI"].ToString(); - m_log.DebugFormat("[GATEKEEPER SERVICE]: Request to login foreign agent {0} {1} @ {2} ({3}) at destination {4}", - aCircuit.firstname, aCircuit.lastname, authURL, aCircuit.AgentID, destination.RegionName); + m_log.InfoFormat("[GATEKEEPER SERVICE]: Login request for {0} {1} @ {2} ({3}) at {4} using viewer {5}, channel {6}, IP {7}, Mac {8}, Id0 {9}", + aCircuit.firstname, aCircuit.lastname, authURL, aCircuit.AgentID, destination.RegionName, + aCircuit.Viewer, aCircuit.Channel, aCircuit.IPAddress, aCircuit.Mac, aCircuit.Id0); + + // + // Check client + // + if (m_AllowedClients != string.Empty) + { + Regex arx = new Regex(m_AllowedClients); + Match am = arx.Match(aCircuit.Viewer); + + if (!am.Success) + { + m_log.InfoFormat("[GATEKEEPER SERVICE]: Login failed, reason: client {0} is not allowed", aCircuit.Viewer); + return false; + } + } + + if (m_DeniedClients != string.Empty) + { + Regex drx = new Regex(m_DeniedClients); + Match dm = drx.Match(aCircuit.Viewer); + + if (dm.Success) + { + m_log.InfoFormat("[GATEKEEPER SERVICE]: Login failed, reason: client {0} is denied", aCircuit.Viewer); + return false; + } + } // // Authenticate the user diff --git a/OpenSim/Services/LLLoginService/LLLoginService.cs b/OpenSim/Services/LLLoginService/LLLoginService.cs index d364aa4b9f..9bcc3dd225 100644 --- a/OpenSim/Services/LLLoginService/LLLoginService.cs +++ b/OpenSim/Services/LLLoginService/LLLoginService.cs @@ -77,7 +77,11 @@ namespace OpenSim.Services.LLLoginService protected string m_MapTileURL; protected string m_SearchURL; + protected string m_AllowedClients; + protected string m_DeniedClients; + IConfig m_LoginServerConfig; + IConfig m_ClientsConfig; public LLLoginService(IConfigSource config, ISimulationService simService, ILibraryService libraryService) { @@ -105,7 +109,10 @@ namespace OpenSim.Services.LLLoginService m_GatekeeperURL = m_LoginServerConfig.GetString("GatekeeperURI", string.Empty); m_MapTileURL = m_LoginServerConfig.GetString("MapTileURL", string.Empty); m_SearchURL = m_LoginServerConfig.GetString("SearchURL", string.Empty); - + + m_AllowedClients = m_LoginServerConfig.GetString("AllowedClients", string.Empty); + m_DeniedClients = m_LoginServerConfig.GetString("DeniedClients", string.Empty); + // These are required; the others aren't if (accountService == string.Empty || authService == string.Empty) throw new Exception("LoginService is missing service specifications"); @@ -215,10 +222,37 @@ namespace OpenSim.Services.LLLoginService bool success = false; UUID session = UUID.Random(); - m_log.InfoFormat("[LLOGIN SERVICE]: Login request for {0} {1} from {2} with user agent {3} starting in {4}", - firstName, lastName, clientIP.Address.ToString(), clientVersion, startLocation); + m_log.InfoFormat("[LLOGIN SERVICE]: Login request for {0} {1} at {2} using viewer {3}, channel {4}, IP {5}, Mac {6}, Id0 {7}", + firstName, lastName, startLocation, clientVersion, channel, clientIP.Address.ToString(), mac, id0); try { + // + // Check client + // + if (m_AllowedClients != string.Empty) + { + Regex arx = new Regex(m_AllowedClients); + Match am = arx.Match(clientVersion); + + if (!am.Success) + { + m_log.InfoFormat("[LLOGIN SERVICE]: Login failed, reason: client {0} is not allowed", clientVersion); + return LLFailedLoginResponse.LoginBlockedProblem; + } + } + + if (m_DeniedClients != string.Empty) + { + Regex drx = new Regex(m_DeniedClients); + Match dm = drx.Match(clientVersion); + + if (dm.Success) + { + m_log.InfoFormat("[LLOGIN SERVICE]: Login failed, reason: client {0} is denied", clientVersion); + return LLFailedLoginResponse.LoginBlockedProblem; + } + } + // // Get the account and check that it exists // diff --git a/bin/Robust.HG.ini.example b/bin/Robust.HG.ini.example index f2f2a66992..572497c0b3 100644 --- a/bin/Robust.HG.ini.example +++ b/bin/Robust.HG.ini.example @@ -197,6 +197,23 @@ ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003 SRV_AssetServerURI = "http://127.0.0.1:8002" SRV_ProfileServerURI = "http://127.0.0.1:8002/user" + ;; Regular expressions for controlling which client versions are accepted/denied. + ;; An empty string means nothing is checked. + ;; + ;; Example 1: allow only these 3 types of clients (any version of them) + ;; AllowedClients = "Imprudence|Hippo|Second Life" + ;; + ;; Example 2: allow all clients except these + ;; DeniedClients = "Twisted|Crawler|Cryolife|FuckLife|StreetLife|GreenLife|AntiLife|KORE-Phaze|Synlyfe|Purple Second Life|SecondLi |Emerald" + ;; + ;; Note that these are regular expressions, so every character counts. + ;; Also note that this is very weak security and should not be trusted as a reliable means + ;; for keeping bad clients out; modified clients can fake their identifiers. + ;; + ;; + ;AllowedClients = "" + ;DeniedClients = "" + [GridInfoService] ; These settings are used to return information on a get_grid_info call. ; Client launcher scripts and third-party clients make use of this to @@ -256,6 +273,22 @@ ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003 ; If you run this gatekeeper server behind a proxy, set this to true ; HasProxy = false + ;; Regular expressions for controlling which client versions are accepted/denied. + ;; An empty string means nothing is checked. + ;; + ;; Example 1: allow only these 3 types of clients (any version of them) + ;; AllowedClients = "Imprudence|Hippo|Second Life" + ;; + ;; Example 2: allow all clients except these + ;; DeniedClients = "Twisted|Crawler|Cryolife|FuckLife|StreetLife|GreenLife|AntiLife|KORE-Phaze|Synlyfe|Purple Second Life|SecondLi |Emerald" + ;; + ;; Note that these are regular expressions, so every character counts. + ;; Also note that this is very weak security and should not be trusted as a reliable means + ;; for keeping bad clients out; modified clients can fake their identifiers. + ;; + ;; + ;AllowedClients = "" + ;DeniedClients = "" [UserAgentService] LocalServiceModule = "OpenSim.Services.HypergridService.dll:UserAgentService" diff --git a/bin/Robust.ini.example b/bin/Robust.ini.example index aef0596a03..047e9eeed6 100644 --- a/bin/Robust.ini.example +++ b/bin/Robust.ini.example @@ -176,6 +176,23 @@ ServiceConnectors = "8003/OpenSim.Server.Handlers.dll:AssetServiceConnector,8003 ; If you run this login server behind a proxy, set this to true ; HasProxy = false + ;; Regular expressions for controlling which client versions are accepted/denied. + ;; An empty string means nothing is checked. + ;; + ;; Example 1: allow only these 3 types of clients (any version of them) + ;; AllowedClients = "Imprudence|Hippo|Second Life" + ;; + ;; Example 2: allow all clients except these + ;; DeniedClients = "Twisted|Crawler|Cryolife|FuckLife|StreetLife|GreenLife|AntiLife|KORE-Phaze|Synlyfe|Purple Second Life|SecondLi |Emerald" + ;; + ;; Note that these are regular expressions, so every character counts. + ;; Also note that this is very weak security and should not be trusted as a reliable means + ;; for keeping bad clients out; modified clients can fake their identifiers. + ;; + ;; + ;AllowedClients = "" + ;DeniedClients = "" + [GridInfoService] ; These settings are used to return information on a get_grid_info call. ; Client launcher scripts and third-party clients make use of this to diff --git a/bin/config-include/StandaloneCommon.ini.example b/bin/config-include/StandaloneCommon.ini.example index dcebd63184..67efa11564 100644 --- a/bin/config-include/StandaloneCommon.ini.example +++ b/bin/config-include/StandaloneCommon.ini.example @@ -83,6 +83,23 @@ SRV_AssetServerURI = "http://127.0.0.1:9000" SRV_ProfileServerURI = "http://127.0.0.1:9000" + ;; Regular expressions for controlling which client versions are accepted/denied. + ;; An empty string means nothing is checked. + ;; + ;; Example 1: allow only these 3 types of clients (any version of them) + ;; AllowedClients = "Imprudence|Hippo|Second Life" + ;; + ;; Example 2: allow all clients except these + ;; DeniedClients = "Twisted|Crawler|Cryolife|FuckLife|StreetLife|GreenLife|AntiLife|KORE-Phaze|Synlyfe|Purple Second Life|SecondLi |Emerald" + ;; + ;; Note that these are regular expressions, so every character counts. + ;; Also note that this is very weak security and should not be trusted as a reliable means + ;; for keeping bad clients out; modified clients can fake their identifiers. + ;; + ;; + ;AllowedClients = "" + ;DeniedClients = "" + [GatekeeperService] ExternalName = "http://127.0.0.1:9000" @@ -90,6 +107,23 @@ ; If false, HG TPs happen only to the Default regions specified in [GridService] section AllowTeleportsToAnyRegion = true + ;; Regular expressions for controlling which client versions are accepted/denied. + ;; An empty string means nothing is checked. + ;; + ;; Example 1: allow only these 3 types of clients (any version of them) + ;; AllowedClients = "Imprudence|Hippo|Second Life" + ;; + ;; Example 2: allow all clients except these + ;; DeniedClients = "Twisted|Crawler|Cryolife|FuckLife|StreetLife|GreenLife|AntiLife|KORE-Phaze|Synlyfe|Purple Second Life|SecondLi |Emerald" + ;; + ;; Note that these are regular expressions, so every character counts. + ;; Also note that this is very weak security and should not be trusted as a reliable means + ;; for keeping bad clients out; modified clients can fake their identifiers. + ;; + ;; + ;AllowedClients = "" + ;DeniedClients = "" + [GridInfoService] ; These settings are used to return information on a get_grid_info call. ; Client launcher scripts and third-party clients make use of this to From 5e3893ca5c8c985bf9b2a1e1dbc94d48a4eb3e96 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Wed, 27 Apr 2011 23:57:26 +0100 Subject: [PATCH 36/45] add the executable bit to all bundled DLLs so that these are preserved when using git or untarring releases under cygwin on windows Mono doesn't care if the --- bin/Axiom.MathLib.dll | Bin bin/BulletDotNET.dll | Bin bin/C5.dll | Bin bin/CSJ2K.dll | Bin bin/Community.CsharpSqlite.Sqlite.dll | Bin bin/Community.CsharpSqlite.dll | Bin bin/CookComputing.XmlRpcV2.dll | Bin bin/DotNetOpenId.dll | Bin bin/DotNetOpenMail.dll | Bin bin/Fadd.Globalization.Yaml.dll | Bin bin/Fadd.dll | Bin bin/GlynnTucker.Cache.dll | Bin bin/Google.ProtocolBuffers.dll | Bin bin/HttpServer.dll | Bin bin/HttpServer_OpenSim.dll | Bin bin/Iesi.Collections.dll | Bin bin/Kds.Serialization.dll | Bin bin/Modified.XnaDevRu.BulletX.dll | Bin bin/Mono.Addins.CecilReflector.dll | Bin bin/Mono.Addins.Setup.dll | Bin bin/Mono.Addins.dll | Bin bin/Mono.Data.Sqlite.dll | Bin bin/Mono.Data.SqliteClient.dll | Bin bin/MonoXnaCompactMaths.dll | Bin bin/MySql.Data.dll | Bin bin/NDesk.Options.dll | Bin bin/Newtonsoft.Json.Net20.dll | Bin bin/Nini.dll | Bin bin/Npgsql.dll | Bin bin/Ode.NET.dll | Bin bin/OpenMetaverse.StructuredData.dll | Bin bin/OpenMetaverse.dll | Bin bin/OpenMetaverseTypes.dll | Bin bin/PhysX-wrapper.dll | Bin bin/PhysX_Wrapper_Dotnet.dll | Bin bin/PrimMesher.dll | Bin bin/PumaCode.SvnDotNet.dll | Bin bin/Tools.dll | Bin bin/Warp3D.dll | Bin bin/XMLRPC.dll | Bin bin/libbulletnet.dll | Bin bin/libdb_dotNET43.dll | Bin bin/log4net.dll | Bin bin/nunit.framework.dll | Bin bin/ode.dll | Bin bin/openjpeg-dotnet-x86_64.dll | Bin bin/openjpeg-dotnet.dll | Bin bin/xunit.dll | Bin bin/zlib.net.dll | Bin 49 files changed, 0 insertions(+), 0 deletions(-) mode change 100644 => 100755 bin/Axiom.MathLib.dll mode change 100644 => 100755 bin/BulletDotNET.dll mode change 100644 => 100755 bin/C5.dll mode change 100644 => 100755 bin/CSJ2K.dll mode change 100644 => 100755 bin/Community.CsharpSqlite.Sqlite.dll mode change 100644 => 100755 bin/Community.CsharpSqlite.dll mode change 100644 => 100755 bin/CookComputing.XmlRpcV2.dll mode change 100644 => 100755 bin/DotNetOpenId.dll mode change 100644 => 100755 bin/DotNetOpenMail.dll mode change 100644 => 100755 bin/Fadd.Globalization.Yaml.dll mode change 100644 => 100755 bin/Fadd.dll mode change 100644 => 100755 bin/GlynnTucker.Cache.dll mode change 100644 => 100755 bin/Google.ProtocolBuffers.dll mode change 100644 => 100755 bin/HttpServer.dll mode change 100644 => 100755 bin/HttpServer_OpenSim.dll mode change 100644 => 100755 bin/Iesi.Collections.dll mode change 100644 => 100755 bin/Kds.Serialization.dll mode change 100644 => 100755 bin/Modified.XnaDevRu.BulletX.dll mode change 100644 => 100755 bin/Mono.Addins.CecilReflector.dll mode change 100644 => 100755 bin/Mono.Addins.Setup.dll mode change 100644 => 100755 bin/Mono.Addins.dll mode change 100644 => 100755 bin/Mono.Data.Sqlite.dll mode change 100644 => 100755 bin/Mono.Data.SqliteClient.dll mode change 100644 => 100755 bin/MonoXnaCompactMaths.dll mode change 100644 => 100755 bin/MySql.Data.dll mode change 100644 => 100755 bin/NDesk.Options.dll mode change 100644 => 100755 bin/Newtonsoft.Json.Net20.dll mode change 100644 => 100755 bin/Nini.dll mode change 100644 => 100755 bin/Npgsql.dll mode change 100644 => 100755 bin/Ode.NET.dll mode change 100644 => 100755 bin/OpenMetaverse.StructuredData.dll mode change 100644 => 100755 bin/OpenMetaverse.dll mode change 100644 => 100755 bin/OpenMetaverseTypes.dll mode change 100644 => 100755 bin/PhysX-wrapper.dll mode change 100644 => 100755 bin/PhysX_Wrapper_Dotnet.dll mode change 100644 => 100755 bin/PrimMesher.dll mode change 100644 => 100755 bin/PumaCode.SvnDotNet.dll mode change 100644 => 100755 bin/Tools.dll mode change 100644 => 100755 bin/Warp3D.dll mode change 100644 => 100755 bin/XMLRPC.dll mode change 100644 => 100755 bin/libbulletnet.dll mode change 100644 => 100755 bin/libdb_dotNET43.dll mode change 100644 => 100755 bin/log4net.dll mode change 100644 => 100755 bin/nunit.framework.dll mode change 100644 => 100755 bin/ode.dll mode change 100644 => 100755 bin/openjpeg-dotnet-x86_64.dll mode change 100644 => 100755 bin/openjpeg-dotnet.dll mode change 100644 => 100755 bin/xunit.dll mode change 100644 => 100755 bin/zlib.net.dll diff --git a/bin/Axiom.MathLib.dll b/bin/Axiom.MathLib.dll old mode 100644 new mode 100755 diff --git a/bin/BulletDotNET.dll b/bin/BulletDotNET.dll old mode 100644 new mode 100755 diff --git a/bin/C5.dll b/bin/C5.dll old mode 100644 new mode 100755 diff --git a/bin/CSJ2K.dll b/bin/CSJ2K.dll old mode 100644 new mode 100755 diff --git a/bin/Community.CsharpSqlite.Sqlite.dll b/bin/Community.CsharpSqlite.Sqlite.dll old mode 100644 new mode 100755 diff --git a/bin/Community.CsharpSqlite.dll b/bin/Community.CsharpSqlite.dll old mode 100644 new mode 100755 diff --git a/bin/CookComputing.XmlRpcV2.dll b/bin/CookComputing.XmlRpcV2.dll old mode 100644 new mode 100755 diff --git a/bin/DotNetOpenId.dll b/bin/DotNetOpenId.dll old mode 100644 new mode 100755 diff --git a/bin/DotNetOpenMail.dll b/bin/DotNetOpenMail.dll old mode 100644 new mode 100755 diff --git a/bin/Fadd.Globalization.Yaml.dll b/bin/Fadd.Globalization.Yaml.dll old mode 100644 new mode 100755 diff --git a/bin/Fadd.dll b/bin/Fadd.dll old mode 100644 new mode 100755 diff --git a/bin/GlynnTucker.Cache.dll b/bin/GlynnTucker.Cache.dll old mode 100644 new mode 100755 diff --git a/bin/Google.ProtocolBuffers.dll b/bin/Google.ProtocolBuffers.dll old mode 100644 new mode 100755 diff --git a/bin/HttpServer.dll b/bin/HttpServer.dll old mode 100644 new mode 100755 diff --git a/bin/HttpServer_OpenSim.dll b/bin/HttpServer_OpenSim.dll old mode 100644 new mode 100755 diff --git a/bin/Iesi.Collections.dll b/bin/Iesi.Collections.dll old mode 100644 new mode 100755 diff --git a/bin/Kds.Serialization.dll b/bin/Kds.Serialization.dll old mode 100644 new mode 100755 diff --git a/bin/Modified.XnaDevRu.BulletX.dll b/bin/Modified.XnaDevRu.BulletX.dll old mode 100644 new mode 100755 diff --git a/bin/Mono.Addins.CecilReflector.dll b/bin/Mono.Addins.CecilReflector.dll old mode 100644 new mode 100755 diff --git a/bin/Mono.Addins.Setup.dll b/bin/Mono.Addins.Setup.dll old mode 100644 new mode 100755 diff --git a/bin/Mono.Addins.dll b/bin/Mono.Addins.dll old mode 100644 new mode 100755 diff --git a/bin/Mono.Data.Sqlite.dll b/bin/Mono.Data.Sqlite.dll old mode 100644 new mode 100755 diff --git a/bin/Mono.Data.SqliteClient.dll b/bin/Mono.Data.SqliteClient.dll old mode 100644 new mode 100755 diff --git a/bin/MonoXnaCompactMaths.dll b/bin/MonoXnaCompactMaths.dll old mode 100644 new mode 100755 diff --git a/bin/MySql.Data.dll b/bin/MySql.Data.dll old mode 100644 new mode 100755 diff --git a/bin/NDesk.Options.dll b/bin/NDesk.Options.dll old mode 100644 new mode 100755 diff --git a/bin/Newtonsoft.Json.Net20.dll b/bin/Newtonsoft.Json.Net20.dll old mode 100644 new mode 100755 diff --git a/bin/Nini.dll b/bin/Nini.dll old mode 100644 new mode 100755 diff --git a/bin/Npgsql.dll b/bin/Npgsql.dll old mode 100644 new mode 100755 diff --git a/bin/Ode.NET.dll b/bin/Ode.NET.dll old mode 100644 new mode 100755 diff --git a/bin/OpenMetaverse.StructuredData.dll b/bin/OpenMetaverse.StructuredData.dll old mode 100644 new mode 100755 diff --git a/bin/OpenMetaverse.dll b/bin/OpenMetaverse.dll old mode 100644 new mode 100755 diff --git a/bin/OpenMetaverseTypes.dll b/bin/OpenMetaverseTypes.dll old mode 100644 new mode 100755 diff --git a/bin/PhysX-wrapper.dll b/bin/PhysX-wrapper.dll old mode 100644 new mode 100755 diff --git a/bin/PhysX_Wrapper_Dotnet.dll b/bin/PhysX_Wrapper_Dotnet.dll old mode 100644 new mode 100755 diff --git a/bin/PrimMesher.dll b/bin/PrimMesher.dll old mode 100644 new mode 100755 diff --git a/bin/PumaCode.SvnDotNet.dll b/bin/PumaCode.SvnDotNet.dll old mode 100644 new mode 100755 diff --git a/bin/Tools.dll b/bin/Tools.dll old mode 100644 new mode 100755 diff --git a/bin/Warp3D.dll b/bin/Warp3D.dll old mode 100644 new mode 100755 diff --git a/bin/XMLRPC.dll b/bin/XMLRPC.dll old mode 100644 new mode 100755 diff --git a/bin/libbulletnet.dll b/bin/libbulletnet.dll old mode 100644 new mode 100755 diff --git a/bin/libdb_dotNET43.dll b/bin/libdb_dotNET43.dll old mode 100644 new mode 100755 diff --git a/bin/log4net.dll b/bin/log4net.dll old mode 100644 new mode 100755 diff --git a/bin/nunit.framework.dll b/bin/nunit.framework.dll old mode 100644 new mode 100755 diff --git a/bin/ode.dll b/bin/ode.dll old mode 100644 new mode 100755 diff --git a/bin/openjpeg-dotnet-x86_64.dll b/bin/openjpeg-dotnet-x86_64.dll old mode 100644 new mode 100755 diff --git a/bin/openjpeg-dotnet.dll b/bin/openjpeg-dotnet.dll old mode 100644 new mode 100755 diff --git a/bin/xunit.dll b/bin/xunit.dll old mode 100644 new mode 100755 diff --git a/bin/zlib.net.dll b/bin/zlib.net.dll old mode 100644 new mode 100755 From 50aa93137d052c643a6ed44c32f0b4c5b32da79e Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Thu, 28 Apr 2011 00:59:21 +0100 Subject: [PATCH 37/45] Fix a bug where physical objects rezzed with an initial velocity by script do not receive this velocity. This is a minimal fix for the 0.7.1 release, pending a non copy/paste solution. This hopefully addresses http://opensimulator.org/mantis/view.php?id=5457 The bug was introduced in commit 3ba5eeb --- OpenSim/Region/Framework/Scenes/Scene.Inventory.cs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs b/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs index 0f85925c78..0b2b01a944 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs @@ -2039,6 +2039,12 @@ namespace OpenSim.Region.Framework.Scenes if (rot != null) group.UpdateGroupRotationR((Quaternion)rot); + // TODO: This needs to be refactored with the similar code in + // SceneGraph.AddNewSceneObject(SceneObjectGroup sceneObject, bool attachToBackup, Vector3 pos, Quaternion rot, Vector3 vel) + // possibly by allowing this method to take a null rotation. + if (group.RootPart.PhysActor != null && group.RootPart.PhysActor.IsPhysical && vel != Vector3.Zero) + group.RootPart.ApplyImpulse((vel * group.GetMass()), false); + // We can only call this after adding the scene object, since the scene object references the scene // to find out if scripts should be activated at all. group.CreateScriptInstances(param, true, DefaultScriptEngine, 3); From 7ecf6895842eff93155e49d2f7fa4cd833510726 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Thu, 28 Apr 2011 07:28:29 -0700 Subject: [PATCH 38/45] Thank you MrMonkE for a patch that seems to bring the MSSQL data layer up to speed with 0.7.x. --- .../Data/MSSQL/MSSQLGenericTableHandler.cs | 6 +- OpenSim/Data/MSSQL/MSSQLMigration.cs | 3 +- OpenSim/Data/MSSQL/MSSQLSimulationData.cs | 25 ++-- .../Data/MSSQL/Resources/AuthStore.migrations | 8 +- .../Data/MSSQL/Resources/Avatar.migrations | 25 +++- .../MSSQL/Resources/FriendsStore.migrations | 4 +- .../Data/MSSQL/Resources/GridStore.migrations | 13 ++ .../MSSQL/Resources/InventoryStore.migrations | 9 +- .../Data/MSSQL/Resources/Presence.migrations | 17 +-- .../MSSQL/Resources/RegionStore.migrations | 119 +++++++++++++++++- .../MSSQL/Resources/UserAccount.migrations | 2 +- 11 files changed, 199 insertions(+), 32 deletions(-) diff --git a/OpenSim/Data/MSSQL/MSSQLGenericTableHandler.cs b/OpenSim/Data/MSSQL/MSSQLGenericTableHandler.cs index 6a5d6eb58a..f5492b3050 100644 --- a/OpenSim/Data/MSSQL/MSSQLGenericTableHandler.cs +++ b/OpenSim/Data/MSSQL/MSSQLGenericTableHandler.cs @@ -56,15 +56,15 @@ namespace OpenSim.Data.MSSQL string realm, string storeName) { m_Realm = realm; + + m_ConnectionString = connectionString; if (storeName != String.Empty) { - Assembly assem = GetType().Assembly; - m_ConnectionString = connectionString; using (SqlConnection conn = new SqlConnection(m_ConnectionString)) { conn.Open(); - Migration m = new Migration(conn, assem, storeName); + Migration m = new Migration(conn, GetType().Assembly, storeName); m.Update(); } diff --git a/OpenSim/Data/MSSQL/MSSQLMigration.cs b/OpenSim/Data/MSSQL/MSSQLMigration.cs index aea31c9eaa..cd395b84d1 100644 --- a/OpenSim/Data/MSSQL/MSSQLMigration.cs +++ b/OpenSim/Data/MSSQL/MSSQLMigration.cs @@ -61,7 +61,8 @@ namespace OpenSim.Data.MSSQL } catch { - // Something went wrong, so we're version 0 + // Return -1 to indicate table does not exist + return -1; } } return version; diff --git a/OpenSim/Data/MSSQL/MSSQLSimulationData.cs b/OpenSim/Data/MSSQL/MSSQLSimulationData.cs index 80ec65e246..5155e567e6 100644 --- a/OpenSim/Data/MSSQL/MSSQLSimulationData.cs +++ b/OpenSim/Data/MSSQL/MSSQLSimulationData.cs @@ -492,12 +492,11 @@ ELSE using (SqlConnection conn = new SqlConnection(m_connectionString)) using (SqlCommand cmd = new SqlCommand(sql, conn)) { + conn.Open(); foreach (TaskInventoryItem taskItem in items) { cmd.Parameters.AddRange(CreatePrimInventoryParameters(taskItem)); - conn.Open(); cmd.ExecuteNonQuery(); - cmd.Parameters.Clear(); } } @@ -1154,9 +1153,9 @@ VALUES PrimitiveBaseShape baseShape = new PrimitiveBaseShape(); baseShape.Scale = new Vector3( - Convert.ToSingle(shapeRow["ScaleX"]), - Convert.ToSingle(shapeRow["ScaleY"]), - Convert.ToSingle(shapeRow["ScaleZ"])); + (float)Convert.ToDouble(shapeRow["ScaleX"]), + (float)Convert.ToDouble(shapeRow["ScaleY"]), + (float)Convert.ToDouble(shapeRow["ScaleZ"])); // paths baseShape.PCode = Convert.ToByte(shapeRow["PCode"]); @@ -1193,8 +1192,11 @@ VALUES { } - if (!(shapeRow["Media"] is System.DBNull)) + if (!(shapeRow["Media"] is System.DBNull) ) + { baseShape.Media = PrimitiveBaseShape.MediaList.FromXml((string)shapeRow["Media"]); + } + return baseShape; } @@ -1573,7 +1575,16 @@ VALUES parameters.Add(_Database.CreateParameter("Texture", s.TextureEntry)); parameters.Add(_Database.CreateParameter("ExtraParams", s.ExtraParams)); parameters.Add(_Database.CreateParameter("State", s.State)); - parameters.Add(_Database.CreateParameter("Media", null == s.Media ? null : s.Media.ToXml())); + + if(null == s.Media ) + { + parameters.Add(_Database.CreateParameter("Media", DBNull.Value)); + } + else + { + parameters.Add(_Database.CreateParameter("Media", s.Media.ToXml())); + } + return parameters.ToArray(); } diff --git a/OpenSim/Data/MSSQL/Resources/AuthStore.migrations b/OpenSim/Data/MSSQL/Resources/AuthStore.migrations index 5b90ca3d36..eb9129645f 100644 --- a/OpenSim/Data/MSSQL/Resources/AuthStore.migrations +++ b/OpenSim/Data/MSSQL/Resources/AuthStore.migrations @@ -22,7 +22,11 @@ COMMIT BEGIN TRANSACTION -INSERT INTO auth (UUID, passwordHash, passwordSalt, webLoginKey, accountType) SELECT [UUID] AS UUID, [passwordHash] AS passwordHash, [passwordSalt] AS passwordSalt, [webLoginKey] AS webLoginKey, 'UserAccount' as [accountType] FROM users; +IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[users]') AND type in (N'U')) + INSERT INTO auth (UUID, passwordHash, passwordSalt, webLoginKey, accountType) SELECT [UUID] AS UUID, [passwordHash] AS passwordHash, [passwordSalt] AS passwordSalt, [webLoginKey] AS webLoginKey, 'UserAccount' as [accountType] FROM users; + +COMMIT + + -COMMIT \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/Avatar.migrations b/OpenSim/Data/MSSQL/Resources/Avatar.migrations index 759e939caf..5364153363 100644 --- a/OpenSim/Data/MSSQL/Resources/Avatar.migrations +++ b/OpenSim/Data/MSSQL/Resources/Avatar.migrations @@ -13,5 +13,28 @@ PRIMARY KEY CLUSTERED ) ON [PRIMARY] +COMMIT + +:VERSION 2 + +BEGIN TRANSACTION + +CREATE TABLE dbo.Tmp_Avatars + ( + PrincipalID uniqueidentifier NOT NULL, + [Name] varchar(32) NOT NULL, + Value text NOT NULL DEFAULT '', + ) ON [PRIMARY] + TEXTIMAGE_ON [PRIMARY] + +IF EXISTS(SELECT * FROM dbo.Avatars) + EXEC('INSERT INTO dbo.Tmp_Avatars (PrincipalID, Name, Value) + SELECT PrincipalID, CONVERT(text, Name), Value FROM dbo.Avatars WITH (HOLDLOCK TABLOCKX)') + +DROP TABLE dbo.Avatars + +EXECUTE sp_rename N'dbo.Tmp_Avatars', N'Avatars', 'OBJECT' + +COMMIT + -COMMIT \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/FriendsStore.migrations b/OpenSim/Data/MSSQL/Resources/FriendsStore.migrations index f981a91999..4d8ab0f622 100644 --- a/OpenSim/Data/MSSQL/Resources/FriendsStore.migrations +++ b/OpenSim/Data/MSSQL/Resources/FriendsStore.migrations @@ -15,6 +15,8 @@ COMMIT BEGIN TRANSACTION -INSERT INTO Friends (PrincipalID, Friend, Flags, Offered) SELECT [ownerID], [friendID], [friendPerms], 0 FROM userfriends; +IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[userfriends]') AND type in (N'U')) +INSERT INTO Friends (PrincipalID, Friend, Flags, Offered) +SELECT [ownerID], [friendID], [friendPerms], 0 FROM userfriends; COMMIT \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/GridStore.migrations b/OpenSim/Data/MSSQL/Resources/GridStore.migrations index d2ca27a071..c6342fcdfd 100644 --- a/OpenSim/Data/MSSQL/Resources/GridStore.migrations +++ b/OpenSim/Data/MSSQL/Resources/GridStore.migrations @@ -222,4 +222,17 @@ ALTER TABLE [regions] ADD [Token] varchar(255) NOT NULL DEFAULT 0; COMMIT +:VERSION 8 +BEGIN TRANSACTION +ALTER TABLE regions ALTER COLUMN regionName VarChar(128) + +DROP INDEX IX_regions_name ON dbo.regions +ALTER TABLE regions ALTER COLUMN regionName VarChar(128) null + +CREATE NONCLUSTERED INDEX IX_regions_name ON dbo.regions + ( + regionName + ) WITH( STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY] + +COMMIT \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/InventoryStore.migrations b/OpenSim/Data/MSSQL/Resources/InventoryStore.migrations index e2a8d5709b..4a3cb27312 100644 --- a/OpenSim/Data/MSSQL/Resources/InventoryStore.migrations +++ b/OpenSim/Data/MSSQL/Resources/InventoryStore.migrations @@ -238,7 +238,10 @@ alter table inventoryitems COMMIT - - - +:VERSION 8 +ALTER TABLE inventoryitems +ADD CONSTRAINT DF_inventoryitems_creatorID +DEFAULT '00000000-0000-0000-0000-000000000000' FOR creatorID + +:GO \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/Presence.migrations b/OpenSim/Data/MSSQL/Resources/Presence.migrations index 35f78e1f52..bcb6328434 100644 --- a/OpenSim/Data/MSSQL/Resources/Presence.migrations +++ b/OpenSim/Data/MSSQL/Resources/Presence.migrations @@ -7,14 +7,7 @@ CREATE TABLE [Presence] ( [RegionID] uniqueidentifier NOT NULL, [SessionID] uniqueidentifier NOT NULL DEFAULT '00000000-0000-0000-0000-000000000000', [SecureSessionID] uniqueidentifier NOT NULL DEFAULT '00000000-0000-0000-0000-000000000000', -[Online] char(5) NOT NULL DEFAULT 'false', -[Login] char(16) NOT NULL DEFAULT '0', -[Logout] char(16) NOT NULL DEFAULT '0', -[Position] char(64) NOT NULL DEFAULT '<0,0,0>', -[LookAt] char(64) NOT NULL DEFAULT '<0,0,0>', -[HomeRegionID] uniqueidentifier NOT NULL, -[HomePosition] CHAR(64) NOT NULL DEFAULT '<0,0,0>', -[HomeLookAt] CHAR(64) NOT NULL DEFAULT '<0,0,0>', + ) ON [PRIMARY] @@ -27,4 +20,12 @@ BEGIN TRANSACTION CREATE UNIQUE INDEX SessionID ON Presence(SessionID); CREATE INDEX UserID ON Presence(UserID); +COMMIT + +:VERSION 2 + +BEGIN TRANSACTION + +ALTER TABLE Presence ADD LastSeen DateTime + COMMIT \ No newline at end of file diff --git a/OpenSim/Data/MSSQL/Resources/RegionStore.migrations b/OpenSim/Data/MSSQL/Resources/RegionStore.migrations index e2e8cbb117..340b63dec3 100644 --- a/OpenSim/Data/MSSQL/Resources/RegionStore.migrations +++ b/OpenSim/Data/MSSQL/Resources/RegionStore.migrations @@ -1,4 +1,3 @@ - :VERSION 1 CREATE TABLE [dbo].[prims]( @@ -926,11 +925,121 @@ ALTER TABLE regionsettings ADD loaded_creation_datetime int NOT NULL default 0 COMMIT :VERSION 24 --- Added post 0.7 BEGIN TRANSACTION -ALTER TABLE prims ADD COLUMN MediaURL varchar(255) -ALTER TABLE primshapes ADD COLUMN Media TEXT +ALTER TABLE prims ADD MediaURL varchar(255) +ALTER TABLE primshapes ADD Media TEXT NULL -COMMIT \ No newline at end of file +COMMIT + +:VERSION 25 + +BEGIN TRANSACTION +CREATE TABLE "regionwindlight" ( + "region_id" varchar(36) NOT NULL DEFAULT '000000-0000-0000-0000-000000000000', + "water_color_r" [float] NOT NULL DEFAULT '4.000000', + "water_color_g" [float] NOT NULL DEFAULT '38.000000', + "water_color_b" [float] NOT NULL DEFAULT '64.000000', + "water_fog_density_exponent" [float] NOT NULL DEFAULT '4.0', + "underwater_fog_modifier" [float] NOT NULL DEFAULT '0.25', + "reflection_wavelet_scale_1" [float] NOT NULL DEFAULT '2.0', + "reflection_wavelet_scale_2" [float] NOT NULL DEFAULT '2.0', + "reflection_wavelet_scale_3" [float] NOT NULL DEFAULT '2.0', + "fresnel_scale" [float] NOT NULL DEFAULT '0.40', + "fresnel_offset" [float] NOT NULL DEFAULT '0.50', + "refract_scale_above" [float] NOT NULL DEFAULT '0.03', + "refract_scale_below" [float] NOT NULL DEFAULT '0.20', + "blur_multiplier" [float] NOT NULL DEFAULT '0.040', + "big_wave_direction_x" [float] NOT NULL DEFAULT '1.05', + "big_wave_direction_y" [float] NOT NULL DEFAULT '-0.42', + "little_wave_direction_x" [float] NOT NULL DEFAULT '1.11', + "little_wave_direction_y" [float] NOT NULL DEFAULT '-1.16', + "normal_map_texture" varchar(36) NOT NULL DEFAULT '822ded49-9a6c-f61c-cb89-6df54f42cdf4', + "horizon_r" [float] NOT NULL DEFAULT '0.25', + "horizon_g" [float] NOT NULL DEFAULT '0.25', + "horizon_b" [float] NOT NULL DEFAULT '0.32', + "horizon_i" [float] NOT NULL DEFAULT '0.32', + "haze_horizon" [float] NOT NULL DEFAULT '0.19', + "blue_density_r" [float] NOT NULL DEFAULT '0.12', + "blue_density_g" [float] NOT NULL DEFAULT '0.22', + "blue_density_b" [float] NOT NULL DEFAULT '0.38', + "blue_density_i" [float] NOT NULL DEFAULT '0.38', + "haze_density" [float] NOT NULL DEFAULT '0.70', + "density_multiplier" [float] NOT NULL DEFAULT '0.18', + "distance_multiplier" [float] NOT NULL DEFAULT '0.8', + "max_altitude" int NOT NULL DEFAULT '1605', + "sun_moon_color_r" [float] NOT NULL DEFAULT '0.24', + "sun_moon_color_g" [float] NOT NULL DEFAULT '0.26', + "sun_moon_color_b" [float] NOT NULL DEFAULT '0.30', + "sun_moon_color_i" [float] NOT NULL DEFAULT '0.30', + "sun_moon_position" [float] NOT NULL DEFAULT '0.317', + "ambient_r" [float] NOT NULL DEFAULT '0.35', + "ambient_g" [float] NOT NULL DEFAULT '0.35', + "ambient_b" [float] NOT NULL DEFAULT '0.35', + "ambient_i" [float] NOT NULL DEFAULT '0.35', + "east_angle" [float] NOT NULL DEFAULT '0.00', + "sun_glow_focus" [float] NOT NULL DEFAULT '0.10', + "sun_glow_size" [float] NOT NULL DEFAULT '1.75', + "scene_gamma" [float] NOT NULL DEFAULT '1.00', + "star_brightness" [float] NOT NULL DEFAULT '0.00', + "cloud_color_r" [float] NOT NULL DEFAULT '0.41', + "cloud_color_g" [float] NOT NULL DEFAULT '0.41', + "cloud_color_b" [float] NOT NULL DEFAULT '0.41', + "cloud_color_i" [float] NOT NULL DEFAULT '0.41', + "cloud_x" [float] NOT NULL DEFAULT '1.00', + "cloud_y" [float] NOT NULL DEFAULT '0.53', + "cloud_density" [float] NOT NULL DEFAULT '1.00', + "cloud_coverage" [float] NOT NULL DEFAULT '0.27', + "cloud_scale" [float] NOT NULL DEFAULT '0.42', + "cloud_detail_x" [float] NOT NULL DEFAULT '1.00', + "cloud_detail_y" [float] NOT NULL DEFAULT '0.53', + "cloud_detail_density" [float] NOT NULL DEFAULT '0.12', + "cloud_scroll_x" [float] NOT NULL DEFAULT '0.20', + "cloud_scroll_x_lock" tinyint NOT NULL DEFAULT '0', + "cloud_scroll_y" [float] NOT NULL DEFAULT '0.01', + "cloud_scroll_y_lock" tinyint NOT NULL DEFAULT '0', + "draw_classic_clouds" tinyint NOT NULL DEFAULT '1', + PRIMARY KEY ("region_id") +) + +COMMIT TRANSACTION + +:VERSION 26 + +BEGIN TRANSACTION + +ALTER TABLE regionsettings ADD map_tile_ID CHAR(36) NOT NULL DEFAULT '00000000-0000-0000-0000-000000000000' + +COMMIT + +:VERSION 27 #--------------------- + +BEGIN TRANSACTION +ALTER TABLE land ADD MediaType VARCHAR(32) NOT NULL DEFAULT 'none/none' +ALTER TABLE land ADD MediaDescription VARCHAR(255) NOT NULL DEFAULT '' +ALTER TABLE land ADD MediaSize VARCHAR(16) NOT NULL DEFAULT '0,0' +ALTER TABLE land ADD MediaLoop bit NOT NULL DEFAULT 0 +ALTER TABLE land ADD ObscureMusic bit NOT NULL DEFAULT 0 +ALTER TABLE land ADD ObscureMedia bit NOT NULL DEFAULT 0 +COMMIT + +:VERSION 28 #--------------------- + +BEGIN TRANSACTION + +ALTER TABLE prims +ADD CONSTRAINT DF_prims_CreatorID +DEFAULT '00000000-0000-0000-0000-000000000000' +FOR CreatorID + +ALTER TABLE prims ALTER COLUMN CreatorID uniqueidentifier NOT NULL + +ALTER TABLE primitems +ADD CONSTRAINT DF_primitems_CreatorID +DEFAULT '00000000-0000-0000-0000-000000000000' +FOR CreatorID + +ALTER TABLE primitems ALTER COLUMN CreatorID uniqueidentifier NOT NULL + +COMMIT diff --git a/OpenSim/Data/MSSQL/Resources/UserAccount.migrations b/OpenSim/Data/MSSQL/Resources/UserAccount.migrations index 8534e235c9..a81704dcd6 100644 --- a/OpenSim/Data/MSSQL/Resources/UserAccount.migrations +++ b/OpenSim/Data/MSSQL/Resources/UserAccount.migrations @@ -19,7 +19,7 @@ CREATE TABLE [UserAccounts] ( :VERSION 2 BEGIN TRANSACTION - +IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[users]') AND type in (N'U')) INSERT INTO UserAccounts (PrincipalID, ScopeID, FirstName, LastName, Email, ServiceURLs, Created) SELECT [UUID] AS PrincipalID, '00000000-0000-0000-0000-000000000000' AS ScopeID, username AS FirstName, lastname AS LastName, From 5c227da4971b4a43246dbef02dfb84ae744daa82 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Thu, 28 Apr 2011 07:36:34 -0700 Subject: [PATCH 39/45] Added MrMonkE to the contributors. Fixed contributors. --- CONTRIBUTORS.txt | 37 +++++++++++++++++++++++-------------- 1 file changed, 23 insertions(+), 14 deletions(-) diff --git a/CONTRIBUTORS.txt b/CONTRIBUTORS.txt index ef00c19b04..246c669f4a 100644 --- a/CONTRIBUTORS.txt +++ b/CONTRIBUTORS.txt @@ -1,34 +1,36 @@ The following people have contributed to OpenSim (Thank you for your effort!) -Add your name in here if you have committed to OpenSim - = Current OpenSim Developers (in very rough order of appearance) = These folks represent the current core team for OpenSim, and are the people that make the day to day of OpenSim happen. +* justincc +* chi11ken (Genkii) +* dahlia +* Melanie Thielker +* Diva (Crista Lopes, University of California, Irvine) +* Dan Lake (Intel) +* Marck +* Mic Bowman (Intel) +* BlueWall (James Hughes) + += Core Developers Following the White Rabbit = +Core developers who have temporarily (we hope) gone chasing the white rabbit. +They are in all similar to the active core developers, except that they haven't +been that active lately, so their voting rights are awaiting their come back. + * MW (Tribal Media AB) * Adam Frisby (DeepThink Pty Ltd) -* MingChen (DeepThink Pty Ltd) * lbsa71 (Tribal Media AB) -* Sean Dague / sdague (IBM) -* Tedd -* justincc * Teravus (w3z) -* Johan Berntsson (3Di) * Ckrinke (Charles Krinke) -* chi11ken (Genkii) -* adjohn (Genkii) * Dr Scofield aka Dirk Husemann (IBM Research - Zurich) -* dahlia * mikem (3Di) -* Melanie Thielker * Homer_Horwitz -* idb (Ian Brown) -* Diva (Crista Lopes, University of California, Irvine) * nlin (3Di) * Arthur Rodrigo S Valadares (IBM) -* BlueWall (James Hughes) +* John Hurliman = Past Open Sim Developers = These folks are alumns of the OpenSim core group, but are now @@ -44,6 +46,12 @@ where we are today. * Dalien * Darok * Alondria +* Sean Dague / sdague (IBM) +* Tedd +* MingChen (DeepThink Pty Ltd) +* adjohn (Genkii) +* idb (Ian Brown) +* Johan Berntsson (3Di) = Additional OpenSim Contributors = @@ -102,6 +110,7 @@ what it is today. * Misterblue (Intel) * Mircea Kitsune * mpallari +* MrMonkE * nornalbion * Omar Vera Ustariz (IBM) * openlifegrid.com From 549dc5aeb9e693f1f575896796d19b03ec3a732f Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Thu, 28 Apr 2011 08:58:04 -0700 Subject: [PATCH 40/45] Eliminated sAgentCircuitData, a data structure that has been obsolete for quite some time. --- OpenSim/Framework/AgentCircuitData.cs | 69 ------------------- OpenSim/Framework/ClientInfo.cs | 2 +- .../ClientStack/LindenUDP/LLClientView.cs | 2 +- 3 files changed, 2 insertions(+), 71 deletions(-) diff --git a/OpenSim/Framework/AgentCircuitData.cs b/OpenSim/Framework/AgentCircuitData.cs index 3dbc215d46..dbd47d392a 100644 --- a/OpenSim/Framework/AgentCircuitData.cs +++ b/OpenSim/Framework/AgentCircuitData.cs @@ -151,27 +151,6 @@ namespace OpenSim.Framework { } - /// - /// Create AgentCircuitData from a Serializable AgentCircuitData - /// - /// - public AgentCircuitData(sAgentCircuitData cAgent) - { - AgentID = new UUID(cAgent.AgentID); - SessionID = new UUID(cAgent.SessionID); - SecureSessionID = new UUID(cAgent.SecureSessionID); - startpos = new Vector3(cAgent.startposx, cAgent.startposy, cAgent.startposz); - firstname = cAgent.firstname; - lastname = cAgent.lastname; - circuitcode = cAgent.circuitcode; - child = cAgent.child; - InventoryFolder = new UUID(cAgent.InventoryFolder); - BaseFolder = new UUID(cAgent.BaseFolder); - CapsPath = cAgent.CapsPath; - ChildrenCapSeeds = cAgent.ChildrenCapSeeds; - Viewer = cAgent.Viewer; - } - /// /// Pack AgentCircuitData into an OSDMap for transmission over LLSD XML or LLSD json /// @@ -369,52 +348,4 @@ namespace OpenSim.Framework } - /// - /// Serializable Agent Circuit Data - /// - [Serializable] - public class sAgentCircuitData - { - public Guid AgentID; - public Guid BaseFolder; - public string CapsPath = String.Empty; - public Dictionary ChildrenCapSeeds; - public bool child; - public uint circuitcode; - public string firstname; - public Guid InventoryFolder; - public string lastname; - public Guid SecureSessionID; - public Guid SessionID; - public float startposx; - public float startposy; - public float startposz; - public string Viewer; - public string Channel; - public string Mac; - public string Id0; - - public sAgentCircuitData() - { - } - - public sAgentCircuitData(AgentCircuitData cAgent) - { - AgentID = cAgent.AgentID.Guid; - SessionID = cAgent.SessionID.Guid; - SecureSessionID = cAgent.SecureSessionID.Guid; - startposx = cAgent.startpos.X; - startposy = cAgent.startpos.Y; - startposz = cAgent.startpos.Z; - firstname = cAgent.firstname; - lastname = cAgent.lastname; - circuitcode = cAgent.circuitcode; - child = cAgent.child; - InventoryFolder = cAgent.InventoryFolder.Guid; - BaseFolder = cAgent.BaseFolder.Guid; - CapsPath = cAgent.CapsPath; - ChildrenCapSeeds = cAgent.ChildrenCapSeeds; - Viewer = cAgent.Viewer; - } - } } diff --git a/OpenSim/Framework/ClientInfo.cs b/OpenSim/Framework/ClientInfo.cs index fbd18b5636..8292c6aa0b 100644 --- a/OpenSim/Framework/ClientInfo.cs +++ b/OpenSim/Framework/ClientInfo.cs @@ -34,7 +34,7 @@ namespace OpenSim.Framework [Serializable] public class ClientInfo { - public sAgentCircuitData agentcircuit; + public AgentCircuitData agentcircuit; public Dictionary needAck; diff --git a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs index 22bad99641..43903ce1bd 100644 --- a/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/LindenUDP/LLClientView.cs @@ -11635,7 +11635,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP info.userEP = m_userEndPoint; info.proxyEP = null; - info.agentcircuit = new sAgentCircuitData(RequestClientInfo()); + info.agentcircuit = RequestClientInfo(); return info; } From 35f190cc920d3c6fd94c48928d29e15a9b143abf Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Thu, 28 Apr 2011 09:06:57 -0700 Subject: [PATCH 41/45] One less [Serializable] -- ClientInfo. --- OpenSim/Framework/ClientInfo.cs | 1 - 1 file changed, 1 deletion(-) diff --git a/OpenSim/Framework/ClientInfo.cs b/OpenSim/Framework/ClientInfo.cs index 8292c6aa0b..62acb70566 100644 --- a/OpenSim/Framework/ClientInfo.cs +++ b/OpenSim/Framework/ClientInfo.cs @@ -31,7 +31,6 @@ using System.Net; namespace OpenSim.Framework { - [Serializable] public class ClientInfo { public AgentCircuitData agentcircuit; From cd7517ccb9d522f5fecd57523285a0b17c147620 Mon Sep 17 00:00:00 2001 From: "Justin Clark-Casey (justincc)" Date: Thu, 28 Apr 2011 22:59:12 +0100 Subject: [PATCH 42/45] Stop CHANGED_INVENTORY firing twice if a notecard is edited in prim. Addresses http://opensimulator.org/mantis/view.php?id=5444 Fix is to stop the asset transaction calling UpdateInventoryItem() since the caller is doing it anyway, which is more correct. This did not effect scripts. --- .../Agent/AssetTransaction/AgentAssetsTransactions.cs | 9 +++++---- OpenSim/Region/Framework/Scenes/Scene.Inventory.cs | 4 ++++ .../Region/Framework/Scenes/SceneObjectPartInventory.cs | 7 +++++-- 3 files changed, 14 insertions(+), 6 deletions(-) diff --git a/OpenSim/Region/CoreModules/Agent/AssetTransaction/AgentAssetsTransactions.cs b/OpenSim/Region/CoreModules/Agent/AssetTransaction/AgentAssetsTransactions.cs index 771038e1af..9d8082bd33 100644 --- a/OpenSim/Region/CoreModules/Agent/AssetTransaction/AgentAssetsTransactions.cs +++ b/OpenSim/Region/CoreModules/Agent/AssetTransaction/AgentAssetsTransactions.cs @@ -41,8 +41,7 @@ namespace OpenSim.Region.CoreModules.Agent.AssetTransaction /// public class AgentAssetTransactions { -// private static readonly ILog m_log = LogManager.GetLogger( -// MethodBase.GetCurrentMethod().DeclaringType); +// private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); // Fields private bool m_dumpAssetsToFile; @@ -149,6 +148,10 @@ namespace OpenSim.Region.CoreModules.Agent.AssetTransaction if (asset != null) { +// m_log.DebugFormat( +// "[AGENT ASSETS TRANSACTIONS]: Updating item {0} in {1} for transaction {2}", +// item.Name, part.Name, transactionID); + asset.FullID = UUID.Random(); asset.Name = item.Name; asset.Description = item.Description; @@ -156,8 +159,6 @@ namespace OpenSim.Region.CoreModules.Agent.AssetTransaction item.AssetID = asset.FullID; m_Scene.AssetService.Store(asset); - - part.Inventory.UpdateInventoryItem(item); } } } diff --git a/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs b/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs index 0b2b01a944..b0f0de62bb 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.Inventory.cs @@ -1430,6 +1430,10 @@ namespace OpenSim.Region.Framework.Scenes } else // Updating existing item with new perms etc { +// m_log.DebugFormat( +// "[PRIM INVENTORY]: Updating item {0} in {1} for UpdateTaskInventory()", +// currentItem.Name, part.Name); + IAgentAssetTransactions agentTransactions = this.RequestModuleInterface(); if (agentTransactions != null) { diff --git a/OpenSim/Region/Framework/Scenes/SceneObjectPartInventory.cs b/OpenSim/Region/Framework/Scenes/SceneObjectPartInventory.cs index 3281eabd33..3b60f8c6da 100644 --- a/OpenSim/Region/Framework/Scenes/SceneObjectPartInventory.cs +++ b/OpenSim/Region/Framework/Scenes/SceneObjectPartInventory.cs @@ -693,8 +693,9 @@ namespace OpenSim.Region.Framework.Scenes { TaskInventoryItem it = GetInventoryItem(item.ItemID); if (it != null) - { +// m_log.DebugFormat("[PRIM INVENTORY]: Updating item {0} in {1}", item.Name, m_part.Name); + item.ParentID = m_part.UUID; item.ParentPartID = m_part.UUID; @@ -711,14 +712,16 @@ namespace OpenSim.Region.Framework.Scenes m_items[item.ItemID] = item; m_inventorySerial++; } - + if (fireScriptEvents) m_part.TriggerScriptChangedEvent(Changed.INVENTORY); + if (considerChanged) { HasInventoryChanged = true; m_part.ParentGroup.HasGroupChanged = true; } + return true; } else From 9892e115ccdcc8567087041917fb5c7694aa8836 Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Thu, 28 Apr 2011 20:19:54 -0700 Subject: [PATCH 43/45] Fatpack message on agent transfers: 1 message only (UpdateAgent) containing the agent and all attachments. Preserves backwards compatibility -- older sims get passed attachments one by one. Meaning that I finally introduced versioning in the simulation service. --- OpenSim/Framework/ChildAgentDataUpdate.cs | 57 +++++++- .../Framework/Tests/MundaneFrameworkTests.cs | 2 +- .../Avatar/Attachments/AttachmentsModule.cs | 13 +- .../EntityTransfer/EntityTransferModule.cs | 61 +++++---- .../EntityTransfer/HGEntityTransferModule.cs | 5 +- .../Simulation/LocalSimulationConnector.cs | 6 +- .../Simulation/RemoteSimulationConnector.cs | 7 +- OpenSim/Region/Framework/Scenes/Scene.cs | 4 +- .../Framework/Scenes/SceneObjectGroup.cs | 2 + .../Region/Framework/Scenes/ScenePresence.cs | 124 ++++-------------- .../Handlers/Simulation/AgentHandlers.cs | 9 +- .../Simulation/SimulationServiceConnector.cs | 21 +-- .../Services/Interfaces/ISimulationService.cs | 2 +- 13 files changed, 157 insertions(+), 156 deletions(-) diff --git a/OpenSim/Framework/ChildAgentDataUpdate.cs b/OpenSim/Framework/ChildAgentDataUpdate.cs index ce0b2fb784..a626b82302 100644 --- a/OpenSim/Framework/ChildAgentDataUpdate.cs +++ b/OpenSim/Framework/ChildAgentDataUpdate.cs @@ -62,7 +62,7 @@ namespace OpenSim.Framework UUID AgentID { get; set; } OSDMap Pack(); - void Unpack(OSDMap map); + void Unpack(OSDMap map, IScene scene); } /// @@ -122,7 +122,7 @@ namespace OpenSim.Framework return args; } - public void Unpack(OSDMap args) + public void Unpack(OSDMap args, IScene scene) { if (args.ContainsKey("region_handle")) UInt64.TryParse(args["region_handle"].AsString(), out RegionHandle); @@ -329,6 +329,10 @@ namespace OpenSim.Framework public string CallbackURI; + // These two must have the same Count + public List AttachmentObjects; + public List AttachmentObjectStates; + public virtual OSDMap Pack() { m_log.InfoFormat("[CHILDAGENTDATAUPDATE] Pack data"); @@ -441,7 +445,30 @@ namespace OpenSim.Framework if ((CallbackURI != null) && (!CallbackURI.Equals(""))) args["callback_uri"] = OSD.FromString(CallbackURI); + // Attachment objects for fatpack messages + if (AttachmentObjects != null) + { + int i = 0; + OSDArray attObjs = new OSDArray(AttachmentObjects.Count); + foreach (ISceneObject so in AttachmentObjects) + { + OSDMap info = new OSDMap(4); + info["sog"] = OSD.FromString(so.ToXml2()); + info["extra"] = OSD.FromString(so.ExtraToXmlString()); + info["modified"] = OSD.FromBoolean(so.HasGroupChanged); + try + { + info["state"] = OSD.FromString(AttachmentObjectStates[i++]); + } + catch (IndexOutOfRangeException e) + { + m_log.WarnFormat("[CHILD AGENT DATA]: scrtips list is shorter than object list."); + } + attObjs.Add(info); + } + args["attach_objects"] = attObjs; + } return args; } @@ -450,7 +477,7 @@ namespace OpenSim.Framework /// Avoiding reflection makes it painful to write, but that's the price! /// /// - public virtual void Unpack(OSDMap args) + public virtual void Unpack(OSDMap args, IScene scene) { m_log.InfoFormat("[CHILDAGENTDATAUPDATE] Unpack data"); @@ -628,6 +655,26 @@ namespace OpenSim.Framework if (args["callback_uri"] != null) CallbackURI = args["callback_uri"].AsString(); + + // Attachment objects + if (args["attach_objects"] != null && args["attach_objects"].Type == OSDType.Array) + { + OSDArray attObjs = (OSDArray)(args["attach_objects"]); + AttachmentObjects = new List(); + AttachmentObjectStates = new List(); + foreach (OSD o in attObjs) + { + if (o.Type == OSDType.Map) + { + OSDMap info = (OSDMap)o; + ISceneObject so = scene.DeserializeObject(info["sog"].AsString()); + so.ExtraFromXmlString(info["extra"].AsString()); + so.HasGroupChanged = info["modified"].AsBoolean(); + AttachmentObjects.Add(so); + AttachmentObjectStates.Add(info["state"].AsString()); + } + } + } } public AgentData() @@ -655,9 +702,9 @@ namespace OpenSim.Framework return base.Pack(); } - public override void Unpack(OSDMap map) + public override void Unpack(OSDMap map, IScene scene) { - base.Unpack(map); + base.Unpack(map, scene); } } } diff --git a/OpenSim/Framework/Tests/MundaneFrameworkTests.cs b/OpenSim/Framework/Tests/MundaneFrameworkTests.cs index e7f8bfc1f0..76de6be267 100644 --- a/OpenSim/Framework/Tests/MundaneFrameworkTests.cs +++ b/OpenSim/Framework/Tests/MundaneFrameworkTests.cs @@ -115,7 +115,7 @@ namespace OpenSim.Framework.Tests position2 = new AgentPosition(); Assert.IsFalse(position2.AgentID == position1.AgentID, "Test Error, position2 should be a blank uninitialized AgentPosition"); - position2.Unpack(position1.Pack()); + position2.Unpack(position1.Pack(), null); Assert.IsTrue(position2.AgentID == position1.AgentID, "Agent ID didn't unpack the same way it packed"); Assert.IsTrue(position2.Position == position1.Position, "Position didn't unpack the same way it packed"); diff --git a/OpenSim/Region/CoreModules/Avatar/Attachments/AttachmentsModule.cs b/OpenSim/Region/CoreModules/Avatar/Attachments/AttachmentsModule.cs index ff262648e5..520d794e74 100644 --- a/OpenSim/Region/CoreModules/Avatar/Attachments/AttachmentsModule.cs +++ b/OpenSim/Region/CoreModules/Avatar/Attachments/AttachmentsModule.cs @@ -562,14 +562,13 @@ namespace OpenSim.Region.CoreModules.Avatar.Attachments /// /// /// - /// + /// /// - protected void AttachToAgent(ScenePresence avatar, SceneObjectGroup so, uint attachmentpoint, Vector3 AttachOffset, bool silent) + protected void AttachToAgent(ScenePresence avatar, SceneObjectGroup so, uint attachmentpoint, Vector3 attachOffset, bool silent) { - // don't attach attachments to child agents - if (avatar.IsChildAgent) return; -// m_log.DebugFormat("[ATTACHMENTS MODULE]: Adding attachment {0} to avatar {1}", Name, avatar.Name); + m_log.DebugFormat("[ATTACHMENTS MODULE]: Adding attachment {0} to avatar {1} in pt {2} pos {3} {4}", Name, avatar.Name, + attachmentpoint, attachOffset, so.RootPart.AttachedPos); so.DetachFromBackup(); @@ -590,8 +589,8 @@ namespace OpenSim.Region.CoreModules.Avatar.Attachments so.RootPart.PhysActor = null; } - so.AbsolutePosition = AttachOffset; - so.RootPart.AttachedPos = AttachOffset; + so.AbsolutePosition = attachOffset; + so.RootPart.AttachedPos = attachOffset; so.RootPart.IsAttachment = true; so.RootPart.SetParentLocalId(avatar.LocalId); diff --git a/OpenSim/Region/CoreModules/Framework/EntityTransfer/EntityTransferModule.cs b/OpenSim/Region/CoreModules/Framework/EntityTransfer/EntityTransferModule.cs index c88be7ddb0..10547857f7 100644 --- a/OpenSim/Region/CoreModules/Framework/EntityTransfer/EntityTransferModule.cs +++ b/OpenSim/Region/CoreModules/Framework/EntityTransfer/EntityTransferModule.cs @@ -285,11 +285,13 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer } string reason; - if (!m_aScene.SimulationService.QueryAccess(finalDestination, sp.ControllingClient.AgentId, Vector3.Zero, out reason)) + string version; + if (!m_aScene.SimulationService.QueryAccess(finalDestination, sp.ControllingClient.AgentId, Vector3.Zero, out version, out reason)) { sp.ControllingClient.SendTeleportFailed("Teleport failed: " + reason); return; } + m_log.DebugFormat("[ENTITY TRANSFER MODULE]: Destination is running version {0}", version); sp.ControllingClient.SendTeleportStart(teleportFlags); @@ -371,20 +373,6 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer capsPath = finalDestination.ServerURI + CapsUtil.GetCapsSeedPath(agentCircuit.CapsPath); } - // Expect avatar crossing is a heavy-duty function at the destination. - // That is where MakeRoot is called, which fetches appearance and inventory. - // Plus triggers OnMakeRoot, which spawns a series of asynchronous updates. - //m_commsProvider.InterRegion.ExpectAvatarCrossing(reg.RegionHandle, avatar.ControllingClient.AgentId, - // position, false); - - //{ - // avatar.ControllingClient.SendTeleportFailed("Problem with destination."); - // // We should close that agent we just created over at destination... - // List lst = new List(); - // lst.Add(reg.RegionHandle); - // SendCloseChildAgentAsync(avatar.UUID, lst); - // return; - //} SetInTransit(sp.UUID); @@ -426,7 +414,7 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer // TeleportFinish makes the client send CompleteMovementIntoRegion (at the destination), which // trigers a whole shebang of things there, including MakeRoot. So let's wait for confirmation - // that the client contacted the destination before we send the attachments and close things here. + // that the client contacted the destination before we close things here. if (!WaitForCallback(sp.UUID)) { m_log.WarnFormat( @@ -437,14 +425,20 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer return; } - // CrossAttachmentsIntoNewRegion is a synchronous call. We shouldn't need to wait after it - CrossAttachmentsIntoNewRegion(finalDestination, sp, true); + // For backwards compatibility + if (version == "Unknown" || version == string.Empty) + { + // CrossAttachmentsIntoNewRegion is a synchronous call. We shouldn't need to wait after it + m_log.DebugFormat("[ENTITY TRANSFER MODULE]: Old simulator, sending attachments one by one..."); + CrossAttachmentsIntoNewRegion(finalDestination, sp, true); + } + + // May need to logout or other cleanup + AgentHasMovedAway(sp, logout); // Well, this is it. The agent is over there. KillEntity(sp.Scene, sp.LocalId); - // May need to logout or other cleanup - AgentHasMovedAway(sp.ControllingClient.SessionId, logout); // Now let's make it officially a child agent sp.MakeChildAgent(); @@ -513,8 +507,13 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer } - protected virtual void AgentHasMovedAway(UUID sessionID, bool logout) + protected virtual void AgentHasMovedAway(ScenePresence sp, bool logout) { + foreach (SceneObjectGroup sop in sp.Attachments) + { + sop.Scene.DeleteSceneObject(sop, true); + } + sp.Attachments.Clear(); } protected void KillEntity(Scene scene, uint localID) @@ -784,7 +783,8 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer GridRegion neighbourRegion = scene.GridService.GetRegionByPosition(scene.RegionInfo.ScopeID, (int)x, (int)y); string reason; - if (!scene.SimulationService.QueryAccess(neighbourRegion, agent.ControllingClient.AgentId, newpos, out reason)) + string version; + if (!scene.SimulationService.QueryAccess(neighbourRegion, agent.ControllingClient.AgentId, newpos, out version, out reason)) { agent.ControllingClient.SendAlertMessage("Cannot region cross into banned parcel"); if (r == null) @@ -804,7 +804,7 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer agent.InTransit(); CrossAgentToNewRegionDelegate d = CrossAgentToNewRegionAsync; - d.BeginInvoke(agent, newpos, neighbourx, neighboury, neighbourRegion, isFlying, CrossAgentToNewRegionCompleted, d); + d.BeginInvoke(agent, newpos, neighbourx, neighboury, neighbourRegion, isFlying, version, CrossAgentToNewRegionCompleted, d); return true; } @@ -861,17 +861,17 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer icon.EndInvoke(iar); } - public delegate ScenePresence CrossAgentToNewRegionDelegate(ScenePresence agent, Vector3 pos, uint neighbourx, uint neighboury, GridRegion neighbourRegion, bool isFlying); + public delegate ScenePresence CrossAgentToNewRegionDelegate(ScenePresence agent, Vector3 pos, uint neighbourx, uint neighboury, GridRegion neighbourRegion, bool isFlying, string version); /// /// This Closes child agents on neighbouring regions /// Calls an asynchronous method to do so.. so it doesn't lag the sim. /// - protected ScenePresence CrossAgentToNewRegionAsync(ScenePresence agent, Vector3 pos, uint neighbourx, uint neighboury, GridRegion neighbourRegion, bool isFlying) + protected ScenePresence CrossAgentToNewRegionAsync(ScenePresence agent, Vector3 pos, uint neighbourx, uint neighboury, GridRegion neighbourRegion, bool isFlying, string version) { ulong neighbourHandle = Utils.UIntsToLong((uint)(neighbourx * Constants.RegionSize), (uint)(neighboury * Constants.RegionSize)); - m_log.DebugFormat("[ENTITY TRANSFER MODULE]: Crossing agent {0} {1} to {2}-{3}", agent.Firstname, agent.Lastname, neighbourx, neighboury); + m_log.DebugFormat("[ENTITY TRANSFER MODULE]: Crossing agent {0} {1} to {2}-{3} running version {4}", agent.Firstname, agent.Lastname, neighbourx, neighboury, version); Scene m_scene = agent.Scene; @@ -945,7 +945,14 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer agent.SendOtherAgentsAvatarDataToMe(); agent.SendOtherAgentsAppearanceToMe(); - CrossAttachmentsIntoNewRegion(neighbourRegion, agent, true); + // Backwards compatibility + if (version == "Unknown" || version == string.Empty) + { + m_log.DebugFormat("[ENTITY TRANSFER MODULE]: Old neighbor, passing attachments one by one..."); + CrossAttachmentsIntoNewRegion(neighbourRegion, agent, true); + } + + AgentHasMovedAway(agent, false); // the user may change their profile information in other region, // so the userinfo in UserProfileCache is not reliable any more, delete it diff --git a/OpenSim/Region/CoreModules/Framework/EntityTransfer/HGEntityTransferModule.cs b/OpenSim/Region/CoreModules/Framework/EntityTransfer/HGEntityTransferModule.cs index 79e76b4cbf..5c53f787bf 100644 --- a/OpenSim/Region/CoreModules/Framework/EntityTransfer/HGEntityTransferModule.cs +++ b/OpenSim/Region/CoreModules/Framework/EntityTransfer/HGEntityTransferModule.cs @@ -142,11 +142,12 @@ namespace OpenSim.Region.CoreModules.Framework.EntityTransfer return false; } - protected override void AgentHasMovedAway(UUID sessionID, bool logout) + protected override void AgentHasMovedAway(ScenePresence sp, bool logout) { + base.AgentHasMovedAway(sp, logout); if (logout) // Log them out of this grid - m_aScene.PresenceService.LogoutAgent(sessionID); + m_aScene.PresenceService.LogoutAgent(sp.ControllingClient.SessionId); } protected override bool CreateAgent(ScenePresence sp, GridRegion reg, GridRegion finalDestination, AgentCircuitData agentCircuit, uint teleportFlags, out string reason, out bool logout) diff --git a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/LocalSimulationConnector.cs b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/LocalSimulationConnector.cs index a298b65820..2cf02b5544 100644 --- a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/LocalSimulationConnector.cs +++ b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/LocalSimulationConnector.cs @@ -41,6 +41,9 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Simulation public class LocalSimulationConnectorModule : ISharedRegionModule, ISimulationService { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + // Version of this service + private const string m_Version = "SIMULATION/0.1"; + private List m_sceneList = new List(); private IEntityTransferModule m_AgentTransferModule; @@ -257,9 +260,10 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Simulation return false; } - public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string reason) + public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string version, out string reason) { reason = "Communications failure"; + version = m_Version; if (destination == null) return false; diff --git a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/RemoteSimulationConnector.cs b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/RemoteSimulationConnector.cs index 67f4d6030c..7858f2afd3 100644 --- a/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/RemoteSimulationConnector.cs +++ b/OpenSim/Region/CoreModules/ServiceConnectorsOut/Simulation/RemoteSimulationConnector.cs @@ -229,19 +229,20 @@ namespace OpenSim.Region.CoreModules.ServiceConnectorsOut.Simulation } - public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string reason) + public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string version, out string reason) { reason = "Communications failure"; + version = "Unknown"; if (destination == null) return false; // Try local first - if (m_localBackend.QueryAccess(destination, id, position, out reason)) + if (m_localBackend.QueryAccess(destination, id, position, out version, out reason)) return true; // else do the remote thing if (!m_localBackend.IsLocalRegion(destination.RegionHandle)) - return m_remoteConnector.QueryAccess(destination, id, position, out reason); + return m_remoteConnector.QueryAccess(destination, id, position, out version, out reason); return false; diff --git a/OpenSim/Region/Framework/Scenes/Scene.cs b/OpenSim/Region/Framework/Scenes/Scene.cs index 01de82451f..696c6ee453 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.cs @@ -2316,7 +2316,9 @@ namespace OpenSim.Region.Framework.Scenes /// public bool IncomingCreateObject(ISceneObject sog) { - //m_log.Debug(" >>> IncomingCreateObject(sog) <<< " + ((SceneObjectGroup)sog).AbsolutePosition + " deleted? " + ((SceneObjectGroup)sog).IsDeleted); + //m_log.DebugFormat(" >>> IncomingCreateObject(sog) <<< {0} deleted? {1} isAttach? {2}", ((SceneObjectGroup)sog).AbsolutePosition, + // ((SceneObjectGroup)sog).IsDeleted, ((SceneObjectGroup)sog).RootPart.IsAttachment); + SceneObjectGroup newObject; try { diff --git a/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs b/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs index 19a95062b4..bccbe683f7 100644 --- a/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs +++ b/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs @@ -3361,6 +3361,8 @@ namespace OpenSim.Region.Framework.Scenes { SceneObjectGroup sog = Copy(false); sog.m_isDeleted = false; + sog.RootPart.IsAttachment = false; + sog.RootPart.GroupPosition = sog.RootPart.AttachedPos; return sog; } diff --git a/OpenSim/Region/Framework/Scenes/ScenePresence.cs b/OpenSim/Region/Framework/Scenes/ScenePresence.cs index e4413a95c9..507fc50f99 100644 --- a/OpenSim/Region/Framework/Scenes/ScenePresence.cs +++ b/OpenSim/Region/Framework/Scenes/ScenePresence.cs @@ -3078,54 +3078,6 @@ namespace OpenSim.Region.Framework.Scenes cAgent.Appearance = new AvatarAppearance(m_appearance); -/* - try - { - // We might not pass the Wearables in all cases... - // They're only needed so that persistent changes to the appearance - // are preserved in the new region where the user is moving to. - // But in Hypergrid we might not let this happen. - int i = 0; - UUID[] wears = new UUID[m_appearance.Wearables.Length * 2]; - foreach (AvatarWearable aw in m_appearance.Wearables) - { - if (aw != null) - { - wears[i++] = aw.ItemID; - wears[i++] = aw.AssetID; - } - else - { - wears[i++] = UUID.Zero; - wears[i++] = UUID.Zero; - } - } - cAgent.Wearables = wears; - - cAgent.VisualParams = m_appearance.VisualParams; - - if (m_appearance.Texture != null) - cAgent.AgentTextures = m_appearance.Texture.GetBytes(); - } - catch (Exception e) - { - m_log.Warn("[SCENE PRESENCE]: exception in CopyTo " + e.Message); - } - - //Attachments - List attPoints = m_appearance.GetAttachedPoints(); - if (attPoints != null) - { - //m_log.DebugFormat("[SCENE PRESENCE]: attachments {0}", attPoints.Count); - int i = 0; - AvatarAttachment[] attachs = new AvatarAttachment[attPoints.Count]; - foreach (int point in attPoints) - { - attachs[i++] = new AvatarAttachment(point, m_appearance.GetAttachedItem(point), m_appearance.GetAttachedAsset(point)); - } - cAgent.Attachments = attachs; - } -*/ lock (scriptedcontrols) { ControllerData[] controls = new ControllerData[scriptedcontrols.Count]; @@ -3145,9 +3097,21 @@ namespace OpenSim.Region.Framework.Scenes } catch { } - // cAgent.GroupID = ?? - // Groups??? - + // Attachment objects + if (m_attachments != null && m_attachments.Count > 0) + { + cAgent.AttachmentObjects = new List(); + cAgent.AttachmentObjectStates = new List(); + IScriptModule se = m_scene.RequestModuleInterface(); + foreach (SceneObjectGroup sog in m_attachments) + { + // We need to make a copy and pass that copy + // because of transfers withn the same sim + ISceneObject clone = sog.CloneForNewScene(); + cAgent.AttachmentObjects.Add(clone); + cAgent.AttachmentObjectStates.Add(sog.GetStateSnapshot()); + } + } } public void CopyFrom(AgentData cAgent) @@ -3188,50 +3152,6 @@ namespace OpenSim.Region.Framework.Scenes AddToPhysicalScene(isFlying); } -/* - uint i = 0; - try - { - if (cAgent.Wearables == null) - cAgent.Wearables = new UUID[0]; - AvatarWearable[] wears = new AvatarWearable[cAgent.Wearables.Length / 2]; - for (uint n = 0; n < cAgent.Wearables.Length; n += 2) - { - UUID itemId = cAgent.Wearables[n]; - UUID assetId = cAgent.Wearables[n + 1]; - wears[i++] = new AvatarWearable(itemId, assetId); - } - // m_appearance.Wearables = wears; - Primitive.TextureEntry textures = null; - if (cAgent.AgentTextures != null && cAgent.AgentTextures.Length > 1) - textures = new Primitive.TextureEntry(cAgent.AgentTextures, 0, cAgent.AgentTextures.Length); - - byte[] visuals = null; - - if ((cAgent.VisualParams != null) && (cAgent.VisualParams.Length < AvatarAppearance.VISUALPARAM_COUNT)) - visuals = (byte[])cAgent.VisualParams.Clone(); - - m_appearance = new AvatarAppearance(cAgent.AgentID,wears,textures,visuals); - } - catch (Exception e) - { - m_log.Warn("[SCENE PRESENCE]: exception in CopyFrom " + e.Message); - } - - // Attachments - try - { - if (cAgent.Attachments != null) - { - m_appearance.ClearAttachments(); - foreach (AvatarAttachment att in cAgent.Attachments) - { - m_appearance.SetAttachment(att.AttachPoint, att.ItemID, att.AssetID); - } - } - } - catch { } -*/ try { lock (scriptedcontrols) @@ -3261,8 +3181,18 @@ namespace OpenSim.Region.Framework.Scenes } catch { } - //cAgent.GroupID = ?? - //Groups??? + if (cAgent.AttachmentObjects != null && cAgent.AttachmentObjects.Count > 0) + { + m_attachments = new List(); + int i = 0; + foreach (ISceneObject so in cAgent.AttachmentObjects) + { + ((SceneObjectGroup)so).LocalId = 0; + ((SceneObjectGroup)so).RootPart.UpdateFlag = 0; + so.SetState(cAgent.AttachmentObjectStates[i++], m_scene); + m_scene.IncomingCreateObject(so); + } + } } public bool CopyAgent(out IAgentData agent) diff --git a/OpenSim/Server/Handlers/Simulation/AgentHandlers.cs b/OpenSim/Server/Handlers/Simulation/AgentHandlers.cs index 372a59c038..8b6fb4f5c1 100644 --- a/OpenSim/Server/Handlers/Simulation/AgentHandlers.cs +++ b/OpenSim/Server/Handlers/Simulation/AgentHandlers.cs @@ -50,6 +50,7 @@ namespace OpenSim.Server.Handlers.Simulation public class AgentHandler { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + private ISimulationService m_SimulationService; protected bool m_Proxy = false; @@ -275,7 +276,7 @@ namespace OpenSim.Server.Handlers.Simulation AgentData agent = new AgentData(); try { - agent.Unpack(args); + agent.Unpack(args, m_SimulationService.GetScene(destination.RegionHandle)); } catch (Exception ex) { @@ -295,7 +296,7 @@ namespace OpenSim.Server.Handlers.Simulation AgentPosition agent = new AgentPosition(); try { - agent.Unpack(args); + agent.Unpack(args, m_SimulationService.GetScene(destination.RegionHandle)); } catch (Exception ex) { @@ -342,7 +343,8 @@ namespace OpenSim.Server.Handlers.Simulation destination.RegionID = regionID; string reason; - bool result = m_SimulationService.QueryAccess(destination, id, position, out reason); + string version; + bool result = m_SimulationService.QueryAccess(destination, id, position, out version, out reason); responsedata["int_response_code"] = HttpStatusCode.OK; @@ -350,6 +352,7 @@ namespace OpenSim.Server.Handlers.Simulation resp["success"] = OSD.FromBoolean(result); resp["reason"] = OSD.FromString(reason); + resp["version"] = OSD.FromString(version); responsedata["str_response_string"] = OSDParser.SerializeJsonString(resp); } diff --git a/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs b/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs index 93b3ae62a9..415b4f0d37 100644 --- a/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs +++ b/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs @@ -241,7 +241,7 @@ namespace OpenSim.Services.Connectors.Simulation if (args != null) { agent = new CompleteAgentData(); - agent.Unpack(args); + agent.Unpack(args, null); return true; } } @@ -256,9 +256,10 @@ namespace OpenSim.Services.Connectors.Simulation /// /// - public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string reason) + public bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string version, out string reason) { reason = "Failed to contact destination"; + version = "Unknown"; // m_log.DebugFormat("[REMOTE SIMULATION CONNECTOR]: QueryAccess start, position={0}", position); @@ -274,23 +275,27 @@ namespace OpenSim.Services.Connectors.Simulation try { OSDMap result = WebUtil.ServiceOSDRequest(uri, request, "QUERYACCESS", 10000); - bool success = result["success"].AsBoolean(); - reason = result["reason"].AsString(); + OSDMap data = (OSDMap)result["_Result"]; - //m_log.DebugFormat("[REMOTE SIMULATION CONNECTOR]: QueryAccess to {0} returned {1}", uri, success); + bool success = result["success"].AsBoolean(); + reason = data["reason"].AsString(); + if (data["version"] != null) + version = data["version"].AsString(); + + m_log.DebugFormat("[REMOTE SIMULATION CONNECTOR]: QueryAccess to {0} returned {1} version {2} ({3})", uri, success, version, data["version"].AsString()); if (!success) { - if (result.ContainsKey("Message")) + if (data.ContainsKey("Message")) { - string message = result["Message"].AsString(); + string message = data["Message"].AsString(); if (message == "Service request failed: [MethodNotAllowed] MethodNotAllowed") // Old style region { m_log.Info("[REMOTE SIMULATION CONNECTOR]: The above web util error was caused by a TP to a sim that doesn't support QUERYACCESS and can be ignored"); return true; } - reason = result["Message"]; + reason = data["Message"]; } else { diff --git a/OpenSim/Services/Interfaces/ISimulationService.cs b/OpenSim/Services/Interfaces/ISimulationService.cs index 55c9cc5717..5f9ce6d8b8 100644 --- a/OpenSim/Services/Interfaces/ISimulationService.cs +++ b/OpenSim/Services/Interfaces/ISimulationService.cs @@ -67,7 +67,7 @@ namespace OpenSim.Services.Interfaces bool RetrieveAgent(GridRegion destination, UUID id, out IAgentData agent); - bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string reason); + bool QueryAccess(GridRegion destination, UUID id, Vector3 position, out string version, out string reason); /// /// Message from receiving region to departing region, telling it got contacted by the client. From e462b926acc2810303d4ce5fc00f494f4765a7ad Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Fri, 29 Apr 2011 08:01:44 -0700 Subject: [PATCH 44/45] Minor improvement in version checking (Simulation service) --- .../Connectors/Simulation/SimulationServiceConnector.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs b/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs index 415b4f0d37..7545db8200 100644 --- a/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs +++ b/OpenSim/Services/Connectors/Simulation/SimulationServiceConnector.cs @@ -279,7 +279,7 @@ namespace OpenSim.Services.Connectors.Simulation bool success = result["success"].AsBoolean(); reason = data["reason"].AsString(); - if (data["version"] != null) + if (data["version"] != null && data["version"].AsString() != string.Empty) version = data["version"].AsString(); m_log.DebugFormat("[REMOTE SIMULATION CONNECTOR]: QueryAccess to {0} returned {1} version {2} ({3})", uri, success, version, data["version"].AsString()); From 73caa4e94abc920269c2eba8423be49192ef184b Mon Sep 17 00:00:00 2001 From: Diva Canto Date: Fri, 29 Apr 2011 08:42:51 -0700 Subject: [PATCH 45/45] Minor correction to yesterday's changes. Make normal prim crossing (no attach) work well again. --- OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs | 2 -- OpenSim/Region/Framework/Scenes/ScenePresence.cs | 3 +++ 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs b/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs index bccbe683f7..19a95062b4 100644 --- a/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs +++ b/OpenSim/Region/Framework/Scenes/SceneObjectGroup.cs @@ -3361,8 +3361,6 @@ namespace OpenSim.Region.Framework.Scenes { SceneObjectGroup sog = Copy(false); sog.m_isDeleted = false; - sog.RootPart.IsAttachment = false; - sog.RootPart.GroupPosition = sog.RootPart.AttachedPos; return sog; } diff --git a/OpenSim/Region/Framework/Scenes/ScenePresence.cs b/OpenSim/Region/Framework/Scenes/ScenePresence.cs index 507fc50f99..fe4a7d1fa7 100644 --- a/OpenSim/Region/Framework/Scenes/ScenePresence.cs +++ b/OpenSim/Region/Framework/Scenes/ScenePresence.cs @@ -3108,6 +3108,9 @@ namespace OpenSim.Region.Framework.Scenes // We need to make a copy and pass that copy // because of transfers withn the same sim ISceneObject clone = sog.CloneForNewScene(); + // Attachment module assumes that GroupPosition holds the offsets...! + ((SceneObjectGroup)clone).RootPart.GroupPosition = sog.RootPart.AttachedPos; + ((SceneObjectGroup)clone).RootPart.IsAttachment = false; cAgent.AttachmentObjects.Add(clone); cAgent.AttachmentObjectStates.Add(sog.GetStateSnapshot()); }