diff options
Diffstat (limited to 'OpenSim/Region/ClientStack')
5 files changed, 429 insertions, 316 deletions
diff --git a/OpenSim/Region/ClientStack/Linden/UDP/J2KImage.cs b/OpenSim/Region/ClientStack/Linden/UDP/J2KImage.cs index e9e2dca..9dd6663 100644 --- a/OpenSim/Region/ClientStack/Linden/UDP/J2KImage.cs +++ b/OpenSim/Region/ClientStack/Linden/UDP/J2KImage.cs | |||
@@ -202,6 +202,9 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
202 | m_stopPacket = TexturePacketCount(); | 202 | m_stopPacket = TexturePacketCount(); |
203 | } | 203 | } |
204 | 204 | ||
205 | //Give them at least two packets, to play nice with some broken viewers (SL also behaves this way) | ||
206 | if (m_stopPacket == 1 && Layers[0].End > FIRST_PACKET_SIZE) m_stopPacket++; | ||
207 | |||
205 | m_currentPacket = StartPacket; | 208 | m_currentPacket = StartPacket; |
206 | } | 209 | } |
207 | } | 210 | } |
diff --git a/OpenSim/Region/ClientStack/Linden/UDP/LLClientView.cs b/OpenSim/Region/ClientStack/Linden/UDP/LLClientView.cs index 5a2c45c..05e6d27 100644 --- a/OpenSim/Region/ClientStack/Linden/UDP/LLClientView.cs +++ b/OpenSim/Region/ClientStack/Linden/UDP/LLClientView.cs | |||
@@ -157,6 +157,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
157 | public event RequestTaskInventory OnRequestTaskInventory; | 157 | public event RequestTaskInventory OnRequestTaskInventory; |
158 | public event UpdateInventoryItem OnUpdateInventoryItem; | 158 | public event UpdateInventoryItem OnUpdateInventoryItem; |
159 | public event CopyInventoryItem OnCopyInventoryItem; | 159 | public event CopyInventoryItem OnCopyInventoryItem; |
160 | public event MoveItemsAndLeaveCopy OnMoveItemsAndLeaveCopy; | ||
160 | public event MoveInventoryItem OnMoveInventoryItem; | 161 | public event MoveInventoryItem OnMoveInventoryItem; |
161 | public event RemoveInventoryItem OnRemoveInventoryItem; | 162 | public event RemoveInventoryItem OnRemoveInventoryItem; |
162 | public event RemoveInventoryFolder OnRemoveInventoryFolder; | 163 | public event RemoveInventoryFolder OnRemoveInventoryFolder; |
@@ -254,7 +255,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
254 | public event ClassifiedInfoRequest OnClassifiedInfoRequest; | 255 | public event ClassifiedInfoRequest OnClassifiedInfoRequest; |
255 | public event ClassifiedInfoUpdate OnClassifiedInfoUpdate; | 256 | public event ClassifiedInfoUpdate OnClassifiedInfoUpdate; |
256 | public event ClassifiedDelete OnClassifiedDelete; | 257 | public event ClassifiedDelete OnClassifiedDelete; |
257 | public event ClassifiedDelete OnClassifiedGodDelete; | 258 | public event ClassifiedGodDelete OnClassifiedGodDelete; |
258 | public event EventNotificationAddRequest OnEventNotificationAddRequest; | 259 | public event EventNotificationAddRequest OnEventNotificationAddRequest; |
259 | public event EventNotificationRemoveRequest OnEventNotificationRemoveRequest; | 260 | public event EventNotificationRemoveRequest OnEventNotificationRemoveRequest; |
260 | public event EventGodDelete OnEventGodDelete; | 261 | public event EventGodDelete OnEventGodDelete; |
@@ -330,7 +331,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
330 | /// thread servicing the m_primFullUpdates queue after a kill. If this happens the object persists as an | 331 | /// thread servicing the m_primFullUpdates queue after a kill. If this happens the object persists as an |
331 | /// ownerless phantom. | 332 | /// ownerless phantom. |
332 | /// | 333 | /// |
333 | /// All manipulation of this set has to occur under a lock | 334 | /// All manipulation of this set has to occur under an m_entityUpdates.SyncRoot lock |
334 | /// | 335 | /// |
335 | /// </value> | 336 | /// </value> |
336 | protected HashSet<uint> m_killRecord; | 337 | protected HashSet<uint> m_killRecord; |
@@ -338,6 +339,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
338 | // protected HashSet<uint> m_attachmentsSent; | 339 | // protected HashSet<uint> m_attachmentsSent; |
339 | 340 | ||
340 | private int m_moneyBalance; | 341 | private int m_moneyBalance; |
342 | private bool m_deliverPackets = true; | ||
341 | private int m_animationSequenceNumber = 1; | 343 | private int m_animationSequenceNumber = 1; |
342 | private bool m_SendLogoutPacketWhenClosing = true; | 344 | private bool m_SendLogoutPacketWhenClosing = true; |
343 | private AgentUpdateArgs lastarg; | 345 | private AgentUpdateArgs lastarg; |
@@ -378,6 +380,14 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
378 | get { return m_startpos; } | 380 | get { return m_startpos; } |
379 | set { m_startpos = value; } | 381 | set { m_startpos = value; } |
380 | } | 382 | } |
383 | public bool DeliverPackets | ||
384 | { | ||
385 | get { return m_deliverPackets; } | ||
386 | set { | ||
387 | m_deliverPackets = value; | ||
388 | m_udpClient.m_deliverPackets = value; | ||
389 | } | ||
390 | } | ||
381 | public UUID AgentId { get { return m_agentId; } } | 391 | public UUID AgentId { get { return m_agentId; } } |
382 | public UUID ActiveGroupId { get { return m_activeGroupID; } } | 392 | public UUID ActiveGroupId { get { return m_activeGroupID; } } |
383 | public string ActiveGroupName { get { return m_activeGroupName; } } | 393 | public string ActiveGroupName { get { return m_activeGroupName; } } |
@@ -479,18 +489,30 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
479 | 489 | ||
480 | #region Client Methods | 490 | #region Client Methods |
481 | 491 | ||
492 | |||
482 | /// <summary> | 493 | /// <summary> |
483 | /// Shut down the client view | 494 | /// Shut down the client view |
484 | /// </summary> | 495 | /// </summary> |
485 | public void Close() | 496 | public void Close() |
486 | { | 497 | { |
498 | Close(true); | ||
499 | } | ||
500 | |||
501 | /// <summary> | ||
502 | /// Shut down the client view | ||
503 | /// </summary> | ||
504 | public void Close(bool sendStop) | ||
505 | { | ||
487 | m_log.DebugFormat( | 506 | m_log.DebugFormat( |
488 | "[CLIENT]: Close has been called for {0} attached to scene {1}", | 507 | "[CLIENT]: Close has been called for {0} attached to scene {1}", |
489 | Name, m_scene.RegionInfo.RegionName); | 508 | Name, m_scene.RegionInfo.RegionName); |
490 | 509 | ||
491 | // Send the STOP packet | 510 | if (sendStop) |
492 | DisableSimulatorPacket disable = (DisableSimulatorPacket)PacketPool.Instance.GetPacket(PacketType.DisableSimulator); | 511 | { |
493 | OutPacket(disable, ThrottleOutPacketType.Unknown); | 512 | // Send the STOP packet |
513 | DisableSimulatorPacket disable = (DisableSimulatorPacket)PacketPool.Instance.GetPacket(PacketType.DisableSimulator); | ||
514 | OutPacket(disable, ThrottleOutPacketType.Unknown); | ||
515 | } | ||
494 | 516 | ||
495 | IsActive = false; | 517 | IsActive = false; |
496 | 518 | ||
@@ -771,7 +793,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
771 | reply.ChatData.OwnerID = fromAgentID; | 793 | reply.ChatData.OwnerID = fromAgentID; |
772 | reply.ChatData.SourceID = fromAgentID; | 794 | reply.ChatData.SourceID = fromAgentID; |
773 | 795 | ||
774 | OutPacket(reply, ThrottleOutPacketType.Task); | 796 | OutPacket(reply, ThrottleOutPacketType.Unknown); |
775 | } | 797 | } |
776 | 798 | ||
777 | /// <summary> | 799 | /// <summary> |
@@ -1057,6 +1079,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1057 | public virtual void SendLayerData(float[] map) | 1079 | public virtual void SendLayerData(float[] map) |
1058 | { | 1080 | { |
1059 | Util.FireAndForget(DoSendLayerData, map); | 1081 | Util.FireAndForget(DoSendLayerData, map); |
1082 | |||
1083 | // Send it sync, and async. It's not that much data | ||
1084 | // and it improves user experience just so much! | ||
1085 | DoSendLayerData(map); | ||
1060 | } | 1086 | } |
1061 | 1087 | ||
1062 | /// <summary> | 1088 | /// <summary> |
@@ -1069,16 +1095,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1069 | 1095 | ||
1070 | try | 1096 | try |
1071 | { | 1097 | { |
1072 | //for (int y = 0; y < 16; y++) | 1098 | for (int y = 0; y < 16; y++) |
1073 | //{ | 1099 | { |
1074 | // for (int x = 0; x < 16; x++) | 1100 | for (int x = 0; x < 16; x+=4) |
1075 | // { | 1101 | { |
1076 | // SendLayerData(x, y, map); | 1102 | SendLayerPacket(x, y, map); |
1077 | // } | 1103 | } |
1078 | //} | 1104 | } |
1079 | |||
1080 | // Send LayerData in a spiral pattern. Fun! | ||
1081 | SendLayerTopRight(map, 0, 0, 15, 15); | ||
1082 | } | 1105 | } |
1083 | catch (Exception e) | 1106 | catch (Exception e) |
1084 | { | 1107 | { |
@@ -1086,51 +1109,35 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1086 | } | 1109 | } |
1087 | } | 1110 | } |
1088 | 1111 | ||
1089 | private void SendLayerTopRight(float[] map, int x1, int y1, int x2, int y2) | ||
1090 | { | ||
1091 | // Row | ||
1092 | for (int i = x1; i <= x2; i++) | ||
1093 | SendLayerData(i, y1, map); | ||
1094 | |||
1095 | // Column | ||
1096 | for (int j = y1 + 1; j <= y2; j++) | ||
1097 | SendLayerData(x2, j, map); | ||
1098 | |||
1099 | if (x2 - x1 > 0) | ||
1100 | SendLayerBottomLeft(map, x1, y1 + 1, x2 - 1, y2); | ||
1101 | } | ||
1102 | |||
1103 | void SendLayerBottomLeft(float[] map, int x1, int y1, int x2, int y2) | ||
1104 | { | ||
1105 | // Row in reverse | ||
1106 | for (int i = x2; i >= x1; i--) | ||
1107 | SendLayerData(i, y2, map); | ||
1108 | |||
1109 | // Column in reverse | ||
1110 | for (int j = y2 - 1; j >= y1; j--) | ||
1111 | SendLayerData(x1, j, map); | ||
1112 | |||
1113 | if (x2 - x1 > 0) | ||
1114 | SendLayerTopRight(map, x1 + 1, y1, x2, y2 - 1); | ||
1115 | } | ||
1116 | |||
1117 | /// <summary> | 1112 | /// <summary> |
1118 | /// Sends a set of four patches (x, x+1, ..., x+3) to the client | 1113 | /// Sends a set of four patches (x, x+1, ..., x+3) to the client |
1119 | /// </summary> | 1114 | /// </summary> |
1120 | /// <param name="map">heightmap</param> | 1115 | /// <param name="map">heightmap</param> |
1121 | /// <param name="px">X coordinate for patches 0..12</param> | 1116 | /// <param name="px">X coordinate for patches 0..12</param> |
1122 | /// <param name="py">Y coordinate for patches 0..15</param> | 1117 | /// <param name="py">Y coordinate for patches 0..15</param> |
1123 | // private void SendLayerPacket(float[] map, int y, int x) | 1118 | private void SendLayerPacket(int x, int y, float[] map) |
1124 | // { | 1119 | { |
1125 | // int[] patches = new int[4]; | 1120 | int[] patches = new int[4]; |
1126 | // patches[0] = x + 0 + y * 16; | 1121 | patches[0] = x + 0 + y * 16; |
1127 | // patches[1] = x + 1 + y * 16; | 1122 | patches[1] = x + 1 + y * 16; |
1128 | // patches[2] = x + 2 + y * 16; | 1123 | patches[2] = x + 2 + y * 16; |
1129 | // patches[3] = x + 3 + y * 16; | 1124 | patches[3] = x + 3 + y * 16; |
1130 | 1125 | ||
1131 | // Packet layerpack = LLClientView.TerrainManager.CreateLandPacket(map, patches); | 1126 | float[] heightmap = (map.Length == 65536) ? |
1132 | // OutPacket(layerpack, ThrottleOutPacketType.Land); | 1127 | map : |
1133 | // } | 1128 | LLHeightFieldMoronize(map); |
1129 | |||
1130 | try | ||
1131 | { | ||
1132 | Packet layerpack = TerrainCompressor.CreateLandPacket(heightmap, patches); | ||
1133 | OutPacket(layerpack, ThrottleOutPacketType.Land); | ||
1134 | } | ||
1135 | catch | ||
1136 | { | ||
1137 | for (int px = x ; px < x + 4 ; px++) | ||
1138 | SendLayerData(px, y, map); | ||
1139 | } | ||
1140 | } | ||
1134 | 1141 | ||
1135 | /// <summary> | 1142 | /// <summary> |
1136 | /// Sends a specified patch to a client | 1143 | /// Sends a specified patch to a client |
@@ -1150,7 +1157,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1150 | LayerDataPacket layerpack = TerrainCompressor.CreateLandPacket(heightmap, patches); | 1157 | LayerDataPacket layerpack = TerrainCompressor.CreateLandPacket(heightmap, patches); |
1151 | layerpack.Header.Reliable = true; | 1158 | layerpack.Header.Reliable = true; |
1152 | 1159 | ||
1153 | OutPacket(layerpack, ThrottleOutPacketType.Land); | 1160 | OutPacket(layerpack, ThrottleOutPacketType.Task); |
1154 | } | 1161 | } |
1155 | catch (Exception e) | 1162 | catch (Exception e) |
1156 | { | 1163 | { |
@@ -1510,38 +1517,49 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1510 | OutPacket(pc, ThrottleOutPacketType.Unknown); | 1517 | OutPacket(pc, ThrottleOutPacketType.Unknown); |
1511 | } | 1518 | } |
1512 | 1519 | ||
1513 | public void SendKillObject(ulong regionHandle, uint localID) | 1520 | public void SendKillObject(ulong regionHandle, List<uint> localIDs) |
1514 | { | 1521 | { |
1515 | // m_log.DebugFormat("[CLIENT]: Sending KillObjectPacket to {0} for {1} in {2}", Name, localID, regionHandle); | 1522 | // m_log.DebugFormat("[CLIENT]: Sending KillObjectPacket to {0} for {1} in {2}", Name, localID, regionHandle); |
1516 | 1523 | ||
1517 | KillObjectPacket kill = (KillObjectPacket)PacketPool.Instance.GetPacket(PacketType.KillObject); | 1524 | KillObjectPacket kill = (KillObjectPacket)PacketPool.Instance.GetPacket(PacketType.KillObject); |
1518 | // TODO: don't create new blocks if recycling an old packet | 1525 | // TODO: don't create new blocks if recycling an old packet |
1519 | kill.ObjectData = new KillObjectPacket.ObjectDataBlock[1]; | 1526 | kill.ObjectData = new KillObjectPacket.ObjectDataBlock[localIDs.Count]; |
1520 | kill.ObjectData[0] = new KillObjectPacket.ObjectDataBlock(); | 1527 | for (int i = 0 ; i < localIDs.Count ; i++ ) |
1521 | kill.ObjectData[0].ID = localID; | 1528 | { |
1529 | kill.ObjectData[i] = new KillObjectPacket.ObjectDataBlock(); | ||
1530 | kill.ObjectData[i].ID = localIDs[i]; | ||
1531 | } | ||
1522 | kill.Header.Reliable = true; | 1532 | kill.Header.Reliable = true; |
1523 | kill.Header.Zerocoded = true; | 1533 | kill.Header.Zerocoded = true; |
1524 | 1534 | ||
1525 | if (m_scene.GetScenePresence(localID) == null) | 1535 | lock (m_killRecord) |
1526 | { | 1536 | { |
1527 | // We must lock for both manipulating the kill record and sending the packet, in order to avoid a race | 1537 | if (localIDs.Count == 1) |
1528 | // condition where a kill can be processed before an out-of-date update for the same object. | ||
1529 | lock (m_killRecord) | ||
1530 | { | 1538 | { |
1531 | m_killRecord.Add(localID); | 1539 | if (m_scene.GetScenePresence(localIDs[0]) != null) |
1532 | 1540 | { | |
1533 | // The throttle queue used here must match that being used for updates. Otherwise, there is a | 1541 | OutPacket(kill, ThrottleOutPacketType.State); |
1534 | // chance that a kill packet put on a separate queue will be sent to the client before an existing | 1542 | return; |
1535 | // update packet on another queue. Receiving updates after kills results in unowned and undeletable | 1543 | } |
1536 | // scene objects in a viewer until that viewer is relogged in. | 1544 | m_killRecord.Add(localIDs[0]); |
1537 | OutPacket(kill, ThrottleOutPacketType.Task); | 1545 | } |
1546 | else | ||
1547 | { | ||
1548 | lock (m_entityUpdates.SyncRoot) | ||
1549 | { | ||
1550 | foreach (uint localID in localIDs) | ||
1551 | m_killRecord.Add(localID); | ||
1552 | } | ||
1538 | } | 1553 | } |
1539 | } | 1554 | } |
1540 | else | 1555 | |
1541 | { | 1556 | // The throttle queue used here must match that being used for |
1542 | // OutPacket(kill, ThrottleOutPacketType.State); | 1557 | // updates. Otherwise, there is a chance that a kill packet put |
1543 | OutPacket(kill, ThrottleOutPacketType.Task); | 1558 | // on a separate queue will be sent to the client before an |
1544 | } | 1559 | // existing update packet on another queue. Receiving updates |
1560 | // after kills results in unowned and undeletable | ||
1561 | // scene objects in a viewer until that viewer is relogged in. | ||
1562 | OutPacket(kill, ThrottleOutPacketType.Task); | ||
1545 | } | 1563 | } |
1546 | 1564 | ||
1547 | /// <summary> | 1565 | /// <summary> |
@@ -2259,6 +2277,15 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
2259 | OutPacket(sound, ThrottleOutPacketType.Task); | 2277 | OutPacket(sound, ThrottleOutPacketType.Task); |
2260 | } | 2278 | } |
2261 | 2279 | ||
2280 | public void SendTransferAbort(TransferRequestPacket transferRequest) | ||
2281 | { | ||
2282 | TransferAbortPacket abort = (TransferAbortPacket)PacketPool.Instance.GetPacket(PacketType.TransferAbort); | ||
2283 | abort.TransferInfo.TransferID = transferRequest.TransferInfo.TransferID; | ||
2284 | abort.TransferInfo.ChannelType = transferRequest.TransferInfo.ChannelType; | ||
2285 | m_log.Debug("[Assets] Aborting transfer; asset request failed"); | ||
2286 | OutPacket(abort, ThrottleOutPacketType.Task); | ||
2287 | } | ||
2288 | |||
2262 | public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) | 2289 | public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) |
2263 | { | 2290 | { |
2264 | SoundTriggerPacket sound = (SoundTriggerPacket)PacketPool.Instance.GetPacket(PacketType.SoundTrigger); | 2291 | SoundTriggerPacket sound = (SoundTriggerPacket)PacketPool.Instance.GetPacket(PacketType.SoundTrigger); |
@@ -3563,7 +3590,15 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
3563 | /// </summary> | 3590 | /// </summary> |
3564 | public void SendPrimUpdate(ISceneEntity entity, PrimUpdateFlags updateFlags) | 3591 | public void SendPrimUpdate(ISceneEntity entity, PrimUpdateFlags updateFlags) |
3565 | { | 3592 | { |
3566 | //double priority = m_prioritizer.GetUpdatePriority(this, entity); | 3593 | if (entity is SceneObjectPart) |
3594 | { | ||
3595 | SceneObjectPart e = (SceneObjectPart)entity; | ||
3596 | SceneObjectGroup g = e.ParentGroup; | ||
3597 | if (g.RootPart.Shape.State > 30) // HUD | ||
3598 | if (g.OwnerID != AgentId) | ||
3599 | return; // Don't send updates for other people's HUDs | ||
3600 | } | ||
3601 | |||
3567 | uint priority = m_prioritizer.GetUpdatePriority(this, entity); | 3602 | uint priority = m_prioritizer.GetUpdatePriority(this, entity); |
3568 | 3603 | ||
3569 | lock (m_entityUpdates.SyncRoot) | 3604 | lock (m_entityUpdates.SyncRoot) |
@@ -3630,211 +3665,230 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
3630 | 3665 | ||
3631 | // We must lock for both manipulating the kill record and sending the packet, in order to avoid a race | 3666 | // We must lock for both manipulating the kill record and sending the packet, in order to avoid a race |
3632 | // condition where a kill can be processed before an out-of-date update for the same object. | 3667 | // condition where a kill can be processed before an out-of-date update for the same object. |
3633 | lock (m_killRecord) | 3668 | float avgTimeDilation = 1.0f; |
3669 | IEntityUpdate iupdate; | ||
3670 | Int32 timeinqueue; // this is just debugging code & can be dropped later | ||
3671 | |||
3672 | while (updatesThisCall < maxUpdates) | ||
3634 | { | 3673 | { |
3635 | float avgTimeDilation = 1.0f; | 3674 | lock (m_entityUpdates.SyncRoot) |
3636 | IEntityUpdate iupdate; | 3675 | if (!m_entityUpdates.TryDequeue(out iupdate, out timeinqueue)) |
3637 | Int32 timeinqueue; // this is just debugging code & can be dropped later | 3676 | break; |
3638 | |||
3639 | while (updatesThisCall < maxUpdates) | ||
3640 | { | ||
3641 | lock (m_entityUpdates.SyncRoot) | ||
3642 | if (!m_entityUpdates.TryDequeue(out iupdate, out timeinqueue)) | ||
3643 | break; | ||
3644 | 3677 | ||
3645 | EntityUpdate update = (EntityUpdate)iupdate; | 3678 | EntityUpdate update = (EntityUpdate)iupdate; |
3646 | 3679 | ||
3647 | avgTimeDilation += update.TimeDilation; | 3680 | avgTimeDilation += update.TimeDilation; |
3648 | avgTimeDilation *= 0.5f; | 3681 | avgTimeDilation *= 0.5f; |
3649 | 3682 | ||
3650 | if (update.Entity is SceneObjectPart) | 3683 | if (update.Entity is SceneObjectPart) |
3684 | { | ||
3685 | SceneObjectPart part = (SceneObjectPart)update.Entity; | ||
3686 | |||
3687 | // Please do not remove this unless you can demonstrate on the OpenSim mailing list that a client | ||
3688 | // will never receive an update after a prim kill. Even then, keeping the kill record may be a good | ||
3689 | // safety measure. | ||
3690 | // | ||
3691 | // If a Linden Lab 1.23.5 client (and possibly later and earlier) receives an object update | ||
3692 | // after a kill, it will keep displaying the deleted object until relog. OpenSim currently performs | ||
3693 | // updates and kills on different threads with different scheduling strategies, hence this protection. | ||
3694 | // | ||
3695 | // This doesn't appear to apply to child prims - a client will happily ignore these updates | ||
3696 | // after the root prim has been deleted. | ||
3697 | lock (m_killRecord) | ||
3651 | { | 3698 | { |
3652 | SceneObjectPart part = (SceneObjectPart)update.Entity; | ||
3653 | |||
3654 | // Please do not remove this unless you can demonstrate on the OpenSim mailing list that a client | ||
3655 | // will never receive an update after a prim kill. Even then, keeping the kill record may be a good | ||
3656 | // safety measure. | ||
3657 | // | ||
3658 | // If a Linden Lab 1.23.5 client (and possibly later and earlier) receives an object update | ||
3659 | // after a kill, it will keep displaying the deleted object until relog. OpenSim currently performs | ||
3660 | // updates and kills on different threads with different scheduling strategies, hence this protection. | ||
3661 | // | ||
3662 | // This doesn't appear to apply to child prims - a client will happily ignore these updates | ||
3663 | // after the root prim has been deleted. | ||
3664 | if (m_killRecord.Contains(part.LocalId)) | 3699 | if (m_killRecord.Contains(part.LocalId)) |
3665 | { | ||
3666 | // m_log.WarnFormat( | ||
3667 | // "[CLIENT]: Preventing update for prim with local id {0} after client for user {1} told it was deleted", | ||
3668 | // part.LocalId, Name); | ||
3669 | continue; | 3700 | continue; |
3670 | } | 3701 | if (m_killRecord.Contains(part.ParentGroup.RootPart.LocalId)) |
3671 | 3702 | continue; | |
3672 | if (part.ParentGroup.IsAttachment && m_disableFacelights) | 3703 | } |
3704 | |||
3705 | if (part.ParentGroup.IsDeleted) | ||
3706 | continue; | ||
3707 | |||
3708 | if (part.ParentGroup.IsAttachment) | ||
3709 | { // Someone else's HUD, why are we getting these? | ||
3710 | if (part.ParentGroup.OwnerID != AgentId && | ||
3711 | part.ParentGroup.RootPart.Shape.State >= 30) | ||
3712 | continue; | ||
3713 | ScenePresence sp; | ||
3714 | // Owner is not in the sim, don't update it to | ||
3715 | // anyone | ||
3716 | if (!m_scene.TryGetScenePresence(part.OwnerID, out sp)) | ||
3717 | continue; | ||
3718 | |||
3719 | List<SceneObjectGroup> atts = sp.Attachments; | ||
3720 | bool found = false; | ||
3721 | foreach (SceneObjectGroup att in atts) | ||
3673 | { | 3722 | { |
3674 | if (part.ParentGroup.RootPart.Shape.State != (byte)AttachmentPoint.LeftHand && | 3723 | if (att == part.ParentGroup) |
3675 | part.ParentGroup.RootPart.Shape.State != (byte)AttachmentPoint.RightHand) | ||
3676 | { | 3724 | { |
3677 | part.Shape.LightEntry = false; | 3725 | found = true; |
3726 | break; | ||
3678 | } | 3727 | } |
3679 | } | 3728 | } |
3729 | |||
3730 | // It's an attachment of a valid avatar, but | ||
3731 | // doesn't seem to be attached, skip | ||
3732 | if (!found) | ||
3733 | continue; | ||
3680 | } | 3734 | } |
3681 | 3735 | if (part.ParentGroup.IsAttachment && m_disableFacelights) | |
3682 | ++updatesThisCall; | ||
3683 | |||
3684 | #region UpdateFlags to packet type conversion | ||
3685 | |||
3686 | PrimUpdateFlags updateFlags = (PrimUpdateFlags)update.Flags; | ||
3687 | |||
3688 | bool canUseCompressed = true; | ||
3689 | bool canUseImproved = true; | ||
3690 | |||
3691 | // Compressed object updates only make sense for LL primitives | ||
3692 | if (!(update.Entity is SceneObjectPart)) | ||
3693 | { | ||
3694 | canUseCompressed = false; | ||
3695 | } | ||
3696 | |||
3697 | if (updateFlags.HasFlag(PrimUpdateFlags.FullUpdate)) | ||
3698 | { | ||
3699 | canUseCompressed = false; | ||
3700 | canUseImproved = false; | ||
3701 | } | ||
3702 | else | ||
3703 | { | 3736 | { |
3704 | if (updateFlags.HasFlag(PrimUpdateFlags.Velocity) || | 3737 | if (part.ParentGroup.RootPart.Shape.State != (byte)AttachmentPoint.LeftHand && |
3705 | updateFlags.HasFlag(PrimUpdateFlags.Acceleration) || | 3738 | part.ParentGroup.RootPart.Shape.State != (byte)AttachmentPoint.RightHand) |
3706 | updateFlags.HasFlag(PrimUpdateFlags.CollisionPlane) || | ||
3707 | updateFlags.HasFlag(PrimUpdateFlags.Joint)) | ||
3708 | { | 3739 | { |
3709 | canUseCompressed = false; | 3740 | part.Shape.LightEntry = false; |
3710 | } | ||
3711 | |||
3712 | if (updateFlags.HasFlag(PrimUpdateFlags.PrimFlags) || | ||
3713 | updateFlags.HasFlag(PrimUpdateFlags.ParentID) || | ||
3714 | updateFlags.HasFlag(PrimUpdateFlags.Scale) || | ||
3715 | updateFlags.HasFlag(PrimUpdateFlags.PrimData) || | ||
3716 | updateFlags.HasFlag(PrimUpdateFlags.Text) || | ||
3717 | updateFlags.HasFlag(PrimUpdateFlags.NameValue) || | ||
3718 | updateFlags.HasFlag(PrimUpdateFlags.ExtraData) || | ||
3719 | updateFlags.HasFlag(PrimUpdateFlags.TextureAnim) || | ||
3720 | updateFlags.HasFlag(PrimUpdateFlags.Sound) || | ||
3721 | updateFlags.HasFlag(PrimUpdateFlags.Particles) || | ||
3722 | updateFlags.HasFlag(PrimUpdateFlags.Material) || | ||
3723 | updateFlags.HasFlag(PrimUpdateFlags.ClickAction) || | ||
3724 | updateFlags.HasFlag(PrimUpdateFlags.MediaURL) || | ||
3725 | updateFlags.HasFlag(PrimUpdateFlags.Joint)) | ||
3726 | { | ||
3727 | canUseImproved = false; | ||
3728 | } | 3741 | } |
3729 | } | 3742 | } |
3730 | 3743 | } | |
3731 | #endregion UpdateFlags to packet type conversion | 3744 | |
3732 | 3745 | ++updatesThisCall; | |
3733 | #region Block Construction | 3746 | |
3734 | 3747 | #region UpdateFlags to packet type conversion | |
3735 | // TODO: Remove this once we can build compressed updates | 3748 | |
3749 | PrimUpdateFlags updateFlags = (PrimUpdateFlags)update.Flags; | ||
3750 | |||
3751 | bool canUseCompressed = true; | ||
3752 | bool canUseImproved = true; | ||
3753 | |||
3754 | // Compressed object updates only make sense for LL primitives | ||
3755 | if (!(update.Entity is SceneObjectPart)) | ||
3756 | { | ||
3736 | canUseCompressed = false; | 3757 | canUseCompressed = false; |
3737 | 3758 | } | |
3738 | if (!canUseImproved && !canUseCompressed) | 3759 | |
3739 | { | 3760 | if (updateFlags.HasFlag(PrimUpdateFlags.FullUpdate)) |
3740 | if (update.Entity is ScenePresence) | 3761 | { |
3741 | { | 3762 | canUseCompressed = false; |
3742 | objectUpdateBlocks.Value.Add(CreateAvatarUpdateBlock((ScenePresence)update.Entity)); | 3763 | canUseImproved = false; |
3743 | objectUpdates.Value.Add(update); | 3764 | } |
3744 | } | 3765 | else |
3745 | else | 3766 | { |
3746 | { | 3767 | if (updateFlags.HasFlag(PrimUpdateFlags.Velocity) || |
3747 | objectUpdateBlocks.Value.Add(CreatePrimUpdateBlock((SceneObjectPart)update.Entity, this.m_agentId)); | 3768 | updateFlags.HasFlag(PrimUpdateFlags.Acceleration) || |
3748 | objectUpdates.Value.Add(update); | 3769 | updateFlags.HasFlag(PrimUpdateFlags.CollisionPlane) || |
3749 | } | 3770 | updateFlags.HasFlag(PrimUpdateFlags.Joint)) |
3750 | } | ||
3751 | else if (!canUseImproved) | ||
3752 | { | 3771 | { |
3753 | compressedUpdateBlocks.Value.Add(CreateCompressedUpdateBlock((SceneObjectPart)update.Entity, updateFlags)); | 3772 | canUseCompressed = false; |
3754 | compressedUpdates.Value.Add(update); | ||
3755 | } | 3773 | } |
3756 | else | 3774 | |
3775 | if (updateFlags.HasFlag(PrimUpdateFlags.PrimFlags) || | ||
3776 | updateFlags.HasFlag(PrimUpdateFlags.ParentID) || | ||
3777 | updateFlags.HasFlag(PrimUpdateFlags.Scale) || | ||
3778 | updateFlags.HasFlag(PrimUpdateFlags.PrimData) || | ||
3779 | updateFlags.HasFlag(PrimUpdateFlags.Text) || | ||
3780 | updateFlags.HasFlag(PrimUpdateFlags.NameValue) || | ||
3781 | updateFlags.HasFlag(PrimUpdateFlags.ExtraData) || | ||
3782 | updateFlags.HasFlag(PrimUpdateFlags.TextureAnim) || | ||
3783 | updateFlags.HasFlag(PrimUpdateFlags.Sound) || | ||
3784 | updateFlags.HasFlag(PrimUpdateFlags.Particles) || | ||
3785 | updateFlags.HasFlag(PrimUpdateFlags.Material) || | ||
3786 | updateFlags.HasFlag(PrimUpdateFlags.ClickAction) || | ||
3787 | updateFlags.HasFlag(PrimUpdateFlags.MediaURL) || | ||
3788 | updateFlags.HasFlag(PrimUpdateFlags.Joint)) | ||
3757 | { | 3789 | { |
3758 | if (update.Entity is ScenePresence && ((ScenePresence)update.Entity).UUID == AgentId) | 3790 | canUseImproved = false; |
3759 | { | ||
3760 | // Self updates go into a special list | ||
3761 | terseAgentUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); | ||
3762 | terseAgentUpdates.Value.Add(update); | ||
3763 | } | ||
3764 | else | ||
3765 | { | ||
3766 | // Everything else goes here | ||
3767 | terseUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); | ||
3768 | terseUpdates.Value.Add(update); | ||
3769 | } | ||
3770 | } | 3791 | } |
3771 | |||
3772 | #endregion Block Construction | ||
3773 | } | 3792 | } |
3774 | |||
3775 | |||
3776 | #region Packet Sending | ||
3777 | ushort timeDilation = Utils.FloatToUInt16(avgTimeDilation, 0.0f, 1.0f); | ||
3778 | 3793 | ||
3779 | if (terseAgentUpdateBlocks.IsValueCreated) | 3794 | #endregion UpdateFlags to packet type conversion |
3780 | { | ||
3781 | List<ImprovedTerseObjectUpdatePacket.ObjectDataBlock> blocks = terseAgentUpdateBlocks.Value; | ||
3782 | 3795 | ||
3783 | ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); | 3796 | #region Block Construction |
3784 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3785 | packet.RegionData.TimeDilation = timeDilation; | ||
3786 | packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | ||
3787 | 3797 | ||
3788 | for (int i = 0; i < blocks.Count; i++) | 3798 | // TODO: Remove this once we can build compressed updates |
3789 | packet.ObjectData[i] = blocks[i]; | 3799 | canUseCompressed = false; |
3790 | // If any of the packets created from this call go unacknowledged, all of the updates will be resent | ||
3791 | OutPacket(packet, ThrottleOutPacketType.Unknown, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(terseAgentUpdates.Value, oPacket); }); | ||
3792 | } | ||
3793 | 3800 | ||
3794 | if (objectUpdateBlocks.IsValueCreated) | 3801 | if (!canUseImproved && !canUseCompressed) |
3795 | { | 3802 | { |
3796 | List<ObjectUpdatePacket.ObjectDataBlock> blocks = objectUpdateBlocks.Value; | 3803 | if (update.Entity is ScenePresence) |
3797 | 3804 | { | |
3798 | ObjectUpdatePacket packet = (ObjectUpdatePacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdate); | 3805 | objectUpdateBlocks.Value.Add(CreateAvatarUpdateBlock((ScenePresence)update.Entity)); |
3799 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | 3806 | } |
3800 | packet.RegionData.TimeDilation = timeDilation; | 3807 | else |
3801 | packet.ObjectData = new ObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | 3808 | { |
3802 | 3809 | objectUpdateBlocks.Value.Add(CreatePrimUpdateBlock((SceneObjectPart)update.Entity, this.m_agentId)); | |
3803 | for (int i = 0; i < blocks.Count; i++) | 3810 | } |
3804 | packet.ObjectData[i] = blocks[i]; | ||
3805 | // If any of the packets created from this call go unacknowledged, all of the updates will be resent | ||
3806 | OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(objectUpdates.Value, oPacket); }); | ||
3807 | } | 3811 | } |
3808 | 3812 | else if (!canUseImproved) | |
3809 | if (compressedUpdateBlocks.IsValueCreated) | ||
3810 | { | 3813 | { |
3811 | List<ObjectUpdateCompressedPacket.ObjectDataBlock> blocks = compressedUpdateBlocks.Value; | 3814 | compressedUpdateBlocks.Value.Add(CreateCompressedUpdateBlock((SceneObjectPart)update.Entity, updateFlags)); |
3812 | |||
3813 | ObjectUpdateCompressedPacket packet = (ObjectUpdateCompressedPacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdateCompressed); | ||
3814 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3815 | packet.RegionData.TimeDilation = timeDilation; | ||
3816 | packet.ObjectData = new ObjectUpdateCompressedPacket.ObjectDataBlock[blocks.Count]; | ||
3817 | |||
3818 | for (int i = 0; i < blocks.Count; i++) | ||
3819 | packet.ObjectData[i] = blocks[i]; | ||
3820 | // If any of the packets created from this call go unacknowledged, all of the updates will be resent | ||
3821 | OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(compressedUpdates.Value, oPacket); }); | ||
3822 | } | 3815 | } |
3823 | 3816 | else | |
3824 | if (terseUpdateBlocks.IsValueCreated) | ||
3825 | { | 3817 | { |
3826 | List<ImprovedTerseObjectUpdatePacket.ObjectDataBlock> blocks = terseUpdateBlocks.Value; | 3818 | if (update.Entity is ScenePresence && ((ScenePresence)update.Entity).UUID == AgentId) |
3827 | 3819 | // Self updates go into a special list | |
3828 | ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); | 3820 | terseAgentUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); |
3829 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | 3821 | else |
3830 | packet.RegionData.TimeDilation = timeDilation; | 3822 | // Everything else goes here |
3831 | packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | 3823 | terseUpdateBlocks.Value.Add(CreateImprovedTerseBlock(update.Entity, updateFlags.HasFlag(PrimUpdateFlags.Textures))); |
3832 | |||
3833 | for (int i = 0; i < blocks.Count; i++) | ||
3834 | packet.ObjectData[i] = blocks[i]; | ||
3835 | // If any of the packets created from this call go unacknowledged, all of the updates will be resent | ||
3836 | OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(terseUpdates.Value, oPacket); }); | ||
3837 | } | 3824 | } |
3825 | |||
3826 | #endregion Block Construction | ||
3827 | } | ||
3828 | |||
3829 | #region Packet Sending | ||
3830 | |||
3831 | const float TIME_DILATION = 1.0f; | ||
3832 | ushort timeDilation = Utils.FloatToUInt16(avgTimeDilation, 0.0f, 1.0f); | ||
3833 | |||
3834 | if (terseAgentUpdateBlocks.IsValueCreated) | ||
3835 | { | ||
3836 | List<ImprovedTerseObjectUpdatePacket.ObjectDataBlock> blocks = terseAgentUpdateBlocks.Value; | ||
3837 | |||
3838 | ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); | ||
3839 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3840 | packet.RegionData.TimeDilation = timeDilation; | ||
3841 | packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | ||
3842 | |||
3843 | for (int i = 0; i < blocks.Count; i++) | ||
3844 | packet.ObjectData[i] = blocks[i]; | ||
3845 | |||
3846 | OutPacket(packet, ThrottleOutPacketType.Unknown, true); | ||
3847 | } | ||
3848 | |||
3849 | if (objectUpdateBlocks.IsValueCreated) | ||
3850 | { | ||
3851 | List<ObjectUpdatePacket.ObjectDataBlock> blocks = objectUpdateBlocks.Value; | ||
3852 | |||
3853 | ObjectUpdatePacket packet = (ObjectUpdatePacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdate); | ||
3854 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3855 | packet.RegionData.TimeDilation = timeDilation; | ||
3856 | packet.ObjectData = new ObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | ||
3857 | |||
3858 | for (int i = 0; i < blocks.Count; i++) | ||
3859 | packet.ObjectData[i] = blocks[i]; | ||
3860 | |||
3861 | OutPacket(packet, ThrottleOutPacketType.Task, true); | ||
3862 | } | ||
3863 | |||
3864 | if (compressedUpdateBlocks.IsValueCreated) | ||
3865 | { | ||
3866 | List<ObjectUpdateCompressedPacket.ObjectDataBlock> blocks = compressedUpdateBlocks.Value; | ||
3867 | |||
3868 | ObjectUpdateCompressedPacket packet = (ObjectUpdateCompressedPacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdateCompressed); | ||
3869 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3870 | packet.RegionData.TimeDilation = timeDilation; | ||
3871 | packet.ObjectData = new ObjectUpdateCompressedPacket.ObjectDataBlock[blocks.Count]; | ||
3872 | |||
3873 | for (int i = 0; i < blocks.Count; i++) | ||
3874 | packet.ObjectData[i] = blocks[i]; | ||
3875 | |||
3876 | OutPacket(packet, ThrottleOutPacketType.Task, true); | ||
3877 | } | ||
3878 | |||
3879 | if (terseUpdateBlocks.IsValueCreated) | ||
3880 | { | ||
3881 | List<ImprovedTerseObjectUpdatePacket.ObjectDataBlock> blocks = terseUpdateBlocks.Value; | ||
3882 | |||
3883 | ImprovedTerseObjectUpdatePacket packet = new ImprovedTerseObjectUpdatePacket(); | ||
3884 | packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; | ||
3885 | packet.RegionData.TimeDilation = timeDilation; | ||
3886 | packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[blocks.Count]; | ||
3887 | |||
3888 | for (int i = 0; i < blocks.Count; i++) | ||
3889 | packet.ObjectData[i] = blocks[i]; | ||
3890 | |||
3891 | OutPacket(packet, ThrottleOutPacketType.Task, true); | ||
3838 | } | 3892 | } |
3839 | 3893 | ||
3840 | #endregion Packet Sending | 3894 | #endregion Packet Sending |
@@ -4318,37 +4372,44 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
4318 | if (bl[i].BannedUserID == UUID.Zero) | 4372 | if (bl[i].BannedUserID == UUID.Zero) |
4319 | continue; | 4373 | continue; |
4320 | BannedUsers.Add(bl[i].BannedUserID); | 4374 | BannedUsers.Add(bl[i].BannedUserID); |
4321 | } | ||
4322 | 4375 | ||
4323 | EstateOwnerMessagePacket packet = new EstateOwnerMessagePacket(); | 4376 | if (BannedUsers.Count >= 50 || (i == (bl.Length - 1) && BannedUsers.Count > 0)) |
4324 | packet.AgentData.TransactionID = UUID.Random(); | 4377 | { |
4325 | packet.AgentData.AgentID = AgentId; | 4378 | EstateOwnerMessagePacket packet = new EstateOwnerMessagePacket(); |
4326 | packet.AgentData.SessionID = SessionId; | 4379 | packet.AgentData.TransactionID = UUID.Random(); |
4327 | packet.MethodData.Invoice = invoice; | 4380 | packet.AgentData.AgentID = AgentId; |
4328 | packet.MethodData.Method = Utils.StringToBytes("setaccess"); | 4381 | packet.AgentData.SessionID = SessionId; |
4382 | packet.MethodData.Invoice = invoice; | ||
4383 | packet.MethodData.Method = Utils.StringToBytes("setaccess"); | ||
4329 | 4384 | ||
4330 | EstateOwnerMessagePacket.ParamListBlock[] returnblock = new EstateOwnerMessagePacket.ParamListBlock[6 + BannedUsers.Count]; | 4385 | EstateOwnerMessagePacket.ParamListBlock[] returnblock = new EstateOwnerMessagePacket.ParamListBlock[6 + BannedUsers.Count]; |
4331 | 4386 | ||
4332 | for (int i = 0; i < (6 + BannedUsers.Count); i++) | 4387 | int j; |
4333 | { | 4388 | for (j = 0; j < (6 + BannedUsers.Count); j++) |
4334 | returnblock[i] = new EstateOwnerMessagePacket.ParamListBlock(); | 4389 | { |
4335 | } | 4390 | returnblock[j] = new EstateOwnerMessagePacket.ParamListBlock(); |
4336 | int j = 0; | 4391 | } |
4392 | j = 0; | ||
4337 | 4393 | ||
4338 | returnblock[j].Parameter = Utils.StringToBytes(estateID.ToString()); j++; | 4394 | returnblock[j].Parameter = Utils.StringToBytes(estateID.ToString()); j++; |
4339 | returnblock[j].Parameter = Utils.StringToBytes(((int)Constants.EstateAccessCodex.EstateBans).ToString()); j++; | 4395 | returnblock[j].Parameter = Utils.StringToBytes(((int)Constants.EstateAccessCodex.EstateBans).ToString()); j++; |
4340 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; | 4396 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; |
4341 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; | 4397 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; |
4342 | returnblock[j].Parameter = Utils.StringToBytes(BannedUsers.Count.ToString()); j++; | 4398 | returnblock[j].Parameter = Utils.StringToBytes(BannedUsers.Count.ToString()); j++; |
4343 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; | 4399 | returnblock[j].Parameter = Utils.StringToBytes("0"); j++; |
4344 | 4400 | ||
4345 | foreach (UUID banned in BannedUsers) | 4401 | foreach (UUID banned in BannedUsers) |
4346 | { | 4402 | { |
4347 | returnblock[j].Parameter = banned.GetBytes(); j++; | 4403 | returnblock[j].Parameter = banned.GetBytes(); j++; |
4404 | } | ||
4405 | packet.ParamList = returnblock; | ||
4406 | packet.Header.Reliable = true; | ||
4407 | OutPacket(packet, ThrottleOutPacketType.Task); | ||
4408 | |||
4409 | BannedUsers.Clear(); | ||
4410 | } | ||
4348 | } | 4411 | } |
4349 | packet.ParamList = returnblock; | 4412 | |
4350 | packet.Header.Reliable = false; | ||
4351 | OutPacket(packet, ThrottleOutPacketType.Task); | ||
4352 | } | 4413 | } |
4353 | 4414 | ||
4354 | public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) | 4415 | public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) |
@@ -4646,14 +4707,14 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
4646 | 4707 | ||
4647 | if (notifyCount > 0) | 4708 | if (notifyCount > 0) |
4648 | { | 4709 | { |
4649 | if (notifyCount > 32) | 4710 | // if (notifyCount > 32) |
4650 | { | 4711 | // { |
4651 | m_log.InfoFormat( | 4712 | // m_log.InfoFormat( |
4652 | "[LAND]: More than {0} avatars own prims on this parcel. Only sending back details of first {0}" | 4713 | // "[LAND]: More than {0} avatars own prims on this parcel. Only sending back details of first {0}" |
4653 | + " - a developer might want to investigate whether this is a hard limit", 32); | 4714 | // + " - a developer might want to investigate whether this is a hard limit", 32); |
4654 | 4715 | // | |
4655 | notifyCount = 32; | 4716 | // notifyCount = 32; |
4656 | } | 4717 | // } |
4657 | 4718 | ||
4658 | ParcelObjectOwnersReplyPacket.DataBlock[] dataBlock | 4719 | ParcelObjectOwnersReplyPacket.DataBlock[] dataBlock |
4659 | = new ParcelObjectOwnersReplyPacket.DataBlock[notifyCount]; | 4720 | = new ParcelObjectOwnersReplyPacket.DataBlock[notifyCount]; |
@@ -5169,6 +5230,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
5169 | AddLocalPacketHandler(PacketType.TransferAbort, HandleTransferAbort, false); | 5230 | AddLocalPacketHandler(PacketType.TransferAbort, HandleTransferAbort, false); |
5170 | AddLocalPacketHandler(PacketType.MuteListRequest, HandleMuteListRequest, false); | 5231 | AddLocalPacketHandler(PacketType.MuteListRequest, HandleMuteListRequest, false); |
5171 | AddLocalPacketHandler(PacketType.UseCircuitCode, HandleUseCircuitCode); | 5232 | AddLocalPacketHandler(PacketType.UseCircuitCode, HandleUseCircuitCode); |
5233 | AddLocalPacketHandler(PacketType.CreateNewOutfitAttachments, HandleCreateNewOutfitAttachments); | ||
5172 | AddLocalPacketHandler(PacketType.AgentHeightWidth, HandleAgentHeightWidth, false); | 5234 | AddLocalPacketHandler(PacketType.AgentHeightWidth, HandleAgentHeightWidth, false); |
5173 | AddLocalPacketHandler(PacketType.InventoryDescendents, HandleInventoryDescendents); | 5235 | AddLocalPacketHandler(PacketType.InventoryDescendents, HandleInventoryDescendents); |
5174 | AddLocalPacketHandler(PacketType.DirPlacesQuery, HandleDirPlacesQuery); | 5236 | AddLocalPacketHandler(PacketType.DirPlacesQuery, HandleDirPlacesQuery); |
@@ -5268,6 +5330,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
5268 | (x.CameraLeftAxis != lastarg.CameraLeftAxis) || | 5330 | (x.CameraLeftAxis != lastarg.CameraLeftAxis) || |
5269 | (x.CameraUpAxis != lastarg.CameraUpAxis) || | 5331 | (x.CameraUpAxis != lastarg.CameraUpAxis) || |
5270 | (x.ControlFlags != lastarg.ControlFlags) || | 5332 | (x.ControlFlags != lastarg.ControlFlags) || |
5333 | (x.ControlFlags != 0) || | ||
5271 | (x.Far != lastarg.Far) || | 5334 | (x.Far != lastarg.Far) || |
5272 | (x.Flags != lastarg.Flags) || | 5335 | (x.Flags != lastarg.Flags) || |
5273 | (x.State != lastarg.State) || | 5336 | (x.State != lastarg.State) || |
@@ -5641,7 +5704,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
5641 | args.Channel = ch; | 5704 | args.Channel = ch; |
5642 | args.From = String.Empty; | 5705 | args.From = String.Empty; |
5643 | args.Message = Utils.BytesToString(msg); | 5706 | args.Message = Utils.BytesToString(msg); |
5644 | args.Type = ChatTypeEnum.Shout; | 5707 | args.Type = ChatTypeEnum.Region; //Behaviour in SL is that the response can be heard from any distance |
5645 | args.Position = new Vector3(); | 5708 | args.Position = new Vector3(); |
5646 | args.Scene = Scene; | 5709 | args.Scene = Scene; |
5647 | args.Sender = this; | 5710 | args.Sender = this; |
@@ -9628,7 +9691,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
9628 | handlerUpdateMuteListEntry(this, UpdateMuteListEntry.MuteData.MuteID, | 9691 | handlerUpdateMuteListEntry(this, UpdateMuteListEntry.MuteData.MuteID, |
9629 | Utils.BytesToString(UpdateMuteListEntry.MuteData.MuteName), | 9692 | Utils.BytesToString(UpdateMuteListEntry.MuteData.MuteName), |
9630 | UpdateMuteListEntry.MuteData.MuteType, | 9693 | UpdateMuteListEntry.MuteData.MuteType, |
9631 | UpdateMuteListEntry.AgentData.AgentID); | 9694 | UpdateMuteListEntry.MuteData.MuteFlags); |
9632 | return true; | 9695 | return true; |
9633 | } | 9696 | } |
9634 | return false; | 9697 | return false; |
@@ -9643,8 +9706,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
9643 | { | 9706 | { |
9644 | handlerRemoveMuteListEntry(this, | 9707 | handlerRemoveMuteListEntry(this, |
9645 | RemoveMuteListEntry.MuteData.MuteID, | 9708 | RemoveMuteListEntry.MuteData.MuteID, |
9646 | Utils.BytesToString(RemoveMuteListEntry.MuteData.MuteName), | 9709 | Utils.BytesToString(RemoveMuteListEntry.MuteData.MuteName)); |
9647 | RemoveMuteListEntry.AgentData.AgentID); | ||
9648 | return true; | 9710 | return true; |
9649 | } | 9711 | } |
9650 | return false; | 9712 | return false; |
@@ -9692,6 +9754,37 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
9692 | { | 9754 | { |
9693 | return true; | 9755 | return true; |
9694 | } | 9756 | } |
9757 | |||
9758 | private bool HandleCreateNewOutfitAttachments(IClientAPI sender, Packet Pack) | ||
9759 | { | ||
9760 | CreateNewOutfitAttachmentsPacket packet = (CreateNewOutfitAttachmentsPacket)Pack; | ||
9761 | |||
9762 | #region Packet Session and User Check | ||
9763 | if (m_checkPackets) | ||
9764 | { | ||
9765 | if (packet.AgentData.SessionID != SessionId || | ||
9766 | packet.AgentData.AgentID != AgentId) | ||
9767 | return true; | ||
9768 | } | ||
9769 | #endregion | ||
9770 | MoveItemsAndLeaveCopy handlerMoveItemsAndLeaveCopy = null; | ||
9771 | List<InventoryItemBase> items = new List<InventoryItemBase>(); | ||
9772 | foreach (CreateNewOutfitAttachmentsPacket.ObjectDataBlock n in packet.ObjectData) | ||
9773 | { | ||
9774 | InventoryItemBase b = new InventoryItemBase(); | ||
9775 | b.ID = n.OldItemID; | ||
9776 | b.Folder = n.OldFolderID; | ||
9777 | items.Add(b); | ||
9778 | } | ||
9779 | |||
9780 | handlerMoveItemsAndLeaveCopy = OnMoveItemsAndLeaveCopy; | ||
9781 | if (handlerMoveItemsAndLeaveCopy != null) | ||
9782 | { | ||
9783 | handlerMoveItemsAndLeaveCopy(this, items, packet.HeaderData.NewFolderID); | ||
9784 | } | ||
9785 | |||
9786 | return true; | ||
9787 | } | ||
9695 | 9788 | ||
9696 | private bool HandleAgentHeightWidth(IClientAPI sender, Packet Pack) | 9789 | private bool HandleAgentHeightWidth(IClientAPI sender, Packet Pack) |
9697 | { | 9790 | { |
@@ -10118,6 +10211,20 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
10118 | groupProfileReply.GroupData.MaturePublish = d.MaturePublish; | 10211 | groupProfileReply.GroupData.MaturePublish = d.MaturePublish; |
10119 | groupProfileReply.GroupData.OwnerRole = d.OwnerRole; | 10212 | groupProfileReply.GroupData.OwnerRole = d.OwnerRole; |
10120 | 10213 | ||
10214 | Scene scene = (Scene)m_scene; | ||
10215 | if (scene.Permissions.IsGod(sender.AgentId) && (!sender.IsGroupMember(groupProfileRequest.GroupData.GroupID))) | ||
10216 | { | ||
10217 | ScenePresence p; | ||
10218 | if (scene.TryGetScenePresence(sender.AgentId, out p)) | ||
10219 | { | ||
10220 | if (p.GodLevel >= 200) | ||
10221 | { | ||
10222 | groupProfileReply.GroupData.OpenEnrollment = true; | ||
10223 | groupProfileReply.GroupData.MembershipFee = 0; | ||
10224 | } | ||
10225 | } | ||
10226 | } | ||
10227 | |||
10121 | OutPacket(groupProfileReply, ThrottleOutPacketType.Task); | 10228 | OutPacket(groupProfileReply, ThrottleOutPacketType.Task); |
10122 | } | 10229 | } |
10123 | return true; | 10230 | return true; |
@@ -10690,11 +10797,16 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
10690 | 10797 | ||
10691 | StartLure handlerStartLure = OnStartLure; | 10798 | StartLure handlerStartLure = OnStartLure; |
10692 | if (handlerStartLure != null) | 10799 | if (handlerStartLure != null) |
10693 | handlerStartLure(startLureRequest.Info.LureType, | 10800 | { |
10694 | Utils.BytesToString( | 10801 | for (int i = 0 ; i < startLureRequest.TargetData.Length ; i++) |
10695 | startLureRequest.Info.Message), | 10802 | { |
10696 | startLureRequest.TargetData[0].TargetID, | 10803 | handlerStartLure(startLureRequest.Info.LureType, |
10697 | this); | 10804 | Utils.BytesToString( |
10805 | startLureRequest.Info.Message), | ||
10806 | startLureRequest.TargetData[i].TargetID, | ||
10807 | this); | ||
10808 | } | ||
10809 | } | ||
10698 | return true; | 10810 | return true; |
10699 | } | 10811 | } |
10700 | private bool HandleTeleportLureRequest(IClientAPI sender, Packet Pack) | 10812 | private bool HandleTeleportLureRequest(IClientAPI sender, Packet Pack) |
@@ -10808,10 +10920,11 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
10808 | } | 10920 | } |
10809 | #endregion | 10921 | #endregion |
10810 | 10922 | ||
10811 | ClassifiedDelete handlerClassifiedGodDelete = OnClassifiedGodDelete; | 10923 | ClassifiedGodDelete handlerClassifiedGodDelete = OnClassifiedGodDelete; |
10812 | if (handlerClassifiedGodDelete != null) | 10924 | if (handlerClassifiedGodDelete != null) |
10813 | handlerClassifiedGodDelete( | 10925 | handlerClassifiedGodDelete( |
10814 | classifiedGodDelete.Data.ClassifiedID, | 10926 | classifiedGodDelete.Data.ClassifiedID, |
10927 | classifiedGodDelete.Data.QueryID, | ||
10815 | this); | 10928 | this); |
10816 | return true; | 10929 | return true; |
10817 | } | 10930 | } |
@@ -11189,7 +11302,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
11189 | { | 11302 | { |
11190 | // It's a ghost! tell the client to delete it from view. | 11303 | // It's a ghost! tell the client to delete it from view. |
11191 | simClient.SendKillObject(Scene.RegionInfo.RegionHandle, | 11304 | simClient.SendKillObject(Scene.RegionInfo.RegionHandle, |
11192 | localId); | 11305 | new List<uint>() { localId }); |
11193 | } | 11306 | } |
11194 | else | 11307 | else |
11195 | { | 11308 | { |
@@ -11578,22 +11691,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
11578 | /// <param name="Pack">OpenMetaverse.packet</param> | 11691 | /// <param name="Pack">OpenMetaverse.packet</param> |
11579 | public void ProcessInPacket(Packet packet) | 11692 | public void ProcessInPacket(Packet packet) |
11580 | { | 11693 | { |
11581 | if (m_debugPacketLevel > 0) | 11694 | if (m_debugPacketLevel >= 255) |
11582 | { | 11695 | m_log.DebugFormat("[CLIENT]: Packet IN {0}", packet.Type); |
11583 | bool outputPacket = true; | ||
11584 | |||
11585 | if (m_debugPacketLevel <= 255 && packet.Type == PacketType.AgentUpdate) | ||
11586 | outputPacket = false; | ||
11587 | |||
11588 | if (m_debugPacketLevel <= 200 && packet.Type == PacketType.RequestImage) | ||
11589 | outputPacket = false; | ||
11590 | |||
11591 | if (m_debugPacketLevel <= 100 && (packet.Type == PacketType.ViewerEffect || packet.Type == PacketType.AgentAnimation)) | ||
11592 | outputPacket = false; | ||
11593 | |||
11594 | if (outputPacket) | ||
11595 | m_log.DebugFormat("[CLIENT]: Packet IN {0}", packet.Type); | ||
11596 | } | ||
11597 | 11696 | ||
11598 | if (!ProcessPacketMethod(packet)) | 11697 | if (!ProcessPacketMethod(packet)) |
11599 | m_log.Warn("[CLIENT]: unhandled packet " + packet.Type); | 11698 | m_log.Warn("[CLIENT]: unhandled packet " + packet.Type); |
@@ -11835,7 +11934,10 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
11835 | 11934 | ||
11836 | // m_log.DebugFormat("[CLIENT]: {0} requesting asset {1}", Name, requestID); | 11935 | // m_log.DebugFormat("[CLIENT]: {0} requesting asset {1}", Name, requestID); |
11837 | 11936 | ||
11937 | |||
11938 | //Note, the bool returned from the below function is useless since it is always false. | ||
11838 | m_assetService.Get(requestID.ToString(), transferRequest, AssetReceived); | 11939 | m_assetService.Get(requestID.ToString(), transferRequest, AssetReceived); |
11940 | |||
11839 | } | 11941 | } |
11840 | 11942 | ||
11841 | /// <summary> | 11943 | /// <summary> |
diff --git a/OpenSim/Region/ClientStack/Linden/UDP/LLUDPClient.cs b/OpenSim/Region/ClientStack/Linden/UDP/LLUDPClient.cs index ca5501d..95a8e23 100644 --- a/OpenSim/Region/ClientStack/Linden/UDP/LLUDPClient.cs +++ b/OpenSim/Region/ClientStack/Linden/UDP/LLUDPClient.cs | |||
@@ -158,6 +158,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
158 | 158 | ||
159 | private int m_defaultRTO = 1000; // 1sec is the recommendation in the RFC | 159 | private int m_defaultRTO = 1000; // 1sec is the recommendation in the RFC |
160 | private int m_maxRTO = 60000; | 160 | private int m_maxRTO = 60000; |
161 | public bool m_deliverPackets = true; | ||
161 | 162 | ||
162 | /// <summary> | 163 | /// <summary> |
163 | /// Default constructor | 164 | /// Default constructor |
@@ -430,6 +431,13 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
430 | if (category >= 0 && category < m_packetOutboxes.Length) | 431 | if (category >= 0 && category < m_packetOutboxes.Length) |
431 | { | 432 | { |
432 | OpenSim.Framework.LocklessQueue<OutgoingPacket> queue = m_packetOutboxes[category]; | 433 | OpenSim.Framework.LocklessQueue<OutgoingPacket> queue = m_packetOutboxes[category]; |
434 | |||
435 | if (m_deliverPackets == false) | ||
436 | { | ||
437 | queue.Enqueue(packet); | ||
438 | return true; | ||
439 | } | ||
440 | |||
433 | TokenBucket bucket = m_throttleCategories[category]; | 441 | TokenBucket bucket = m_throttleCategories[category]; |
434 | 442 | ||
435 | // Don't send this packet if there is already a packet waiting in the queue | 443 | // Don't send this packet if there is already a packet waiting in the queue |
@@ -479,6 +487,8 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
479 | /// <returns>True if any packets were sent, otherwise false</returns> | 487 | /// <returns>True if any packets were sent, otherwise false</returns> |
480 | public bool DequeueOutgoing() | 488 | public bool DequeueOutgoing() |
481 | { | 489 | { |
490 | if (m_deliverPackets == false) return false; | ||
491 | |||
482 | OutgoingPacket packet; | 492 | OutgoingPacket packet; |
483 | OpenSim.Framework.LocklessQueue<OutgoingPacket> queue; | 493 | OpenSim.Framework.LocklessQueue<OutgoingPacket> queue; |
484 | TokenBucket bucket; | 494 | TokenBucket bucket; |
diff --git a/OpenSim/Region/ClientStack/Linden/UDP/LLUDPServer.cs b/OpenSim/Region/ClientStack/Linden/UDP/LLUDPServer.cs index aff90c5..ab6674d 100644 --- a/OpenSim/Region/ClientStack/Linden/UDP/LLUDPServer.cs +++ b/OpenSim/Region/ClientStack/Linden/UDP/LLUDPServer.cs | |||
@@ -960,7 +960,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
960 | if (m_scene.TryGetClient(udpClient.AgentID, out client)) | 960 | if (m_scene.TryGetClient(udpClient.AgentID, out client)) |
961 | { | 961 | { |
962 | client.IsLoggingOut = true; | 962 | client.IsLoggingOut = true; |
963 | client.Close(); | 963 | client.Close(false); |
964 | } | 964 | } |
965 | } | 965 | } |
966 | 966 | ||
@@ -972,6 +972,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
972 | 972 | ||
973 | while (base.IsRunning) | 973 | while (base.IsRunning) |
974 | { | 974 | { |
975 | m_scene.ThreadAlive(1); | ||
975 | try | 976 | try |
976 | { | 977 | { |
977 | IncomingPacket incomingPacket = null; | 978 | IncomingPacket incomingPacket = null; |
@@ -1014,6 +1015,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP | |||
1014 | 1015 | ||
1015 | while (base.IsRunning) | 1016 | while (base.IsRunning) |
1016 | { | 1017 | { |
1018 | m_scene.ThreadAlive(2); | ||
1017 | try | 1019 | try |
1018 | { | 1020 | { |
1019 | m_packetSent = false; | 1021 | m_packetSent = false; |
diff --git a/OpenSim/Region/ClientStack/Linden/UDP/OpenSimUDPBase.cs b/OpenSim/Region/ClientStack/Linden/UDP/OpenSimUDPBase.cs index 6eebd9d..d2779ba 100644 --- a/OpenSim/Region/ClientStack/Linden/UDP/OpenSimUDPBase.cs +++ b/OpenSim/Region/ClientStack/Linden/UDP/OpenSimUDPBase.cs | |||
@@ -100,10 +100,6 @@ namespace OpenMetaverse | |||
100 | const int SIO_UDP_CONNRESET = -1744830452; | 100 | const int SIO_UDP_CONNRESET = -1744830452; |
101 | 101 | ||
102 | IPEndPoint ipep = new IPEndPoint(m_localBindAddress, m_udpPort); | 102 | IPEndPoint ipep = new IPEndPoint(m_localBindAddress, m_udpPort); |
103 | |||
104 | m_log.DebugFormat( | ||
105 | "[UDPBASE]: Binding UDP listener using internal IP address config {0}:{1}", | ||
106 | ipep.Address, ipep.Port); | ||
107 | 103 | ||
108 | m_udpSocket = new Socket( | 104 | m_udpSocket = new Socket( |
109 | AddressFamily.InterNetwork, | 105 | AddressFamily.InterNetwork, |