/* * 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.Runtime; using System.Text; using System.Threading; using log4net; using OpenMetaverse; using OpenMetaverse.Packets; using OpenMetaverse.Messages.Linden; using OpenMetaverse.StructuredData; using OpenSim.Framework; using OpenSim.Framework.Client; using OpenSim.Framework.Monitoring; using OpenSim.Region.Framework.Interfaces; using OpenSim.Region.Framework.Scenes; using OpenSim.Services.Interfaces; using AssetLandmark = OpenSim.Framework.AssetLandmark; using Caps = OpenSim.Framework.Capabilities.Caps; using PermissionMask = OpenSim.Framework.PermissionMask; using RegionFlags = OpenMetaverse.RegionFlags; namespace OpenSim.Region.ClientStack.LindenUDP { public delegate bool PacketMethod(IClientAPI simClient, Packet packet); /// /// Handles new client connections /// Constructor takes a single Packet and authenticates everything /// public class LLClientView : IClientAPI, IClientCore, IClientIM, IClientChat, IClientInventory, IStatsCollector, IClientIPEndpoint { /// /// Debug packet level. See OpenSim.RegisterConsoleCommands() for more details. /// public int DebugPacketLevel { get; set; } #region Events public event BinaryGenericMessage OnBinaryGenericMessage; public event Action OnLogout; public event ObjectPermissions OnObjectPermissions; public event Action OnConnectionClosed; public event ViewerEffectEventHandler OnViewerEffect; public event ImprovedInstantMessage OnInstantMessage; public event ChatMessage OnChatFromClient; public event RezObject OnRezObject; public event DeRezObject OnDeRezObject; public event RezRestoreToWorld OnRezRestoreToWorld; public event ModifyTerrain OnModifyTerrain; public event Action OnRegionHandShakeReply; public event GenericCall1 OnRequestWearables; 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 Action OnCompleteMovementToRegion; public event UpdateAgent OnPreAgentUpdate; public event UpdateAgent OnAgentUpdate; public event UpdateAgent OnAgentCameraUpdate; public event AgentRequestSit OnAgentRequestSit; public event AgentSit OnAgentSit; public event AvatarPickerRequest OnAvatarPickerRequest; public event ChangeAnim OnChangeAnim; public event Action OnRequestAvatarsData; public event LinkObjects OnLinkObjects; public event DelinkObjects OnDelinkObjects; public event GrabObject OnGrabObject; public event DeGrabObject OnDeGrabObject; public event SpinStart OnSpinStart; public event SpinStop OnSpinStop; public event ObjectDuplicate OnObjectDuplicate; public event ObjectDuplicateOnRay OnObjectDuplicateOnRay; public event MoveObject OnGrabUpdate; public event SpinObject OnSpinUpdate; public event AddNewPrim OnAddPrim; public event RequestGodlikePowers OnRequestGodlikePowers; public event GodKickUser OnGodKickUser; public event ObjectExtraParams OnUpdateExtraParams; public event UpdateShape OnUpdatePrimShape; 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 ObjectIncludeInSearch OnObjectIncludeInSearch; public event RequestObjectPropertiesFamily OnRequestObjectPropertiesFamily; public event UpdatePrimFlags OnUpdatePrimFlags; public event UpdatePrimTexture OnUpdatePrimTexture; public event ClientChangeObject onClientChangeObject; public event UpdateVector OnUpdatePrimGroupPosition; public event UpdatePrimRotation OnUpdatePrimGroupRotation; public event UpdateVector OnUpdatePrimGroupScale; public event RequestMapBlocks OnRequestMapBlocks; public event RequestMapName OnMapNameRequest; public event TeleportLocationRequest OnTeleportLocationRequest; public event TeleportLandmarkRequest OnTeleportLandmarkRequest; public event TeleportCancel OnTeleportCancel; public event RequestAvatarProperties OnRequestAvatarProperties; public event SetAlwaysRun OnSetAlwaysRun; public event FetchInventory OnAgentDataUpdateRequest; public event TeleportLocationRequest OnSetStartLocationRequest; public event UpdateAvatarProperties OnUpdateAvatarProperties; 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 MoveItemsAndLeaveCopy OnMoveItemsAndLeaveCopy; public event MoveInventoryItem OnMoveInventoryItem; public event RemoveInventoryItem OnRemoveInventoryItem; public event RemoveInventoryFolder OnRemoveInventoryFolder; public event UDPAssetUploadRequest OnAssetUploadRequest; public event XferReceive OnXferReceive; public event RequestXfer OnRequestXfer; public event ConfirmXfer OnConfirmXfer; public event AbortXfer OnAbortXfer; public event RequestTerrain OnRequestTerrain; public event RezScript OnRezScript; public event UpdateTaskInventory OnUpdateTaskInventory; public event MoveTaskInventory OnMoveTaskItem; public event RemoveTaskInventory OnRemoveTaskItem; 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 ParcelBuy OnParcelBuy; public event UUIDNameRequest OnTeleportHomeRequest; public event UUIDNameRequest OnUUIDGroupNameRequest; public event ScriptAnswer OnScriptAnswer; public event RequestPayPrice OnRequestPayPrice; public event ObjectSaleInfo OnObjectSaleInfo; public event ObjectBuy OnObjectBuy; public event AgentSit OnUndo; public event AgentSit OnRedo; public event LandUndo OnLandUndo; public event ForceReleaseControls OnForceReleaseControls; public event GodLandStatRequest OnLandStatRequest; public event RequestObjectPropertiesFamily OnObjectGroupRequest; public event DetailedEstateDataRequest OnDetailedEstateDataRequest; public event SetEstateFlagsRequest OnSetEstateFlagsRequest; public event SetEstateTerrainDetailTexture OnSetEstateTerrainDetailTexture; public event SetEstateTerrainTextureHeights OnSetEstateTerrainTextureHeights; public event CommitEstateTerrainTextureRequest OnCommitEstateTerrainTextureRequest; public event SetRegionTerrainSettings OnSetRegionTerrainSettings; public event BakeTerrain OnBakeTerrain; public event RequestTerrain OnUploadTerrain; public event EstateChangeInfo OnEstateChangeInfo; public event EstateManageTelehub OnEstateManageTelehub; 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 RegionHandleRequest OnRegionHandleRequest; public event ParcelInfoRequest OnParcelInfoRequest; public event ScriptReset OnScriptReset; public event GetScriptRunning OnGetScriptRunning; public event SetScriptRunning OnSetScriptRunning; public event Action OnAutoPilotGo; 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 ClassifiedGodDelete 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 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 PlacesQuery OnPlacesQuery; public event AgentFOV OnAgentFOV; 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 ChangeInventoryItemFlags OnChangeInventoryItemFlags; public event MuteListEntryUpdate OnUpdateMuteListEntry; public event MuteListEntryRemove OnRemoveMuteListEntry; public event GodlikeMessage onGodlikeMessage; public event GodUpdateRegionInfoUpdate OnGodUpdateRegionInfoUpdate; public event GenericCall2 OnUpdateThrottles; #pragma warning disable 0067 // still unused public event GenericMessage OnGenericMessage; public event TextureRequest OnRequestTexture; public event StatusChange OnChildAgentStatus; public event GenericCall2 OnStopMovement; public event Action OnRemoveAvatar; public event DisconnectUser OnDisconnectUser; public event RequestAsset OnRequestAsset; public event BuyObjectInventory OnBuyObjectInventory; public event SetEstateTerrainBaseTexture OnSetEstateTerrainBaseTexture; public event TerrainUnacked OnUnackedTerrain; public event CachedTextureRequest OnCachedTextureRequest; public event UpdateVector OnUpdatePrimSinglePosition; public event StartAnim OnStartAnim; public event StopAnim OnStopAnim; public event UpdatePrimSingleRotation OnUpdatePrimSingleRotation; public event UpdatePrimSingleRotationPosition OnUpdatePrimSingleRotationPosition; public event UpdatePrimGroupRotation OnUpdatePrimGroupMouseRotation; public event UpdateVector OnUpdatePrimScale; #pragma warning restore 0067 #endregion Events #region Class Members // LLClientView Only public delegate void BinaryGenericMessage(Object sender, string method, byte[][] args); /// Used to adjust Sun Orbit values so Linden based viewers properly position sun private const float m_sunPainDaHalfOrbitalCutoff = 4.712388980384689858f; private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static string LogHeader = "[LLCLIENTVIEW]"; /// /// Handles UDP texture download. /// public LLImageManager ImageManager { get; private set; } public JobEngine m_asyncPacketProcess; private readonly LLUDPServer m_udpServer; private readonly LLUDPClient m_udpClient; private readonly UUID m_sessionId; private readonly UUID m_secureSessionId; protected readonly UUID m_agentId; private readonly uint m_circuitCode; private readonly byte[] m_channelVersion = Utils.EmptyBytes; private readonly IGroupsModule m_GroupsModule; // private int m_cachedTextureSerial; private PriorityQueue m_entityUpdates; private PriorityQueue m_entityProps; private Prioritizer m_prioritizer; private bool m_disableFacelights; // needs optimization private HashSet GroupsInView = new HashSet(); #pragma warning disable 0414 private bool m_VelocityInterpolate; #pragma warning restore 0414 private const uint MaxTransferBytesPerPacket = 600; private bool m_SupportObjectAnimations; /// /// Maintain a record of all the objects killed. This allows us to stop an update being sent from the /// thread servicing the m_primFullUpdates queue after a kill. If this happens the object persists as an /// ownerless phantom. /// /// All manipulation of this set has to occur under an m_entityUpdates.SyncRoot lock /// /// protected List m_killRecord; // protected HashSet m_attachmentsSent; private bool m_deliverPackets = true; private bool m_SendLogoutPacketWhenClosing = true; /// /// We retain a single AgentUpdateArgs so that we can constantly reuse it rather than construct a new one for /// every single incoming AgentUpdate. Every client sends 10 AgentUpdate UDP messages per second, even if it /// is doing absolutely nothing. /// /// /// This does mean that agent updates must be processed synchronously, at least for each client, and called methods /// cannot retain a reference to it outside of that method. /// private AgentUpdateArgs m_thisAgentUpdateArgs = new AgentUpdateArgs(); protected Dictionary m_packetHandlers = new Dictionary(); protected Dictionary m_genericPacketHandlers = new Dictionary(); //PauPaw:Local Generic Message handlers protected Scene m_scene; protected string m_firstName; protected string m_lastName; protected Vector3 m_startpos; protected UUID m_activeGroupID; protected string m_activeGroupName = String.Empty; protected ulong m_activeGroupPowers; protected Dictionary m_groupPowers = new Dictionary(); protected int m_terrainCheckerCount; protected uint m_agentFOVCounter; protected IAssetService m_assetService; #endregion Class Members #region Properties public LLUDPClient UDPClient { get { return m_udpClient; } } public LLUDPServer UDPServer { get { return m_udpServer; } } public IPEndPoint RemoteEndPoint { get { return m_udpClient.RemoteEndPoint; } } public UUID SecureSessionId { get { return m_secureSessionId; } } public IScene Scene { get { return m_scene; } } public UUID SessionId { get { return m_sessionId; } } public Vector3 StartPos { get { return m_startpos; } set { m_startpos = value; } } public bool DeliverPackets { get { return m_deliverPackets; } set { m_deliverPackets = value; m_udpClient.m_deliverPackets = value; } } public UUID AgentId { get { return m_agentId; } } public ISceneAgent SceneAgent { get; set; } public UUID ActiveGroupId { get { return m_activeGroupID; } set { m_activeGroupID = value; } } public string ActiveGroupName { get { return m_activeGroupName; } set { m_activeGroupName = value; } } public ulong ActiveGroupPowers { get { return m_activeGroupPowers; } set { m_activeGroupPowers = value; } } public bool IsGroupMember(UUID groupID) { return m_groupPowers.ContainsKey(groupID); } public int PingTimeMS { get { if (UDPClient != null) return UDPClient.PingTimeMS; return 0; } } /// /// Entity update queues /// public PriorityQueue EntityUpdateQueue { get { return m_entityUpdates; } } /// /// First name of the agent/avatar represented by the client /// public string FirstName { get { return m_firstName; } } /// /// Last name of the agent/avatar represented by the client /// public string LastName { get { return m_lastName; } } /// /// Full name of the client (first name and last name) /// public string Name { get { return FirstName + " " + LastName; } } public uint CircuitCode { get { return m_circuitCode; } } protected int m_animationSequenceNumber = (int)(Util.GetTimeStampTicks() & 0x5fffafL); public int NextAnimationSequenceNumber { get { int ret = Interlocked.Increment(ref m_animationSequenceNumber); if (ret <= 0) { m_animationSequenceNumber = (int)(Util.GetTimeStampTicks() & 0xafff5fL); ret = Interlocked.Increment(ref m_animationSequenceNumber); } return ret; } set { m_animationSequenceNumber = value; } } /// /// As well as it's function in IClientAPI, in LLClientView we are locking on this property in order to /// prevent race conditions by different threads calling Close(). /// public bool IsActive { get; set; } /// /// Used to synchronise threads when client is being closed. /// public Object CloseSyncLock { get; private set; } public bool IsLoggingOut { get; set; } public bool DisableFacelights { get { return m_disableFacelights; } set { m_disableFacelights = value; } } public List SelectedObjects {get; private set;} public bool SendLogoutPacketWhenClosing { set { m_SendLogoutPacketWhenClosing = value; } } #endregion Properties // ~LLClientView() // { // m_log.DebugFormat("{0} Destructor called for {1}, circuit code {2}", LogHeader, Name, CircuitCode); // } /// /// Constructor /// public LLClientView(Scene scene, LLUDPServer udpServer, LLUDPClient udpClient, AuthenticateResponse sessionInfo, UUID agentId, UUID sessionId, uint circuitCode) { // DebugPacketLevel = 1; CloseSyncLock = new Object(); SelectedObjects = new List(); RegisterInterface(this); RegisterInterface(this); RegisterInterface(this); m_scene = scene; int pcap = 512; if(pcap > m_scene.Entities.Count) pcap = m_scene.Entities.Count; m_entityUpdates = new PriorityQueue(pcap); m_entityProps = new PriorityQueue(pcap); m_killRecord = new List(); // m_attachmentsSent = new HashSet(); m_assetService = m_scene.RequestModuleInterface(); m_GroupsModule = scene.RequestModuleInterface(); ImageManager = new LLImageManager(this, m_assetService, Scene.RequestModuleInterface()); m_channelVersion = Util.StringToBytes256(scene.GetSimulatorVersion()); m_agentId = agentId; m_sessionId = sessionId; m_secureSessionId = sessionInfo.LoginInfo.SecureSession; m_circuitCode = circuitCode; m_firstName = sessionInfo.LoginInfo.First; m_lastName = sessionInfo.LoginInfo.Last; m_startpos = sessionInfo.LoginInfo.StartPos; m_udpServer = udpServer; m_udpClient = udpClient; m_udpClient.OnQueueEmpty += HandleQueueEmpty; m_udpClient.HasUpdates += HandleHasUpdates; m_udpClient.OnPacketStats += PopulateStats; m_prioritizer = new Prioritizer(m_scene); RegisterLocalPacketHandlers(); string name = string.Format("AsyncInUDP-{0}",m_agentId.ToString()); m_asyncPacketProcess = new JobEngine(name, name, 10000); IsActive = true; } #region Client Methods /// /// Close down the client view /// public void Close() { Close(true, false); } public void Close(bool sendStop, bool force) { // We lock here to prevent race conditions between two threads calling close simultaneously (e.g. // a simultaneous relog just as a client is being closed out due to no packet ack from the old connection. lock (CloseSyncLock) { // We still perform a force close inside the sync lock since this is intended to attempt close where // there is some unidentified connection problem, not where we have issues due to deadlock if (!IsActive && !force) { m_log.DebugFormat( "{0} Not attempting to close inactive client {1} in {2} since force flag is not set", LogHeader, Name, m_scene.Name); return; } IsActive = false; CloseWithoutChecks(sendStop); } } /// /// Closes down the client view without first checking whether it is active. /// /// /// This exists because LLUDPServer has to set IsActive = false in earlier synchronous code before calling /// CloseWithoutIsActiveCheck asynchronously. /// /// Callers must lock ClosingSyncLock before calling. /// public void CloseWithoutChecks(bool sendStop) { m_log.DebugFormat( "[CLIENT]: Close has been called for {0} attached to scene {1}", Name, m_scene.RegionInfo.RegionName); if (sendStop) { // Send the STOP packet DisableSimulatorPacket disable = (DisableSimulatorPacket)PacketPool.Instance.GetPacket(PacketType.DisableSimulator); OutPacket(disable, ThrottleOutPacketType.Unknown); } // Fire the callback for this connection closing if (OnConnectionClosed != null) OnConnectionClosed(this); m_asyncPacketProcess.Stop(); // Flush all of the packets out of the UDP server for this client if (m_udpServer != null) m_udpServer.Flush(m_udpClient); // Remove ourselves from the scene m_scene.RemoveClient(AgentId, true); SceneAgent = null; // We can't reach into other scenes and close the connection // We need to do this over grid communications //m_scene.CloseAllAgents(CircuitCode); // Disable UDP handling for this client m_udpClient.OnQueueEmpty -= HandleQueueEmpty; m_udpClient.HasUpdates -= HandleHasUpdates; m_udpClient.OnPacketStats -= PopulateStats; m_udpClient.Shutdown(); // Shutdown the image manager ImageManager.Close(); ImageManager = null; m_entityUpdates.Close(); m_entityProps.Close(); m_killRecord.Clear(); GroupsInView.Clear(); if(m_scene.GetNumberOfClients() == 0) { GCSettings.LargeObjectHeapCompactionMode = GCLargeObjectHeapCompactionMode.CompactOnce; GC.Collect(); GC.WaitForPendingFinalizers(); GC.Collect(); GCSettings.LargeObjectHeapCompactionMode = GCLargeObjectHeapCompactionMode.Default; } } public void Kick(string message) { if (!SceneAgent.IsChildAgent) { KickUserPacket kupack = (KickUserPacket)PacketPool.Instance.GetPacket(PacketType.KickUser); kupack.UserInfo.AgentID = AgentId; kupack.UserInfo.SessionID = SessionId; kupack.TargetBlock.TargetIP = 0; kupack.TargetBlock.TargetPort = 0; kupack.UserInfo.Reason = Util.StringToBytes256(message); OutPacket(kupack, ThrottleOutPacketType.Task); // You must sleep here or users get no message! Thread.Sleep(500); } } public void Stop() { } #endregion Client Methods #region Packet Handling public void PopulateStats(int inPackets, int outPackets, int unAckedBytes) { NetworkStats handlerNetworkStatsUpdate = OnNetworkStatsUpdate; if (handlerNetworkStatsUpdate != null) { handlerNetworkStatsUpdate(inPackets, outPackets, unAckedBytes); } } /// /// Add a handler for the given packet type. /// /// /// The packet is handled on its own thread. If packets must be handled in the order in which they /// are received then please use the synchronous version of this method. /// /// /// /// true if the handler was added. This is currently always the case. public bool AddLocalPacketHandler(PacketType packetType, PacketMethod handler) { return AddLocalPacketHandler(packetType, handler, true); } /// /// Add a handler for the given packet type. /// /// /// /// /// If true, when the packet is received it is handled on its own thread rather than on the main inward bound /// packet handler thread. This vastly increases respnosiveness but some packets need to be handled /// synchronously. /// /// true if the handler was added. This is currently always the case. public bool AddLocalPacketHandler(PacketType packetType, PacketMethod handler, bool doAsync) { bool result = false; lock (m_packetHandlers) { if (!m_packetHandlers.ContainsKey(packetType)) { m_packetHandlers.Add( packetType, new PacketProcessor() { method = handler, Async = doAsync}); result = true; } } return result; } public bool AddGenericPacketHandler(string MethodName, GenericMessage handler) { MethodName = MethodName.ToLower().Trim(); bool result = false; lock (m_genericPacketHandlers) { if (!m_genericPacketHandlers.ContainsKey(MethodName)) { m_genericPacketHandlers.Add(MethodName, handler); result = true; } } return result; } /// /// Try to process a packet using registered packet handlers /// /// /// True if a handler was found which successfully processed the packet. protected bool ProcessPacketMethod(Packet packet) { bool result = false; PacketProcessor pprocessor; if (m_packetHandlers.TryGetValue(packet.Type, out pprocessor)) { //there is a local handler for this packet type if (pprocessor.Async) { object obj = new AsyncPacketProcess(this, pprocessor.method, packet); m_asyncPacketProcess.QueueJob(packet.Type.ToString(), () => ProcessSpecificPacketAsync(obj)); result = true; } else { result = pprocessor.method(this, packet); } } return result; } public void ProcessSpecificPacketAsync(object state) { AsyncPacketProcess packetObject = (AsyncPacketProcess)state; try { packetObject.result = packetObject.Method(packetObject.ClientView, packetObject.Pack); } catch (Exception e) { // Make sure that we see any exception caused by the asynchronous operation. m_log.Error( string.Format( "[LLCLIENTVIEW]: Caught exception while processing {0} for {1} ", packetObject.Pack, Name), e); } } #endregion Packet Handling # region Setup public virtual void Start() { m_asyncPacketProcess.Start(); m_scene.AddNewAgent(this, PresenceType.User); // RefreshGroupMembership(); } # endregion public void ActivateGesture(UUID assetId, UUID gestureId) { } public void DeactivateGesture(UUID assetId, UUID gestureId) { } // Sound public void SoundTrigger(UUID soundId, UUID owerid, UUID Objectid, UUID ParentId, float Gain, Vector3 Position, UInt64 Handle) { } #region Scene/Avatar to Client public void SendRegionHandshake(RegionInfo regionInfo, RegionHandshakeArgs args) { RegionHandshakePacket handshake = (RegionHandshakePacket)PacketPool.Instance.GetPacket(PacketType.RegionHandshake); handshake.RegionInfo = new RegionHandshakePacket.RegionInfoBlock(); handshake.RegionInfo.BillableFactor = args.billableFactor; handshake.RegionInfo.IsEstateManager = args.isEstateManager; handshake.RegionInfo.TerrainHeightRange00 = args.terrainHeightRange0; handshake.RegionInfo.TerrainHeightRange01 = args.terrainHeightRange1; handshake.RegionInfo.TerrainHeightRange10 = args.terrainHeightRange2; handshake.RegionInfo.TerrainHeightRange11 = args.terrainHeightRange3; handshake.RegionInfo.TerrainStartHeight00 = args.terrainStartHeight0; handshake.RegionInfo.TerrainStartHeight01 = args.terrainStartHeight1; handshake.RegionInfo.TerrainStartHeight10 = args.terrainStartHeight2; handshake.RegionInfo.TerrainStartHeight11 = args.terrainStartHeight3; handshake.RegionInfo.SimAccess = args.simAccess; handshake.RegionInfo.WaterHeight = args.waterHeight; handshake.RegionInfo.RegionFlags = args.regionFlags; handshake.RegionInfo.SimName = Util.StringToBytes256(args.regionName); handshake.RegionInfo.SimOwner = args.SimOwner; handshake.RegionInfo.TerrainBase0 = args.terrainBase0; handshake.RegionInfo.TerrainBase1 = args.terrainBase1; handshake.RegionInfo.TerrainBase2 = args.terrainBase2; handshake.RegionInfo.TerrainBase3 = args.terrainBase3; handshake.RegionInfo.TerrainDetail0 = args.terrainDetail0; handshake.RegionInfo.TerrainDetail1 = args.terrainDetail1; handshake.RegionInfo.TerrainDetail2 = args.terrainDetail2; handshake.RegionInfo.TerrainDetail3 = args.terrainDetail3; handshake.RegionInfo.CacheID = UUID.Random(); //I guess this is for the client to remember an old setting? handshake.RegionInfo2 = new RegionHandshakePacket.RegionInfo2Block(); handshake.RegionInfo2.RegionID = regionInfo.RegionID; handshake.RegionInfo3 = new RegionHandshakePacket.RegionInfo3Block(); handshake.RegionInfo3.CPUClassID = 9; handshake.RegionInfo3.CPURatio = 1; handshake.RegionInfo3.ColoName = Utils.EmptyBytes; handshake.RegionInfo3.ProductName = Util.StringToBytes256(regionInfo.RegionType); handshake.RegionInfo3.ProductSKU = Utils.EmptyBytes; handshake.RegionInfo4 = new RegionHandshakePacket.RegionInfo4Block[1]; handshake.RegionInfo4[0] = new RegionHandshakePacket.RegionInfo4Block(); handshake.RegionInfo4[0].RegionFlagsExtended = args.regionFlags; handshake.RegionInfo4[0].RegionProtocols = 0; // 1 here would indicate that SSB is supported OutPacket(handshake, ThrottleOutPacketType.Unknown); } public void MoveAgentIntoRegion(RegionInfo regInfo, Vector3 pos, Vector3 look) { m_thisAgentUpdateArgs.CameraAtAxis.X = float.MinValue; m_thisAgentUpdateArgs.lastUpdateTS = 0; m_thisAgentUpdateArgs.ControlFlags = 0; AgentMovementCompletePacket mov = (AgentMovementCompletePacket)PacketPool.Instance.GetPacket(PacketType.AgentMovementComplete); mov.SimData.ChannelVersion = m_channelVersion; mov.AgentData.SessionID = m_sessionId; mov.AgentData.AgentID = AgentId; mov.Data.RegionHandle = regInfo.RegionHandle; mov.Data.Timestamp = (uint)Util.UnixTimeSinceEpoch(); if ((pos.X == 0) && (pos.Y == 0) && (pos.Z == 0)) { mov.Data.Position = m_startpos; } else { mov.Data.Position = pos; } mov.Data.LookAt = look; // Hack to get this out immediately and skip the throttles OutPacket(mov, ThrottleOutPacketType.Unknown); } public void SendChatMessage( string message, byte type, Vector3 fromPos, string fromName, UUID fromAgentID, UUID ownerID, byte source, byte audible) { ChatFromSimulatorPacket reply = (ChatFromSimulatorPacket)PacketPool.Instance.GetPacket(PacketType.ChatFromSimulator); reply.ChatData.Audible = audible; reply.ChatData.Message = Util.StringToBytes1024(message); reply.ChatData.ChatType = type; reply.ChatData.SourceType = source; reply.ChatData.Position = fromPos; reply.ChatData.FromName = Util.StringToBytes256(fromName); reply.ChatData.OwnerID = ownerID; reply.ChatData.SourceID = fromAgentID; OutPacket(reply, ThrottleOutPacketType.Unknown); } /// /// Send an instant message to this client /// // public void SendInstantMessage(GridInstantMessage im) { if (((Scene)(m_scene)).Permissions.CanInstantMessage(new UUID(im.fromAgentID), new UUID(im.toAgentID))) { ImprovedInstantMessagePacket msg = (ImprovedInstantMessagePacket)PacketPool.Instance.GetPacket(PacketType.ImprovedInstantMessage); msg.AgentData.AgentID = new UUID(im.fromAgentID); msg.AgentData.SessionID = UUID.Zero; msg.MessageBlock.FromAgentName = Util.StringToBytes256(im.fromAgentName); msg.MessageBlock.Dialog = im.dialog; msg.MessageBlock.FromGroup = im.fromGroup; // this is odd if (im.imSessionID == UUID.Zero.Guid) msg.MessageBlock.ID = new UUID(im.fromAgentID) ^ new UUID(im.toAgentID); else msg.MessageBlock.ID = new UUID(im.imSessionID); msg.MessageBlock.Offline = im.offline; msg.MessageBlock.ParentEstateID = im.ParentEstateID; msg.MessageBlock.Position = im.Position; msg.MessageBlock.RegionID = new UUID(im.RegionID); msg.MessageBlock.Timestamp = im.timestamp; msg.MessageBlock.ToAgentID = new UUID(im.toAgentID); msg.MessageBlock.Message = Util.StringToBytes1024(im.message); msg.MessageBlock.BinaryBucket = im.binaryBucket; OutPacket(msg, ThrottleOutPacketType.Task); } } public void SendGenericMessage(string method, UUID invoice, List message) { GenericMessagePacket gmp = new GenericMessagePacket(); gmp.AgentData.AgentID = AgentId; gmp.AgentData.SessionID = m_sessionId; gmp.AgentData.TransactionID = invoice; gmp.MethodData.Method = Util.StringToBytes256(method); gmp.ParamList = new GenericMessagePacket.ParamListBlock[message.Count]; int i = 0; foreach (string val in message) { gmp.ParamList[i] = new GenericMessagePacket.ParamListBlock(); gmp.ParamList[i++].Parameter = Util.StringToBytes256(val); } OutPacket(gmp, ThrottleOutPacketType.Task); } public void SendGenericMessage(string method, UUID invoice, List message) { GenericMessagePacket gmp = new GenericMessagePacket(); gmp.AgentData.AgentID = AgentId; gmp.AgentData.SessionID = m_sessionId; gmp.AgentData.TransactionID = invoice; gmp.MethodData.Method = Util.StringToBytes256(method); gmp.ParamList = new GenericMessagePacket.ParamListBlock[message.Count]; int i = 0; foreach (byte[] val in message) { gmp.ParamList[i] = new GenericMessagePacket.ParamListBlock(); gmp.ParamList[i++].Parameter = val; } OutPacket(gmp, ThrottleOutPacketType.Task); } public void SendGroupActiveProposals(UUID groupID, UUID transactionID, GroupActiveProposals[] Proposals) { int i = 0; foreach (GroupActiveProposals Proposal in Proposals) { GroupActiveProposalItemReplyPacket GAPIRP = new GroupActiveProposalItemReplyPacket(); GAPIRP.AgentData.AgentID = AgentId; GAPIRP.AgentData.GroupID = groupID; GAPIRP.TransactionData.TransactionID = transactionID; GAPIRP.TransactionData.TotalNumItems = ((uint)i+1); GroupActiveProposalItemReplyPacket.ProposalDataBlock ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock(); GAPIRP.ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock[1]; ProposalData.VoteCast = Utils.StringToBytes("false"); ProposalData.VoteID = new UUID(Proposal.VoteID); ProposalData.VoteInitiator = new UUID(Proposal.VoteInitiator); ProposalData.Majority = (float)Convert.ToInt32(Proposal.Majority); ProposalData.Quorum = Convert.ToInt32(Proposal.Quorum); ProposalData.TerseDateID = Utils.StringToBytes(Proposal.TerseDateID); ProposalData.StartDateTime = Utils.StringToBytes(Proposal.StartDateTime); ProposalData.EndDateTime = Utils.StringToBytes(Proposal.EndDateTime); ProposalData.ProposalText = Utils.StringToBytes(Proposal.ProposalText); ProposalData.AlreadyVoted = false; GAPIRP.ProposalData[i] = ProposalData; OutPacket(GAPIRP, ThrottleOutPacketType.Task); i++; } if (Proposals.Length == 0) { GroupActiveProposalItemReplyPacket GAPIRP = new GroupActiveProposalItemReplyPacket(); GAPIRP.AgentData.AgentID = AgentId; GAPIRP.AgentData.GroupID = groupID; GAPIRP.TransactionData.TransactionID = transactionID; GAPIRP.TransactionData.TotalNumItems = 1; GroupActiveProposalItemReplyPacket.ProposalDataBlock ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock(); GAPIRP.ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock[1]; ProposalData.VoteCast = Utils.StringToBytes("false"); ProposalData.VoteID = UUID.Zero; ProposalData.VoteInitiator = UUID.Zero; ProposalData.Majority = 0; ProposalData.Quorum = 0; ProposalData.TerseDateID = Utils.StringToBytes(""); ProposalData.StartDateTime = Utils.StringToBytes(""); ProposalData.EndDateTime = Utils.StringToBytes(""); ProposalData.ProposalText = Utils.StringToBytes(""); ProposalData.AlreadyVoted = false; GAPIRP.ProposalData[0] = ProposalData; OutPacket(GAPIRP, ThrottleOutPacketType.Task); } } public void SendGroupVoteHistory(UUID groupID, UUID transactionID, GroupVoteHistory[] Votes) { int i = 0; foreach (GroupVoteHistory Vote in Votes) { GroupVoteHistoryItemReplyPacket GVHIRP = new GroupVoteHistoryItemReplyPacket(); GVHIRP.AgentData.AgentID = AgentId; GVHIRP.AgentData.GroupID = groupID; GVHIRP.TransactionData.TransactionID = transactionID; GVHIRP.TransactionData.TotalNumItems = ((uint)i+1); GVHIRP.HistoryItemData.VoteID = new UUID(Vote.VoteID); GVHIRP.HistoryItemData.VoteInitiator = new UUID(Vote.VoteInitiator); GVHIRP.HistoryItemData.Majority = (float)Convert.ToInt32(Vote.Majority); GVHIRP.HistoryItemData.Quorum = Convert.ToInt32(Vote.Quorum); GVHIRP.HistoryItemData.TerseDateID = Utils.StringToBytes(Vote.TerseDateID); GVHIRP.HistoryItemData.StartDateTime = Utils.StringToBytes(Vote.StartDateTime); GVHIRP.HistoryItemData.EndDateTime = Utils.StringToBytes(Vote.EndDateTime); GVHIRP.HistoryItemData.VoteType = Utils.StringToBytes(Vote.VoteType); GVHIRP.HistoryItemData.VoteResult = Utils.StringToBytes(Vote.VoteResult); GVHIRP.HistoryItemData.ProposalText = Utils.StringToBytes(Vote.ProposalText); GroupVoteHistoryItemReplyPacket.VoteItemBlock VoteItem = new GroupVoteHistoryItemReplyPacket.VoteItemBlock(); GVHIRP.VoteItem = new GroupVoteHistoryItemReplyPacket.VoteItemBlock[1]; VoteItem.CandidateID = UUID.Zero; VoteItem.NumVotes = 0; //TODO: FIX THIS!!! VoteItem.VoteCast = Utils.StringToBytes("Yes"); GVHIRP.VoteItem[i] = VoteItem; OutPacket(GVHIRP, ThrottleOutPacketType.Task); i++; } if (Votes.Length == 0) { GroupVoteHistoryItemReplyPacket GVHIRP = new GroupVoteHistoryItemReplyPacket(); GVHIRP.AgentData.AgentID = AgentId; GVHIRP.AgentData.GroupID = groupID; GVHIRP.TransactionData.TransactionID = transactionID; GVHIRP.TransactionData.TotalNumItems = 0; GVHIRP.HistoryItemData.VoteID = UUID.Zero; GVHIRP.HistoryItemData.VoteInitiator = UUID.Zero; GVHIRP.HistoryItemData.Majority = 0; GVHIRP.HistoryItemData.Quorum = 0; GVHIRP.HistoryItemData.TerseDateID = Utils.StringToBytes(""); GVHIRP.HistoryItemData.StartDateTime = Utils.StringToBytes(""); GVHIRP.HistoryItemData.EndDateTime = Utils.StringToBytes(""); GVHIRP.HistoryItemData.VoteType = Utils.StringToBytes(""); GVHIRP.HistoryItemData.VoteResult = Utils.StringToBytes(""); GVHIRP.HistoryItemData.ProposalText = Utils.StringToBytes(""); GroupVoteHistoryItemReplyPacket.VoteItemBlock VoteItem = new GroupVoteHistoryItemReplyPacket.VoteItemBlock(); GVHIRP.VoteItem = new GroupVoteHistoryItemReplyPacket.VoteItemBlock[1]; VoteItem.CandidateID = UUID.Zero; VoteItem.NumVotes = 0; //TODO: FIX THIS!!! VoteItem.VoteCast = Utils.StringToBytes("No"); GVHIRP.VoteItem[0] = VoteItem; OutPacket(GVHIRP, ThrottleOutPacketType.Task); } } public void SendGroupAccountingDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID, int amt) { GroupAccountDetailsReplyPacket GADRP = new GroupAccountDetailsReplyPacket(); GADRP.AgentData = new GroupAccountDetailsReplyPacket.AgentDataBlock(); GADRP.AgentData.AgentID = sender.AgentId; GADRP.AgentData.GroupID = groupID; GADRP.HistoryData = new GroupAccountDetailsReplyPacket.HistoryDataBlock[1]; GroupAccountDetailsReplyPacket.HistoryDataBlock History = new GroupAccountDetailsReplyPacket.HistoryDataBlock(); GADRP.MoneyData = new GroupAccountDetailsReplyPacket.MoneyDataBlock(); GADRP.MoneyData.CurrentInterval = 0; GADRP.MoneyData.IntervalDays = 7; GADRP.MoneyData.RequestID = transactionID; GADRP.MoneyData.StartDate = Utils.StringToBytes(DateTime.Today.ToString()); History.Amount = amt; History.Description = Utils.StringToBytes(""); GADRP.HistoryData[0] = History; OutPacket(GADRP, ThrottleOutPacketType.Task); } public void SendGroupAccountingSummary(IClientAPI sender,UUID groupID, uint moneyAmt, int totalTier, int usedTier) { GroupAccountSummaryReplyPacket GASRP = (GroupAccountSummaryReplyPacket)PacketPool.Instance.GetPacket( PacketType.GroupAccountSummaryReply); GASRP.AgentData = new GroupAccountSummaryReplyPacket.AgentDataBlock(); GASRP.AgentData.AgentID = sender.AgentId; GASRP.AgentData.GroupID = groupID; GASRP.MoneyData = new GroupAccountSummaryReplyPacket.MoneyDataBlock(); GASRP.MoneyData.Balance = (int)moneyAmt; GASRP.MoneyData.TotalCredits = totalTier; GASRP.MoneyData.TotalDebits = usedTier; GASRP.MoneyData.StartDate = new byte[1]; GASRP.MoneyData.CurrentInterval = 1; GASRP.MoneyData.GroupTaxCurrent = 0; GASRP.MoneyData.GroupTaxEstimate = 0; GASRP.MoneyData.IntervalDays = 0; GASRP.MoneyData.LandTaxCurrent = 0; GASRP.MoneyData.LandTaxEstimate = 0; GASRP.MoneyData.LastTaxDate = new byte[1]; GASRP.MoneyData.LightTaxCurrent = 0; GASRP.MoneyData.TaxDate = new byte[1]; GASRP.MoneyData.RequestID = sender.AgentId; GASRP.MoneyData.ParcelDirFeeEstimate = 0; GASRP.MoneyData.ParcelDirFeeCurrent = 0; GASRP.MoneyData.ObjectTaxEstimate = 0; GASRP.MoneyData.NonExemptMembers = 0; GASRP.MoneyData.ObjectTaxCurrent = 0; GASRP.MoneyData.LightTaxEstimate = 0; OutPacket(GASRP, ThrottleOutPacketType.Task); } public void SendGroupTransactionsSummaryDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID, int amt) { GroupAccountTransactionsReplyPacket GATRP = (GroupAccountTransactionsReplyPacket)PacketPool.Instance.GetPacket( PacketType.GroupAccountTransactionsReply); GATRP.AgentData = new GroupAccountTransactionsReplyPacket.AgentDataBlock(); GATRP.AgentData.AgentID = sender.AgentId; GATRP.AgentData.GroupID = groupID; GATRP.MoneyData = new GroupAccountTransactionsReplyPacket.MoneyDataBlock(); GATRP.MoneyData.CurrentInterval = 0; GATRP.MoneyData.IntervalDays = 7; GATRP.MoneyData.RequestID = transactionID; GATRP.MoneyData.StartDate = Utils.StringToBytes(DateTime.Today.ToString()); GATRP.HistoryData = new GroupAccountTransactionsReplyPacket.HistoryDataBlock[1]; GroupAccountTransactionsReplyPacket.HistoryDataBlock History = new GroupAccountTransactionsReplyPacket.HistoryDataBlock(); History.Amount = 0; History.Item = Utils.StringToBytes(""); History.Time = Utils.StringToBytes(""); History.Type = 0; History.User = Utils.StringToBytes(""); GATRP.HistoryData[0] = History; OutPacket(GATRP, ThrottleOutPacketType.Task); } public virtual bool CanSendLayerData() { int n = m_udpClient.GetPacketsQueuedCount(ThrottleOutPacketType.Land); if ( n > 128) return false; return true; } /// /// Send the region heightmap to the client /// This method is only called when not doing intellegent terrain patch sending and /// is only called when the scene presence is initially created and sends all of the /// region's patches to the client. /// /// heightmap public virtual void SendLayerData() { Util.FireAndForget(DoSendLayerData, null, "LLClientView.DoSendLayerData"); } /// /// Send terrain layer information to the client. /// /// private void DoSendLayerData(object o) { TerrainData map = m_scene.Heightmap.GetTerrainData(); try { // Send LayerData in a spiral pattern. Fun! SendLayerTopRight(0, 0, map.SizeX / Constants.TerrainPatchSize - 1, map.SizeY / Constants.TerrainPatchSize - 1); } catch (Exception e) { m_log.Error("[CLIENT]: SendLayerData() Failed with exception: " + e.Message, e); } } private void SendLayerTopRight(int x1, int y1, int x2, int y2) { int[] p = new int[2]; // Row p[1] = y1; for (int i = x1; i <= x2; ++i) { p[0] = i; SendLayerData(p); } // Column p[0] = x2; for (int j = y1 + 1; j <= y2; ++j) { p[1] = j; SendLayerData(p); } if (x2 - x1 > 0 && y2 - y1 > 0) SendLayerBottomLeft(x1, y1 + 1, x2 - 1, y2); } void SendLayerBottomLeft(int x1, int y1, int x2, int y2) { int[] p = new int[2]; // Row in reverse p[1] = y2; for (int i = x2; i >= x1; --i) { p[0] = i; SendLayerData(p); } // Column in reverse p[0] = x1; for (int j = y2 - 1; j >= y1; --j) { p[1] = j; SendLayerData(p); } if (x2 - x1 > 0 && y2 - y1 > 0) SendLayerTopRight(x1 + 1, y1, x2, y2 - 1); } static private readonly byte[] TerrainPacketHeader = new byte[] { Helpers.MSG_RELIABLE, // zero code is not as spec 0, 0, 0, 0, // sequence number 0, // extra 11, // ID (high frequency) }; private const int END_OF_PATCHES = 97; private const int STRIDE = 264; public void SendLayerData(int[] map) { if(map == null) return; try { TerrainData terrData = m_scene.Heightmap.GetTerrainData(); byte landPacketType = (terrData.SizeX > Constants.RegionSize || terrData.SizeY > Constants.RegionSize) ? (byte)TerrainPatch.LayerType.LandExtended : (byte)TerrainPatch.LayerType.Land; int numberPatchs = map.Length / 2; UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; Buffer.BlockCopy(TerrainPacketHeader, 0, data, 0, 7); data[7] = landPacketType; //data[8] and data[9] == datablock size to fill later data[10] = 0; // BitPack needs this on reused packets // start data BitPack bitpack = new BitPack(data, 10); bitpack.PackBits(STRIDE, 16); bitpack.PackBitsFromByte(16); bitpack.PackBitsFromByte(landPacketType); int s; int datasize = 0; for (int i = 0; i < numberPatchs; i++) { s = 2 * i; OpenSimTerrainCompressor.CreatePatchFromTerrainData(bitpack, terrData, map[s], map[s + 1]); if (bitpack.BytePos > 950 && i != numberPatchs - 1) { //finish this packet bitpack.PackBitsFromByte(END_OF_PATCHES); // fix the datablock lenght datasize = bitpack.BytePos - 9; data[8] = (byte)datasize; data[9] = (byte)(datasize >> 8); buf.DataLength = bitpack.BytePos + 1; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); // start another buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); data = buf.Data; Buffer.BlockCopy(TerrainPacketHeader, 0, data, 0, 7); data[7] = landPacketType; //data[8] and data[9] == datablock size to fill later data[10] = 0; // BitPack needs this // start data bitpack = new BitPack(data, 10); bitpack.PackBits(STRIDE, 16); bitpack.PackBitsFromByte(16); bitpack.PackBitsFromByte(landPacketType); } } bitpack.PackBitsFromByte(END_OF_PATCHES); datasize = bitpack.BytePos - 9; data[8] = (byte)datasize; data[9] = (byte)(datasize >> 8); buf.DataLength = bitpack.BytePos + 1; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); } catch (Exception e) { m_log.Error("[CLIENT]: SendLayerData() Failed with exception: " + e.Message, e); } } private void DebugSendingPatches(string pWho, int[] pX, int[] pY) { if (m_log.IsDebugEnabled) { int numPatches = pX.Length; string Xs = ""; string Ys = ""; for (int pp = 0; pp < numPatches; pp++) { Xs += String.Format("{0}", (int)pX[pp]) + ","; Ys += String.Format("{0}", (int)pY[pp]) + ","; } m_log.DebugFormat("{0} {1}: numPatches={2}, X={3}, Y={4}", LogHeader, pWho, numPatches, Xs, Ys); } } // wind caching private static Dictionary lastWindVersion = new Dictionary(); private static Dictionary> lastWindPackets = new Dictionary>(); /// /// Send the wind matrix to the client /// /// 16x16 array of wind speeds public virtual void SendWindData(int version, Vector2[] windSpeeds) { // Vector2[] windSpeeds = (Vector2[])o; ulong handle = this.Scene.RegionInfo.RegionHandle; bool isNewData; lock(lastWindPackets) { if(!lastWindVersion.ContainsKey(handle) || !lastWindPackets.ContainsKey(handle)) { lastWindVersion[handle] = 0; lastWindPackets[handle] = new List(); isNewData = true; } else isNewData = lastWindVersion[handle] != version; } if(isNewData) { TerrainPatch[] patches = new TerrainPatch[2]; patches[0] = new TerrainPatch { Data = new float[16 * 16] }; patches[1] = new TerrainPatch { Data = new float[16 * 16] }; for (int x = 0; x < 16 * 16; x++) { patches[0].Data[x] = windSpeeds[x].X; patches[1].Data[x] = windSpeeds[x].Y; } // neither we or viewers have extended wind byte layerType = (byte)TerrainPatch.LayerType.Wind; LayerDataPacket layerpack = OpenSimTerrainCompressor.CreateLayerDataPacketStandardSize( patches, layerType); layerpack.Header.Zerocoded = true; lock(lastWindPackets) { lastWindPackets[handle].Clear(); lastWindPackets[handle].Add(layerpack); lastWindVersion[handle] = version; } } lock(lastWindPackets) foreach(LayerDataPacket pkt in lastWindPackets[handle]) OutPacket(pkt, ThrottleOutPacketType.Wind); } // cloud caching private static Dictionary lastCloudVersion = new Dictionary(); private static Dictionary> lastCloudPackets = new Dictionary>(); /// /// Send the cloud matrix to the client /// /// 16x16 array of cloud densities public virtual void SendCloudData(int version, float[] cloudDensity) { ulong handle = this.Scene.RegionInfo.RegionHandle; bool isNewData; lock(lastWindPackets) { if(!lastCloudVersion.ContainsKey(handle) || !lastCloudPackets.ContainsKey(handle)) { lastCloudVersion[handle] = 0; lastCloudPackets[handle] = new List(); isNewData = true; } else isNewData = lastCloudVersion[handle] != version; } if(isNewData) { TerrainPatch[] patches = new TerrainPatch[1]; patches[0] = new TerrainPatch(); patches[0].Data = new float[16 * 16]; for (int y = 0; y < 16; y++) { for (int x = 0; x < 16; x++) { patches[0].Data[y * 16 + x] = cloudDensity[y * 16 + x]; } } // neither we or viewers have extended clouds byte layerType = (byte)TerrainPatch.LayerType.Cloud; LayerDataPacket layerpack = OpenSimTerrainCompressor.CreateLayerDataPacketStandardSize( patches, layerType); layerpack.Header.Zerocoded = true; lock(lastCloudPackets) { lastCloudPackets[handle].Clear(); lastCloudPackets[handle].Add(layerpack); lastCloudVersion[handle] = version; } } lock(lastCloudPackets) foreach(LayerDataPacket pkt in lastCloudPackets[handle]) OutPacket(pkt, ThrottleOutPacketType.Cloud); } /// /// Tell the client that the given neighbour region is ready to receive a child agent. /// public virtual void InformClientOfNeighbour(ulong neighbourHandle, IPEndPoint neighbourEndPoint) { IPAddress neighbourIP = neighbourEndPoint.Address; ushort neighbourPort = (ushort)neighbourEndPoint.Port; EnableSimulatorPacket enablesimpacket = (EnableSimulatorPacket)PacketPool.Instance.GetPacket(PacketType.EnableSimulator); // TODO: don't create new blocks if recycling an old packet enablesimpacket.SimulatorInfo = new EnableSimulatorPacket.SimulatorInfoBlock(); enablesimpacket.SimulatorInfo.Handle = neighbourHandle; byte[] byteIP = neighbourIP.GetAddressBytes(); enablesimpacket.SimulatorInfo.IP = (uint)byteIP[3] << 24; enablesimpacket.SimulatorInfo.IP += (uint)byteIP[2] << 16; enablesimpacket.SimulatorInfo.IP += (uint)byteIP[1] << 8; enablesimpacket.SimulatorInfo.IP += (uint)byteIP[0]; enablesimpacket.SimulatorInfo.Port = neighbourPort; enablesimpacket.Header.Reliable = true; // ESP's should be reliable. OutPacket(enablesimpacket, ThrottleOutPacketType.Task); } public AgentCircuitData RequestClientInfo() { AgentCircuitData agentData = new AgentCircuitData(); agentData.AgentID = AgentId; agentData.SessionID = m_sessionId; agentData.SecureSessionID = SecureSessionId; agentData.circuitcode = m_circuitCode; agentData.child = false; agentData.firstname = m_firstName; agentData.lastname = m_lastName; ICapabilitiesModule capsModule = m_scene.RequestModuleInterface(); if (capsModule == null) // can happen when shutting down. return agentData; agentData.CapsPath = capsModule.GetCapsPath(m_agentId); agentData.ChildrenCapSeeds = new Dictionary(capsModule.GetChildrenSeeds(m_agentId)); return agentData; } public virtual void CrossRegion(ulong newRegionHandle, Vector3 pos, Vector3 lookAt, IPEndPoint externalIPEndPoint, string capsURL) { Vector3 look = new Vector3(lookAt.X * 10, lookAt.Y * 10, lookAt.Z * 10); //CrossedRegionPacket newSimPack = (CrossedRegionPacket)PacketPool.Instance.GetPacket(PacketType.CrossedRegion); CrossedRegionPacket newSimPack = new CrossedRegionPacket(); // TODO: don't create new blocks if recycling an old packet newSimPack.AgentData = new CrossedRegionPacket.AgentDataBlock(); newSimPack.AgentData.AgentID = AgentId; newSimPack.AgentData.SessionID = m_sessionId; newSimPack.Info = new CrossedRegionPacket.InfoBlock(); newSimPack.Info.Position = pos; newSimPack.Info.LookAt = look; newSimPack.RegionData = new CrossedRegionPacket.RegionDataBlock(); newSimPack.RegionData.RegionHandle = newRegionHandle; byte[] byteIP = externalIPEndPoint.Address.GetAddressBytes(); newSimPack.RegionData.SimIP = (uint)byteIP[3] << 24; newSimPack.RegionData.SimIP += (uint)byteIP[2] << 16; newSimPack.RegionData.SimIP += (uint)byteIP[1] << 8; newSimPack.RegionData.SimIP += (uint)byteIP[0]; newSimPack.RegionData.SimPort = (ushort)externalIPEndPoint.Port; newSimPack.RegionData.SeedCapability = Util.StringToBytes256(capsURL); // Hack to get this out immediately and skip throttles OutPacket(newSimPack, ThrottleOutPacketType.Unknown); } static private readonly byte[] MapBlockItemHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 1, 155 // ID 411 (low frequency bigendian) }; public void SendMapItemReply(mapItemReply[] replies, uint mapitemtype, uint flags) { UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header and agentinfo block Buffer.BlockCopy(MapBlockItemHeader, 0, data, 0, 10); AgentId.ToBytes(data, 10); // 26 Utils.UIntToBytesSafepos(flags, data, 26); // 30 //RequestData block Utils.UIntToBytesSafepos(mapitemtype, data, 30); // 34 int countpos = 34; int pos = 35; int lastpos = 0; int capacity = LLUDPServer.MAXPAYLOAD - pos; int count = 0; mapItemReply mr; for (int k = 0; k < replies.Length; ++k) { lastpos = pos; mr = replies[k]; Utils.UIntToBytesSafepos(mr.x, data, pos); pos += 4; Utils.UIntToBytesSafepos(mr.y, data, pos); pos += 4; mr.id.ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(mr.Extra, data, pos); pos += 4; Utils.IntToBytesSafepos(mr.Extra2, data, pos); pos += 4; byte[] itemName = Util.StringToBytes256(mr.name); data[pos++] = (byte)itemName.Length; if (itemName.Length > 0) Buffer.BlockCopy(itemName, 0, data, pos, itemName.Length); pos += itemName.Length; if (pos < capacity) ++count; else { // prepare next packet UDPPacketBuffer newbuf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(data, 0, newbuf.Data, 0, 34); // copy the block we already did int alreadyDone = pos - lastpos; Buffer.BlockCopy(data, lastpos, newbuf.Data, 35, alreadyDone); // 34 is datablock size // finish current data[countpos] = (byte)count; buf.DataLength = lastpos; // send it m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); buf = newbuf; data = buf.Data; pos = alreadyDone + 35; capacity = LLUDPServer.MAXPAYLOAD - pos; count = 1; } } if (count > 0) { data[countpos] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); } } static private readonly byte[] MapBlockReplyHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 1, 153 // ID 409 (low frequency bigendian) }; public void SendMapBlock(List mapBlocks, uint flags) { ushort[] sizes = new ushort[2 * mapBlocks.Count]; bool needSizes = false; int sizesptr = 0; // check if we will need sizes block and get them aside int count = 0; ushort ut; foreach (MapBlockData md in mapBlocks) { ut = md.SizeX; sizes[count++] = ut; if (ut > 256) needSizes = true; ut = md.SizeY; sizes[count++] = ut; if (ut > 256) needSizes = true; } UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header and agentinfo block Buffer.BlockCopy(MapBlockReplyHeader, 0, data, 0, 10); AgentId.ToBytes(data, 10); // 26 Utils.UIntToBytesSafepos(flags, data, 26); // 30 int countpos = 30; int pos = 31; int lastpos = 0; int capacity = LLUDPServer.MAXPAYLOAD - pos; count = 0; foreach (MapBlockData md in mapBlocks) { lastpos = pos; Utils.UInt16ToBytes(md.X, data, pos); pos += 2; Utils.UInt16ToBytes(md.Y, data, pos); pos += 2; byte[] regionName = Util.StringToBytes256(md.Name); data[pos++] = (byte)regionName.Length; if(regionName.Length > 0) Buffer.BlockCopy(regionName, 0, data, pos, regionName.Length); pos += regionName.Length; data[pos++] = md.Access; Utils.UIntToBytesSafepos(md.RegionFlags, data, pos); pos += 4; data[pos++] = md.WaterHeight; data[pos++] = md.Agents; md.MapImageId.ToBytes(data, pos); pos += 16; if(needSizes) capacity -= 4; // 2 shorts per entry if(pos < capacity) ++count; else { // prepare next packet UDPPacketBuffer newbuf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(data, 0, newbuf.Data, 0, 30); // copy the block we already did int alreadyDone = pos - lastpos; Buffer.BlockCopy(data, lastpos, newbuf.Data, 31, alreadyDone); // 30 is datablock size // finish current data[countpos] = (byte)count; if (needSizes) { data[lastpos++] = (byte)count; while (--count >= 0) { Utils.UInt16ToBytes(sizes[sizesptr++], data, lastpos); lastpos += 2; Utils.UInt16ToBytes(sizes[sizesptr++], data, lastpos); lastpos += 2; } } else data[lastpos++] = 0; buf.DataLength = lastpos; // send it m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); buf = newbuf; data = buf.Data; pos = alreadyDone + 31; capacity = LLUDPServer.MAXPAYLOAD - pos; if (needSizes) capacity -= 4; // 2 shorts per entry count = 1; } } if (count > 0) { data[countpos] = (byte)count; if (needSizes) { data[pos++] = (byte)count; while (--count >= 0) { Utils.UInt16ToBytes(sizes[sizesptr++], data, pos); pos += 2; Utils.UInt16ToBytes(sizes[sizesptr++], data, pos); pos += 2; } } else data[pos++] = 0; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Land, null, false, false); } } public void SendLocalTeleport(Vector3 position, Vector3 lookAt, uint flags) { TeleportLocalPacket tpLocal = (TeleportLocalPacket)PacketPool.Instance.GetPacket(PacketType.TeleportLocal); tpLocal.Info.AgentID = AgentId; tpLocal.Info.TeleportFlags = flags; tpLocal.Info.LocationID = 2; tpLocal.Info.LookAt = lookAt; tpLocal.Info.Position = position; // Hack to get this out immediately and skip throttles OutPacket(tpLocal, ThrottleOutPacketType.Unknown); } public virtual void SendRegionTeleport(ulong regionHandle, byte simAccess, IPEndPoint newRegionEndPoint, uint locationID, uint flags, string capsURL) { //TeleportFinishPacket teleport = (TeleportFinishPacket)PacketPool.Instance.GetPacket(PacketType.TeleportFinish); TeleportFinishPacket teleport = new TeleportFinishPacket(); teleport.Info.AgentID = AgentId; teleport.Info.RegionHandle = regionHandle; teleport.Info.SimAccess = simAccess; teleport.Info.SeedCapability = Util.StringToBytes256(capsURL); IPAddress oIP = newRegionEndPoint.Address; byte[] byteIP = oIP.GetAddressBytes(); uint ip = (uint)byteIP[3] << 24; ip += (uint)byteIP[2] << 16; ip += (uint)byteIP[1] << 8; ip += (uint)byteIP[0]; teleport.Info.SimIP = ip; teleport.Info.SimPort = (ushort)newRegionEndPoint.Port; teleport.Info.LocationID = 4; teleport.Info.TeleportFlags = 1 << 4; // Hack to get this out immediately and skip throttles. OutPacket(teleport, ThrottleOutPacketType.Unknown); } /// /// Inform the client that a teleport attempt has failed /// public void SendTeleportFailed(string reason) { TeleportFailedPacket tpFailed = (TeleportFailedPacket)PacketPool.Instance.GetPacket(PacketType.TeleportFailed); tpFailed.Info.AgentID = AgentId; tpFailed.Info.Reason = Util.StringToBytes256(reason); tpFailed.AlertInfo = new TeleportFailedPacket.AlertInfoBlock[0]; // Hack to get this out immediately and skip throttles OutPacket(tpFailed, ThrottleOutPacketType.Unknown); } /// /// /// public void SendTeleportStart(uint flags) { TeleportStartPacket tpStart = (TeleportStartPacket)PacketPool.Instance.GetPacket(PacketType.TeleportStart); //TeleportStartPacket tpStart = new TeleportStartPacket(); tpStart.Info.TeleportFlags = flags; //16; // Teleport via location // Hack to get this out immediately and skip throttles OutPacket(tpStart, ThrottleOutPacketType.Unknown); } public void SendTeleportProgress(uint flags, string message) { TeleportProgressPacket tpProgress = (TeleportProgressPacket)PacketPool.Instance.GetPacket(PacketType.TeleportProgress); tpProgress.AgentData.AgentID = this.AgentId; tpProgress.Info.TeleportFlags = flags; tpProgress.Info.Message = Util.StringToBytes256(message); // Hack to get this out immediately and skip throttles OutPacket(tpProgress, ThrottleOutPacketType.Unknown); } public void SendMoneyBalance(UUID transaction, bool success, byte[] description, int balance, int transactionType, UUID sourceID, bool sourceIsGroup, UUID destID, bool destIsGroup, int amount, string item) { MoneyBalanceReplyPacket money = (MoneyBalanceReplyPacket)PacketPool.Instance.GetPacket(PacketType.MoneyBalanceReply); money.MoneyData.AgentID = AgentId; money.MoneyData.TransactionID = transaction; money.MoneyData.TransactionSuccess = success; money.MoneyData.Description = description; money.MoneyData.MoneyBalance = balance; money.TransactionInfo.TransactionType = transactionType; money.TransactionInfo.SourceID = sourceID; money.TransactionInfo.IsSourceGroup = sourceIsGroup; money.TransactionInfo.DestID = destID; money.TransactionInfo.IsDestGroup = destIsGroup; money.TransactionInfo.Amount = amount; money.TransactionInfo.ItemDescription = Util.StringToBytes256(item); OutPacket(money, ThrottleOutPacketType.Task); } public void SendPayPrice(UUID objectID, int[] payPrice) { if (payPrice[0] == 0 && payPrice[1] == 0 && payPrice[2] == 0 && payPrice[3] == 0 && payPrice[4] == 0) return; PayPriceReplyPacket payPriceReply = (PayPriceReplyPacket)PacketPool.Instance.GetPacket(PacketType.PayPriceReply); payPriceReply.ObjectData.ObjectID = objectID; payPriceReply.ObjectData.DefaultPayPrice = payPrice[0]; payPriceReply.ButtonData = new PayPriceReplyPacket.ButtonDataBlock[4]; payPriceReply.ButtonData[0] = new PayPriceReplyPacket.ButtonDataBlock(); payPriceReply.ButtonData[0].PayButton = payPrice[1]; payPriceReply.ButtonData[1] = new PayPriceReplyPacket.ButtonDataBlock(); payPriceReply.ButtonData[1].PayButton = payPrice[2]; payPriceReply.ButtonData[2] = new PayPriceReplyPacket.ButtonDataBlock(); payPriceReply.ButtonData[2].PayButton = payPrice[3]; payPriceReply.ButtonData[3] = new PayPriceReplyPacket.ButtonDataBlock(); payPriceReply.ButtonData[3].PayButton = payPrice[4]; OutPacket(payPriceReply, ThrottleOutPacketType.Task); } public void SendKillObject(List localIDs) { // foreach (uint id in localIDs) // m_log.DebugFormat("[CLIENT]: Sending KillObjectPacket to {0} for {1} in {2}", Name, id, regionHandle); // remove pending entities to reduce looping chances. lock (m_entityProps.SyncRoot) m_entityProps.Remove(localIDs); lock (m_entityUpdates.SyncRoot) m_entityUpdates.Remove(localIDs); KillObjectPacket kill = (KillObjectPacket)PacketPool.Instance.GetPacket(PacketType.KillObject); int perpacket = localIDs.Count; if(perpacket > 200) perpacket = 200; int nsent = 0; kill.ObjectData = new KillObjectPacket.ObjectDataBlock[perpacket]; for (int i = 0 ; i < localIDs.Count ; i++ ) { kill.ObjectData[nsent] = new KillObjectPacket.ObjectDataBlock(); kill.ObjectData[nsent].ID = localIDs[i]; if(++nsent >= 200) { OutPacket(kill, ThrottleOutPacketType.Task); perpacket = localIDs.Count - i - 1; if(perpacket == 0) break; if(perpacket > 200) perpacket = 200; kill = (KillObjectPacket)PacketPool.Instance.GetPacket(PacketType.KillObject); kill.ObjectData = new KillObjectPacket.ObjectDataBlock[perpacket]; nsent = 0; } } if(nsent != 0) { OutPacket(kill, ThrottleOutPacketType.Task); } } /// /// Send information about the items contained in a folder to the client. /// /// /// XXX This method needs some refactoring loving /// /// The owner of the folder /// The id of the folder /// The items contained in the folder identified by folderID /// /// Do we need to send folder information? /// Do we need to send item information? public void SendInventoryFolderDetails(UUID ownerID, UUID folderID, List items, List folders, int version, bool fetchFolders, bool fetchItems) { // An inventory descendents packet consists of a single agent section and an inventory details // section for each inventory item. The size of each inventory item is approximately 550 bytes. // limit to what may fit on MTU int MAX_ITEMS_PER_PACKET = 5; int MAX_FOLDERS_PER_PACKET = 6; int totalItems = fetchItems ? items.Count : 0; int totalFolders = fetchFolders ? folders.Count : 0; int itemsSent = 0; int foldersSent = 0; int foldersToSend = 0; int itemsToSend = 0; InventoryDescendentsPacket currentPacket = null; // Handle empty folders // if (totalItems == 0 && totalFolders == 0) currentPacket = CreateInventoryDescendentsPacket(ownerID, folderID, version, items.Count + folders.Count, 0, 0); // To preserve SL compatibility, we will NOT combine folders and items in one packet // while (itemsSent < totalItems || foldersSent < totalFolders) { if (currentPacket == null) // Start a new packet { foldersToSend = totalFolders - foldersSent; if (foldersToSend > MAX_FOLDERS_PER_PACKET) foldersToSend = MAX_FOLDERS_PER_PACKET; if (foldersToSend == 0) { itemsToSend = totalItems - itemsSent; if (itemsToSend > MAX_ITEMS_PER_PACKET) itemsToSend = MAX_ITEMS_PER_PACKET; } currentPacket = CreateInventoryDescendentsPacket(ownerID, folderID, version, items.Count + folders.Count, foldersToSend, itemsToSend); } if (foldersToSend-- > 0) currentPacket.FolderData[foldersSent % MAX_FOLDERS_PER_PACKET] = CreateFolderDataBlock(folders[foldersSent++]); else if (itemsToSend-- > 0) currentPacket.ItemData[itemsSent % MAX_ITEMS_PER_PACKET] = CreateItemDataBlock(items[itemsSent++]); else { // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending inventory folder details packet to {0} for folder {1}", Name, folderID); OutPacket(currentPacket, ThrottleOutPacketType.Asset, false); currentPacket = null; } } if (currentPacket != null) { // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending inventory folder details packet to {0} for folder {1}", Name, folderID); OutPacket(currentPacket, ThrottleOutPacketType.Asset, false); } } private InventoryDescendentsPacket.FolderDataBlock CreateFolderDataBlock(InventoryFolderBase folder) { InventoryDescendentsPacket.FolderDataBlock newBlock = new InventoryDescendentsPacket.FolderDataBlock(); newBlock.FolderID = folder.ID; newBlock.Name = Util.StringToBytes256(folder.Name); newBlock.ParentID = folder.ParentID; newBlock.Type = (sbyte)folder.Type; //if (newBlock.Type == InventoryItemBase.SUITCASE_FOLDER_TYPE) // newBlock.Type = InventoryItemBase.SUITCASE_FOLDER_FAKE_TYPE; return newBlock; } private InventoryDescendentsPacket.ItemDataBlock CreateItemDataBlock(InventoryItemBase item) { InventoryDescendentsPacket.ItemDataBlock newBlock = new InventoryDescendentsPacket.ItemDataBlock(); newBlock.ItemID = item.ID; newBlock.AssetID = item.AssetID; newBlock.CreatorID = item.CreatorIdAsUuid; newBlock.BaseMask = item.BasePermissions; newBlock.Description = Util.StringToBytes256(item.Description); newBlock.EveryoneMask = item.EveryOnePermissions; newBlock.OwnerMask = item.CurrentPermissions; newBlock.FolderID = item.Folder; newBlock.InvType = (sbyte)item.InvType; newBlock.Name = Util.StringToBytes256(item.Name); newBlock.NextOwnerMask = item.NextPermissions; newBlock.OwnerID = item.Owner; newBlock.Type = (sbyte)item.AssetType; newBlock.GroupID = item.GroupID; newBlock.GroupOwned = item.GroupOwned; newBlock.GroupMask = item.GroupPermissions; newBlock.CreationDate = item.CreationDate; newBlock.SalePrice = item.SalePrice; newBlock.SaleType = item.SaleType; newBlock.Flags = item.Flags & 0x2000ff; newBlock.CRC = Helpers.InventoryCRC(newBlock.CreationDate, newBlock.SaleType, newBlock.InvType, newBlock.Type, newBlock.AssetID, newBlock.GroupID, newBlock.SalePrice, newBlock.OwnerID, newBlock.CreatorID, newBlock.ItemID, newBlock.FolderID, newBlock.EveryoneMask, newBlock.Flags, newBlock.OwnerMask, newBlock.GroupMask, newBlock.NextOwnerMask); return newBlock; } private void AddNullFolderBlockToDecendentsPacket(ref InventoryDescendentsPacket packet) { packet.FolderData = new InventoryDescendentsPacket.FolderDataBlock[1]; packet.FolderData[0] = new InventoryDescendentsPacket.FolderDataBlock(); packet.FolderData[0].FolderID = UUID.Zero; packet.FolderData[0].ParentID = UUID.Zero; packet.FolderData[0].Type = -1; packet.FolderData[0].Name = new byte[0]; } private void AddNullItemBlockToDescendentsPacket(ref InventoryDescendentsPacket packet) { packet.ItemData = new InventoryDescendentsPacket.ItemDataBlock[1]; packet.ItemData[0] = new InventoryDescendentsPacket.ItemDataBlock(); packet.ItemData[0].ItemID = UUID.Zero; packet.ItemData[0].AssetID = UUID.Zero; packet.ItemData[0].CreatorID = UUID.Zero; packet.ItemData[0].BaseMask = 0; packet.ItemData[0].Description = new byte[0]; packet.ItemData[0].EveryoneMask = 0; packet.ItemData[0].OwnerMask = 0; packet.ItemData[0].FolderID = UUID.Zero; packet.ItemData[0].InvType = (sbyte)0; packet.ItemData[0].Name = new byte[0]; packet.ItemData[0].NextOwnerMask = 0; packet.ItemData[0].OwnerID = UUID.Zero; packet.ItemData[0].Type = -1; packet.ItemData[0].GroupID = UUID.Zero; packet.ItemData[0].GroupOwned = false; packet.ItemData[0].GroupMask = 0; packet.ItemData[0].CreationDate = 0; packet.ItemData[0].SalePrice = 0; packet.ItemData[0].SaleType = 0; packet.ItemData[0].Flags = 0; // No need to add CRC } private InventoryDescendentsPacket CreateInventoryDescendentsPacket(UUID ownerID, UUID folderID, int version, int descendents, int folders, int items) { InventoryDescendentsPacket descend = (InventoryDescendentsPacket)PacketPool.Instance.GetPacket(PacketType.InventoryDescendents); descend.Header.Zerocoded = true; descend.AgentData.AgentID = AgentId; descend.AgentData.OwnerID = ownerID; descend.AgentData.FolderID = folderID; descend.AgentData.Version = version; descend.AgentData.Descendents = descendents; if (folders > 0) descend.FolderData = new InventoryDescendentsPacket.FolderDataBlock[folders]; else AddNullFolderBlockToDecendentsPacket(ref descend); if (items > 0) descend.ItemData = new InventoryDescendentsPacket.ItemDataBlock[items]; else AddNullItemBlockToDescendentsPacket(ref descend); return descend; } public void SendInventoryItemDetails(UUID ownerID, InventoryItemBase item) { // Fudge this value. It's only needed to make the CRC anyway const uint FULL_MASK_PERMISSIONS = (uint)0x7fffffff; FetchInventoryReplyPacket inventoryReply = (FetchInventoryReplyPacket)PacketPool.Instance.GetPacket(PacketType.FetchInventoryReply); // TODO: don't create new blocks if recycling an old packet inventoryReply.AgentData.AgentID = AgentId; inventoryReply.InventoryData = new FetchInventoryReplyPacket.InventoryDataBlock[1]; inventoryReply.InventoryData[0] = new FetchInventoryReplyPacket.InventoryDataBlock(); inventoryReply.InventoryData[0].ItemID = item.ID; inventoryReply.InventoryData[0].AssetID = item.AssetID; inventoryReply.InventoryData[0].CreatorID = item.CreatorIdAsUuid; inventoryReply.InventoryData[0].BaseMask = item.BasePermissions; inventoryReply.InventoryData[0].CreationDate = item.CreationDate; inventoryReply.InventoryData[0].Description = Util.StringToBytes256(item.Description); inventoryReply.InventoryData[0].EveryoneMask = item.EveryOnePermissions; inventoryReply.InventoryData[0].FolderID = item.Folder; inventoryReply.InventoryData[0].InvType = (sbyte)item.InvType; inventoryReply.InventoryData[0].Name = Util.StringToBytes256(item.Name); inventoryReply.InventoryData[0].NextOwnerMask = item.NextPermissions; inventoryReply.InventoryData[0].OwnerID = item.Owner; inventoryReply.InventoryData[0].OwnerMask = item.CurrentPermissions; inventoryReply.InventoryData[0].Type = (sbyte)item.AssetType; inventoryReply.InventoryData[0].GroupID = item.GroupID; inventoryReply.InventoryData[0].GroupOwned = item.GroupOwned; inventoryReply.InventoryData[0].GroupMask = item.GroupPermissions; inventoryReply.InventoryData[0].Flags = item.Flags; inventoryReply.InventoryData[0].SalePrice = item.SalePrice; inventoryReply.InventoryData[0].SaleType = item.SaleType; inventoryReply.InventoryData[0].CRC = Helpers.InventoryCRC( 1000, 0, inventoryReply.InventoryData[0].InvType, inventoryReply.InventoryData[0].Type, inventoryReply.InventoryData[0].AssetID, inventoryReply.InventoryData[0].GroupID, 100, inventoryReply.InventoryData[0].OwnerID, inventoryReply.InventoryData[0].CreatorID, inventoryReply.InventoryData[0].ItemID, inventoryReply.InventoryData[0].FolderID, FULL_MASK_PERMISSIONS, 1, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS); inventoryReply.Header.Zerocoded = true; OutPacket(inventoryReply, ThrottleOutPacketType.Asset); } protected void SendBulkUpdateInventoryFolder(InventoryFolderBase folderBase) { // We will use the same transaction id for all the separate packets to be sent out in this update. UUID transactionId = UUID.Random(); List folderDataBlocks = new List(); SendBulkUpdateInventoryFolderRecursive(folderBase, ref folderDataBlocks, transactionId); if (folderDataBlocks.Count > 0) { // We'll end up with some unsent folder blocks if there were some empty folders at the end of the list // Send these now BulkUpdateInventoryPacket bulkUpdate = (BulkUpdateInventoryPacket)PacketPool.Instance.GetPacket(PacketType.BulkUpdateInventory); bulkUpdate.Header.Zerocoded = true; bulkUpdate.AgentData.AgentID = AgentId; bulkUpdate.AgentData.TransactionID = transactionId; bulkUpdate.FolderData = folderDataBlocks.ToArray(); List foo = new List(); bulkUpdate.ItemData = foo.ToArray(); //m_log.Debug("SendBulkUpdateInventory :" + bulkUpdate); OutPacket(bulkUpdate, ThrottleOutPacketType.Asset); } } /// /// Recursively construct bulk update packets to send folders and items /// /// /// /// private void SendBulkUpdateInventoryFolderRecursive( InventoryFolderBase folder, ref List folderDataBlocks, UUID transactionId) { folderDataBlocks.Add(GenerateBulkUpdateFolderDataBlock(folder)); const int MAX_ITEMS_PER_PACKET = 5; IInventoryService invService = m_scene.RequestModuleInterface(); // If there are any items then we have to start sending them off in this packet - the next folder will have // to be in its own bulk update packet. Also, we can only fit 5 items in a packet (at least this was the limit // being used on the Linden grid at 20081203). InventoryCollection contents = invService.GetFolderContent(AgentId, folder.ID); // folder.RequestListOfItems(); List items = contents.Items; while (items.Count > 0) { BulkUpdateInventoryPacket bulkUpdate = (BulkUpdateInventoryPacket)PacketPool.Instance.GetPacket(PacketType.BulkUpdateInventory); bulkUpdate.Header.Zerocoded = true; bulkUpdate.AgentData.AgentID = AgentId; bulkUpdate.AgentData.TransactionID = transactionId; bulkUpdate.FolderData = folderDataBlocks.ToArray(); int itemsToSend = (items.Count > MAX_ITEMS_PER_PACKET ? MAX_ITEMS_PER_PACKET : items.Count); bulkUpdate.ItemData = new BulkUpdateInventoryPacket.ItemDataBlock[itemsToSend]; for (int i = 0; i < itemsToSend; i++) { // Remove from the end of the list so that we don't incur a performance penalty bulkUpdate.ItemData[i] = GenerateBulkUpdateItemDataBlock(items[items.Count - 1]); items.RemoveAt(items.Count - 1); } //m_log.Debug("SendBulkUpdateInventoryRecursive :" + bulkUpdate); OutPacket(bulkUpdate, ThrottleOutPacketType.Asset); folderDataBlocks = new List(); // If we're going to be sending another items packet then it needs to contain just the folder to which those // items belong. if (items.Count > 0) folderDataBlocks.Add(GenerateBulkUpdateFolderDataBlock(folder)); } List subFolders = contents.Folders; foreach (InventoryFolderBase subFolder in subFolders) { SendBulkUpdateInventoryFolderRecursive(subFolder, ref folderDataBlocks, transactionId); } } /// /// Generate a bulk update inventory data block for the given folder /// /// /// private BulkUpdateInventoryPacket.FolderDataBlock GenerateBulkUpdateFolderDataBlock(InventoryFolderBase folder) { BulkUpdateInventoryPacket.FolderDataBlock folderBlock = new BulkUpdateInventoryPacket.FolderDataBlock(); folderBlock.FolderID = folder.ID; folderBlock.ParentID = folder.ParentID; folderBlock.Type = (sbyte)folder.Type; // Leaving this here for now, just in case we need to do this for a while //if (folderBlock.Type == InventoryItemBase.SUITCASE_FOLDER_TYPE) // folderBlock.Type = InventoryItemBase.SUITCASE_FOLDER_FAKE_TYPE; folderBlock.Name = Util.StringToBytes256(folder.Name); return folderBlock; } /// /// Generate a bulk update inventory data block for the given item /// /// /// private BulkUpdateInventoryPacket.ItemDataBlock GenerateBulkUpdateItemDataBlock(InventoryItemBase item) { BulkUpdateInventoryPacket.ItemDataBlock itemBlock = new BulkUpdateInventoryPacket.ItemDataBlock(); itemBlock.ItemID = item.ID; itemBlock.AssetID = item.AssetID; itemBlock.CreatorID = item.CreatorIdAsUuid; itemBlock.BaseMask = item.BasePermissions; itemBlock.Description = Util.StringToBytes256(item.Description); itemBlock.EveryoneMask = item.EveryOnePermissions; itemBlock.FolderID = item.Folder; itemBlock.InvType = (sbyte)item.InvType; itemBlock.Name = Util.StringToBytes256(item.Name); itemBlock.NextOwnerMask = item.NextPermissions; itemBlock.OwnerID = item.Owner; itemBlock.OwnerMask = item.CurrentPermissions; itemBlock.Type = (sbyte)item.AssetType; itemBlock.GroupID = item.GroupID; itemBlock.GroupOwned = item.GroupOwned; itemBlock.GroupMask = item.GroupPermissions; itemBlock.Flags = item.Flags & 0x2000ff; itemBlock.SalePrice = item.SalePrice; itemBlock.SaleType = item.SaleType; itemBlock.CreationDate = item.CreationDate; itemBlock.CRC = Helpers.InventoryCRC( 1000, 0, itemBlock.InvType, itemBlock.Type, itemBlock.AssetID, itemBlock.GroupID, 100, itemBlock.OwnerID, itemBlock.CreatorID, itemBlock.ItemID, itemBlock.FolderID, (uint)PermissionMask.All, 1, (uint)PermissionMask.All, (uint)PermissionMask.All, (uint)PermissionMask.All); return itemBlock; } public void SendBulkUpdateInventory(InventoryNodeBase node) { if (node is InventoryItemBase) SendBulkUpdateInventoryItem((InventoryItemBase)node); else if (node is InventoryFolderBase) SendBulkUpdateInventoryFolder((InventoryFolderBase)node); else if (node != null) m_log.ErrorFormat("[CLIENT]: {0} sent unknown inventory node named {1}", Name, node.Name); else m_log.ErrorFormat("[CLIENT]: {0} sent null inventory node", Name); } protected void SendBulkUpdateInventoryItem(InventoryItemBase item) { const uint FULL_MASK_PERMISSIONS = (uint)0x7ffffff; BulkUpdateInventoryPacket bulkUpdate = (BulkUpdateInventoryPacket)PacketPool.Instance.GetPacket(PacketType.BulkUpdateInventory); bulkUpdate.AgentData.AgentID = AgentId; bulkUpdate.AgentData.TransactionID = UUID.Random(); bulkUpdate.FolderData = new BulkUpdateInventoryPacket.FolderDataBlock[1]; bulkUpdate.FolderData[0] = new BulkUpdateInventoryPacket.FolderDataBlock(); bulkUpdate.FolderData[0].FolderID = UUID.Zero; bulkUpdate.FolderData[0].ParentID = UUID.Zero; bulkUpdate.FolderData[0].Type = -1; bulkUpdate.FolderData[0].Name = new byte[0]; bulkUpdate.ItemData = new BulkUpdateInventoryPacket.ItemDataBlock[1]; bulkUpdate.ItemData[0] = new BulkUpdateInventoryPacket.ItemDataBlock(); bulkUpdate.ItemData[0].ItemID = item.ID; bulkUpdate.ItemData[0].AssetID = item.AssetID; bulkUpdate.ItemData[0].CreatorID = item.CreatorIdAsUuid; bulkUpdate.ItemData[0].BaseMask = item.BasePermissions; bulkUpdate.ItemData[0].CreationDate = item.CreationDate; bulkUpdate.ItemData[0].Description = Util.StringToBytes256(item.Description); bulkUpdate.ItemData[0].EveryoneMask = item.EveryOnePermissions; bulkUpdate.ItemData[0].FolderID = item.Folder; bulkUpdate.ItemData[0].InvType = (sbyte)item.InvType; bulkUpdate.ItemData[0].Name = Util.StringToBytes256(item.Name); bulkUpdate.ItemData[0].NextOwnerMask = item.NextPermissions; bulkUpdate.ItemData[0].OwnerID = item.Owner; bulkUpdate.ItemData[0].OwnerMask = item.CurrentPermissions; bulkUpdate.ItemData[0].Type = (sbyte)item.AssetType; bulkUpdate.ItemData[0].GroupID = item.GroupID; bulkUpdate.ItemData[0].GroupOwned = item.GroupOwned; bulkUpdate.ItemData[0].GroupMask = item.GroupPermissions; bulkUpdate.ItemData[0].Flags = item.Flags & 0x2000ff; bulkUpdate.ItemData[0].SalePrice = item.SalePrice; bulkUpdate.ItemData[0].SaleType = item.SaleType; bulkUpdate.ItemData[0].CRC = Helpers.InventoryCRC(1000, 0, bulkUpdate.ItemData[0].InvType, bulkUpdate.ItemData[0].Type, bulkUpdate.ItemData[0].AssetID, bulkUpdate.ItemData[0].GroupID, 100, bulkUpdate.ItemData[0].OwnerID, bulkUpdate.ItemData[0].CreatorID, bulkUpdate.ItemData[0].ItemID, bulkUpdate.ItemData[0].FolderID, FULL_MASK_PERMISSIONS, 1, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS); bulkUpdate.Header.Zerocoded = true; OutPacket(bulkUpdate, ThrottleOutPacketType.Asset); } public void SendInventoryItemCreateUpdate(InventoryItemBase Item, uint callbackId) { SendInventoryItemCreateUpdate(Item, UUID.Zero, callbackId); } /// IClientAPI.SendInventoryItemCreateUpdate(InventoryItemBase) public void SendInventoryItemCreateUpdate(InventoryItemBase Item, UUID transactionID, uint callbackId) { const uint FULL_MASK_PERMISSIONS = (uint)0x7fffffff; UpdateCreateInventoryItemPacket InventoryReply = (UpdateCreateInventoryItemPacket)PacketPool.Instance.GetPacket( PacketType.UpdateCreateInventoryItem); // TODO: don't create new blocks if recycling an old packet InventoryReply.AgentData.AgentID = AgentId; InventoryReply.AgentData.SimApproved = true; InventoryReply.AgentData.TransactionID = transactionID; InventoryReply.InventoryData = new UpdateCreateInventoryItemPacket.InventoryDataBlock[1]; InventoryReply.InventoryData[0] = new UpdateCreateInventoryItemPacket.InventoryDataBlock(); InventoryReply.InventoryData[0].ItemID = Item.ID; InventoryReply.InventoryData[0].AssetID = Item.AssetID; InventoryReply.InventoryData[0].CreatorID = Item.CreatorIdAsUuid; InventoryReply.InventoryData[0].BaseMask = Item.BasePermissions; InventoryReply.InventoryData[0].Description = Util.StringToBytes256(Item.Description); InventoryReply.InventoryData[0].EveryoneMask = Item.EveryOnePermissions; InventoryReply.InventoryData[0].FolderID = Item.Folder; InventoryReply.InventoryData[0].InvType = (sbyte)Item.InvType; InventoryReply.InventoryData[0].Name = Util.StringToBytes256(Item.Name); InventoryReply.InventoryData[0].NextOwnerMask = Item.NextPermissions; InventoryReply.InventoryData[0].OwnerID = Item.Owner; InventoryReply.InventoryData[0].OwnerMask = Item.CurrentPermissions; InventoryReply.InventoryData[0].Type = (sbyte)Item.AssetType; InventoryReply.InventoryData[0].CallbackID = callbackId; InventoryReply.InventoryData[0].GroupID = Item.GroupID; InventoryReply.InventoryData[0].GroupOwned = Item.GroupOwned; InventoryReply.InventoryData[0].GroupMask = Item.GroupPermissions; InventoryReply.InventoryData[0].Flags = Item.Flags & 0x2000ff; InventoryReply.InventoryData[0].SalePrice = Item.SalePrice; InventoryReply.InventoryData[0].SaleType = Item.SaleType; InventoryReply.InventoryData[0].CreationDate = Item.CreationDate; InventoryReply.InventoryData[0].CRC = Helpers.InventoryCRC(1000, 0, InventoryReply.InventoryData[0].InvType, InventoryReply.InventoryData[0].Type, InventoryReply.InventoryData[0].AssetID, InventoryReply.InventoryData[0].GroupID, 100, InventoryReply.InventoryData[0].OwnerID, InventoryReply.InventoryData[0].CreatorID, InventoryReply.InventoryData[0].ItemID, InventoryReply.InventoryData[0].FolderID, FULL_MASK_PERMISSIONS, 1, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS); InventoryReply.Header.Zerocoded = true; OutPacket(InventoryReply, ThrottleOutPacketType.Asset); } public void SendRemoveInventoryItem(UUID itemID) { RemoveInventoryItemPacket remove = (RemoveInventoryItemPacket)PacketPool.Instance.GetPacket(PacketType.RemoveInventoryItem); // TODO: don't create new blocks if recycling an old packet remove.AgentData.AgentID = AgentId; remove.AgentData.SessionID = m_sessionId; remove.InventoryData = new RemoveInventoryItemPacket.InventoryDataBlock[1]; remove.InventoryData[0] = new RemoveInventoryItemPacket.InventoryDataBlock(); remove.InventoryData[0].ItemID = itemID; remove.Header.Zerocoded = true; OutPacket(remove, ThrottleOutPacketType.Asset); } /* private uint adjustControls(int input) { uint ret = (uint)input; uint masked = ret & 0x0f; masked <<= 19; ret |= masked; return ret; } */ public void SendTakeControls(int controls, bool passToAgent, bool TakeControls) { ScriptControlChangePacket scriptcontrol = (ScriptControlChangePacket)PacketPool.Instance.GetPacket(PacketType.ScriptControlChange); ScriptControlChangePacket.DataBlock[] data = new ScriptControlChangePacket.DataBlock[1]; ScriptControlChangePacket.DataBlock ddata = new ScriptControlChangePacket.DataBlock(); // ddata.Controls = adjustControls(controls); ddata.Controls = (uint)controls; ddata.PassToAgent = passToAgent; ddata.TakeControls = TakeControls; data[0] = ddata; scriptcontrol.Data = data; OutPacket(scriptcontrol, ThrottleOutPacketType.Task); } public void SendTaskInventory(UUID taskID, short serial, byte[] fileName) { ReplyTaskInventoryPacket replytask = (ReplyTaskInventoryPacket)PacketPool.Instance.GetPacket(PacketType.ReplyTaskInventory); replytask.InventoryData.TaskID = taskID; replytask.InventoryData.Serial = serial; replytask.InventoryData.Filename = fileName; // OutPacket(replytask, ThrottleOutPacketType.Task); OutPacket(replytask, ThrottleOutPacketType.Asset); } public void SendXferPacket(ulong xferID, uint packet, byte[] data, bool isTaskInventory) { ThrottleOutPacketType type = ThrottleOutPacketType.Asset; // if (isTaskInventory) // type = ThrottleOutPacketType.Task; SendXferPacketPacket sendXfer = (SendXferPacketPacket)PacketPool.Instance.GetPacket(PacketType.SendXferPacket); sendXfer.XferID.ID = xferID; sendXfer.XferID.Packet = packet; sendXfer.DataPacket.Data = data; OutPacket(sendXfer, type); } public void SendAbortXferPacket(ulong xferID) { AbortXferPacket xferItem = (AbortXferPacket)PacketPool.Instance.GetPacket(PacketType.AbortXfer); xferItem.XferID.ID = xferID; OutPacket(xferItem, ThrottleOutPacketType.Asset); } 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) { EconomyDataPacket economyData = (EconomyDataPacket)PacketPool.Instance.GetPacket(PacketType.EconomyData); economyData.Info.EnergyEfficiency = EnergyEfficiency; economyData.Info.ObjectCapacity = ObjectCapacity; economyData.Info.ObjectCount = ObjectCount; economyData.Info.PriceEnergyUnit = PriceEnergyUnit; economyData.Info.PriceGroupCreate = PriceGroupCreate; economyData.Info.PriceObjectClaim = PriceObjectClaim; economyData.Info.PriceObjectRent = PriceObjectRent; economyData.Info.PriceObjectScaleFactor = PriceObjectScaleFactor; economyData.Info.PriceParcelClaim = PriceParcelClaim; economyData.Info.PriceParcelClaimFactor = PriceParcelClaimFactor; economyData.Info.PriceParcelRent = PriceParcelRent; economyData.Info.PricePublicObjectDecay = PricePublicObjectDecay; economyData.Info.PricePublicObjectDelete = PricePublicObjectDelete; economyData.Info.PriceRentLight = PriceRentLight; economyData.Info.PriceUpload = PriceUpload; economyData.Info.TeleportMinPrice = TeleportMinPrice; economyData.Info.TeleportPriceExponent = TeleportPriceExponent; economyData.Header.Reliable = true; OutPacket(economyData, ThrottleOutPacketType.Task); } public void SendAvatarPickerReply(AvatarPickerReplyAgentDataArgs AgentData, List Data) { //construct the AvatarPickerReply packet. AvatarPickerReplyPacket replyPacket = new AvatarPickerReplyPacket(); replyPacket.AgentData.AgentID = AgentData.AgentID; replyPacket.AgentData.QueryID = AgentData.QueryID; //int i = 0; List data_block = new List(); foreach (AvatarPickerReplyDataArgs arg in Data) { AvatarPickerReplyPacket.DataBlock db = new AvatarPickerReplyPacket.DataBlock(); db.AvatarID = arg.AvatarID; db.FirstName = arg.FirstName; db.LastName = arg.LastName; data_block.Add(db); } replyPacket.Data = data_block.ToArray(); OutPacket(replyPacket, ThrottleOutPacketType.Task); } public void SendAgentDataUpdate(UUID agentid, UUID activegroupid, string firstname, string lastname, ulong grouppowers, string groupname, string grouptitle) { if (agentid == AgentId) { ActiveGroupId = activegroupid; ActiveGroupName = groupname; ActiveGroupPowers = grouppowers; } AgentDataUpdatePacket sendAgentDataUpdate = (AgentDataUpdatePacket)PacketPool.Instance.GetPacket(PacketType.AgentDataUpdate); sendAgentDataUpdate.AgentData.ActiveGroupID = activegroupid; sendAgentDataUpdate.AgentData.AgentID = agentid; sendAgentDataUpdate.AgentData.FirstName = Util.StringToBytes256(firstname); sendAgentDataUpdate.AgentData.GroupName = Util.StringToBytes256(groupname); sendAgentDataUpdate.AgentData.GroupPowers = grouppowers; sendAgentDataUpdate.AgentData.GroupTitle = Util.StringToBytes256(grouptitle); sendAgentDataUpdate.AgentData.LastName = Util.StringToBytes256(lastname); OutPacket(sendAgentDataUpdate, ThrottleOutPacketType.Task); } /// /// Send an alert message to the client. This pops up a brief duration information box at a corner /// /// public void SendAlertMessage(string message) { AlertMessagePacket alertPack = (AlertMessagePacket)PacketPool.Instance.GetPacket(PacketType.AlertMessage); alertPack.AgentInfo = new AlertMessagePacket.AgentInfoBlock[1]; alertPack.AgentInfo[0] = new AlertMessagePacket.AgentInfoBlock(); alertPack.AgentInfo[0].AgentID = AgentId; alertPack.AlertData = new AlertMessagePacket.AlertDataBlock(); alertPack.AlertData.Message = Util.StringToBytes256(message); alertPack.AlertInfo = new AlertMessagePacket.AlertInfoBlock[0]; OutPacket(alertPack, ThrottleOutPacketType.Task); } public void SendAlertMessage(string message, string info) { AlertMessagePacket alertPack = (AlertMessagePacket)PacketPool.Instance.GetPacket(PacketType.AlertMessage); alertPack.AgentInfo = new AlertMessagePacket.AgentInfoBlock[1]; alertPack.AgentInfo[0] = new AlertMessagePacket.AgentInfoBlock(); alertPack.AgentInfo[0].AgentID = AgentId; alertPack.AlertData = new AlertMessagePacket.AlertDataBlock(); alertPack.AlertData.Message = Util.StringToBytes256(message); alertPack.AlertInfo = new AlertMessagePacket.AlertInfoBlock[1]; alertPack.AlertInfo[0] = new AlertMessagePacket.AlertInfoBlock(); alertPack.AlertInfo[0].Message = Util.StringToBytes256(info); alertPack.AlertInfo[0].ExtraParams = new Byte[0]; OutPacket(alertPack, ThrottleOutPacketType.Task); } /// /// Send an agent alert message to the client. /// /// /// On the linden client, if this true then it displays a one button text box placed in the /// middle of the window. If false, the message is displayed in a brief duration blue information box (as for /// the AlertMessage packet). public void SendAgentAlertMessage(string message, bool modal) { // Prepend a slash to make the message come up in the top right // again. // Allow special formats to be sent from aware modules. if (!modal && !message.StartsWith("ALERT: ") && !message.StartsWith("NOTIFY: ") && message != "Home position set." && message != "You died and have been teleported to your home location") message = "/" + message; AgentAlertMessagePacket alertPack = (AgentAlertMessagePacket)PacketPool.Instance.GetPacket(PacketType.AgentAlertMessage); alertPack.AgentData.AgentID = AgentId; alertPack.AlertData.Message = Util.StringToBytes256(message); alertPack.AlertData.Modal = modal; OutPacket(alertPack, ThrottleOutPacketType.Task); } public void SendLoadURL(string objectname, UUID objectID, UUID ownerID, bool groupOwned, string message, string url) { LoadURLPacket loadURL = (LoadURLPacket)PacketPool.Instance.GetPacket(PacketType.LoadURL); loadURL.Data.ObjectName = Util.StringToBytes256(objectname); loadURL.Data.ObjectID = objectID; loadURL.Data.OwnerID = ownerID; loadURL.Data.OwnerIsGroup = groupOwned; loadURL.Data.Message = Util.StringToBytes256(message); loadURL.Data.URL = Util.StringToBytes256(url); OutPacket(loadURL, ThrottleOutPacketType.Task); } public void SendDialog( string objectname, UUID objectID, UUID ownerID, string ownerFirstName, string ownerLastName, string msg, UUID textureID, int ch, string[] buttonlabels) { ScriptDialogPacket dialog = (ScriptDialogPacket)PacketPool.Instance.GetPacket(PacketType.ScriptDialog); dialog.Data.ObjectID = objectID; dialog.Data.ObjectName = Util.StringToBytes256(objectname); // this is the username of the *owner* dialog.Data.FirstName = Util.StringToBytes256(ownerFirstName); dialog.Data.LastName = Util.StringToBytes256(ownerLastName); dialog.Data.Message = Util.StringToBytes(msg,512); dialog.Data.ImageID = textureID; dialog.Data.ChatChannel = ch; ScriptDialogPacket.ButtonsBlock[] buttons = new ScriptDialogPacket.ButtonsBlock[buttonlabels.Length]; for (int i = 0; i < buttonlabels.Length; i++) { buttons[i] = new ScriptDialogPacket.ButtonsBlock(); buttons[i].ButtonLabel = Util.StringToBytes(buttonlabels[i],24); } dialog.Buttons = buttons; dialog.OwnerData = new ScriptDialogPacket.OwnerDataBlock[1]; dialog.OwnerData[0] = new ScriptDialogPacket.OwnerDataBlock(); dialog.OwnerData[0].OwnerID = ownerID; OutPacket(dialog, ThrottleOutPacketType.Task); } public void SendPreLoadSound(UUID objectID, UUID ownerID, UUID soundID) { PreloadSoundPacket preSound = (PreloadSoundPacket)PacketPool.Instance.GetPacket(PacketType.PreloadSound); // TODO: don't create new blocks if recycling an old packet preSound.DataBlock = new PreloadSoundPacket.DataBlockBlock[1]; preSound.DataBlock[0] = new PreloadSoundPacket.DataBlockBlock(); preSound.DataBlock[0].ObjectID = objectID; preSound.DataBlock[0].OwnerID = ownerID; preSound.DataBlock[0].SoundID = soundID; preSound.Header.Zerocoded = true; OutPacket(preSound, ThrottleOutPacketType.Task); } public void SendPlayAttachedSound(UUID soundID, UUID objectID, UUID ownerID, float gain, byte flags) { AttachedSoundPacket sound = (AttachedSoundPacket)PacketPool.Instance.GetPacket(PacketType.AttachedSound); sound.DataBlock.SoundID = soundID; sound.DataBlock.ObjectID = objectID; sound.DataBlock.OwnerID = ownerID; sound.DataBlock.Gain = gain; sound.DataBlock.Flags = flags; OutPacket(sound, ThrottleOutPacketType.Task); } public void SendTransferAbort(TransferRequestPacket transferRequest) { TransferAbortPacket abort = (TransferAbortPacket)PacketPool.Instance.GetPacket(PacketType.TransferAbort); abort.TransferInfo.TransferID = transferRequest.TransferInfo.TransferID; abort.TransferInfo.ChannelType = transferRequest.TransferInfo.ChannelType; m_log.Debug("[Assets] Aborting transfer; asset request failed"); OutPacket(abort, ThrottleOutPacketType.Task); } public void SendTriggeredSound(UUID soundID, UUID ownerID, UUID objectID, UUID parentID, ulong handle, Vector3 position, float gain) { SoundTriggerPacket sound = (SoundTriggerPacket)PacketPool.Instance.GetPacket(PacketType.SoundTrigger); sound.SoundData.SoundID = soundID; sound.SoundData.OwnerID = ownerID; sound.SoundData.ObjectID = objectID; sound.SoundData.ParentID = parentID; sound.SoundData.Handle = handle; sound.SoundData.Position = position; sound.SoundData.Gain = gain; OutPacket(sound, ThrottleOutPacketType.Task); } public void SendAttachedSoundGainChange(UUID objectID, float gain) { AttachedSoundGainChangePacket sound = (AttachedSoundGainChangePacket)PacketPool.Instance.GetPacket(PacketType.AttachedSoundGainChange); sound.DataBlock.ObjectID = objectID; sound.DataBlock.Gain = gain; OutPacket(sound, ThrottleOutPacketType.Task); } public void SendSunPos(Vector3 Position, Vector3 Velocity, ulong CurrentTime, uint SecondsPerSunCycle, uint SecondsPerYear, float OrbitalPosition) { // Viewers based on the Linden viwer code, do wacky things for oribital positions from Midnight to Sunrise // So adjust for that // Contributed by: Godfrey if (OrbitalPosition > m_sunPainDaHalfOrbitalCutoff) // things get weird from midnight to sunrise { OrbitalPosition = (OrbitalPosition - m_sunPainDaHalfOrbitalCutoff) * 0.6666666667f + m_sunPainDaHalfOrbitalCutoff; } SimulatorViewerTimeMessagePacket viewertime = (SimulatorViewerTimeMessagePacket)PacketPool.Instance.GetPacket(PacketType.SimulatorViewerTimeMessage); viewertime.TimeInfo.SunDirection = Position; viewertime.TimeInfo.SunAngVelocity = Velocity; // Sun module used to add 6 hours to adjust for linden sun hour, adding here // to prevent existing code from breaking if it assumed that 6 hours were included. // 21600 == 6 hours * 60 minutes * 60 Seconds viewertime.TimeInfo.UsecSinceStart = CurrentTime + 21600; viewertime.TimeInfo.SecPerDay = SecondsPerSunCycle; viewertime.TimeInfo.SecPerYear = SecondsPerYear; viewertime.TimeInfo.SunPhase = OrbitalPosition; viewertime.Header.Reliable = false; viewertime.Header.Zerocoded = true; OutPacket(viewertime, ThrottleOutPacketType.Task); } // Currently Deprecated public void SendViewerTime(int phase) { /* Console.WriteLine("SunPhase: {0}", phase); SimulatorViewerTimeMessagePacket viewertime = (SimulatorViewerTimeMessagePacket)PacketPool.Instance.GetPacket(PacketType.SimulatorViewerTimeMessage); //viewertime.TimeInfo.SecPerDay = 86400; //viewertime.TimeInfo.SecPerYear = 31536000; viewertime.TimeInfo.SecPerDay = 1000; viewertime.TimeInfo.SecPerYear = 365000; viewertime.TimeInfo.SunPhase = 1; int sunPhase = (phase + 2) / 2; if ((sunPhase < 6) || (sunPhase > 36)) { viewertime.TimeInfo.SunDirection = new Vector3(0f, 0.8f, -0.8f); Console.WriteLine("sending night"); } else { if (sunPhase < 12) { sunPhase = 12; } sunPhase = sunPhase - 12; float yValue = 0.1f * (sunPhase); Console.WriteLine("Computed SunPhase: {0}, yValue: {1}", sunPhase, yValue); if (yValue > 1.2f) { yValue = yValue - 1.2f; } yValue = Util.Clip(yValue, 0, 1); if (sunPhase < 14) { yValue = 1 - yValue; } if (sunPhase < 12) { yValue *= -1; } viewertime.TimeInfo.SunDirection = new Vector3(0f, yValue, 0.3f); Console.WriteLine("sending sun update " + yValue); } viewertime.TimeInfo.SunAngVelocity = new Vector3(0, 0.0f, 10.0f); viewertime.TimeInfo.UsecSinceStart = (ulong)Util.UnixTimeSinceEpoch(); viewertime.Header.Reliable = false; OutPacket(viewertime, ThrottleOutPacketType.Task); */ } public void SendViewerEffect(ViewerEffectPacket.EffectBlock[] effectBlocks) { ViewerEffectPacket packet = (ViewerEffectPacket)PacketPool.Instance.GetPacket(PacketType.ViewerEffect); // packet.AgentData.AgentID = AgentId; // packet.AgentData.SessionID = SessionId; packet.Effect = effectBlocks; // OutPacket(packet, ThrottleOutPacketType.State); OutPacket(packet, ThrottleOutPacketType.Task); } public void SendAvatarProperties(UUID avatarID, string aboutText, string bornOn, Byte[] membershipType, string flAbout, uint flags, UUID flImageID, UUID imageID, string profileURL, UUID partnerID) { AvatarPropertiesReplyPacket avatarReply = (AvatarPropertiesReplyPacket)PacketPool.Instance.GetPacket(PacketType.AvatarPropertiesReply); avatarReply.AgentData.AgentID = AgentId; avatarReply.AgentData.AvatarID = avatarID; if (aboutText != null) avatarReply.PropertiesData.AboutText = Util.StringToBytes1024(aboutText); else avatarReply.PropertiesData.AboutText = Utils.EmptyBytes; avatarReply.PropertiesData.BornOn = Util.StringToBytes256(bornOn); avatarReply.PropertiesData.CharterMember = membershipType; if (flAbout != null) avatarReply.PropertiesData.FLAboutText = Util.StringToBytes256(flAbout); else avatarReply.PropertiesData.FLAboutText = Utils.EmptyBytes; avatarReply.PropertiesData.Flags = flags; avatarReply.PropertiesData.FLImageID = flImageID; avatarReply.PropertiesData.ImageID = imageID; avatarReply.PropertiesData.ProfileURL = Util.StringToBytes256(profileURL); avatarReply.PropertiesData.PartnerID = partnerID; OutPacket(avatarReply, ThrottleOutPacketType.Task); } /// /// Send the client an Estate message blue box pop-down with a single OK button /// /// /// /// /// public void SendBlueBoxMessage(UUID FromAvatarID, String FromAvatarName, String Message) { if (!SceneAgent.IsChildAgent) SendInstantMessage(new GridInstantMessage(null, FromAvatarID, FromAvatarName, AgentId, 1, Message, false, new Vector3())); //SendInstantMessage(FromAvatarID, fromSessionID, Message, AgentId, SessionId, FromAvatarName, (byte)21,(uint) Util.UnixTimeSinceEpoch()); } public void SendLogoutPacket() { // I know this is a bit of a hack, however there are times when you don't // want to send this, but still need to do the rest of the shutdown process // this method gets called from the packet server.. which makes it practically // impossible to do any other way. if (m_SendLogoutPacketWhenClosing) { LogoutReplyPacket logReply = (LogoutReplyPacket)PacketPool.Instance.GetPacket(PacketType.LogoutReply); // TODO: don't create new blocks if recycling an old packet logReply.AgentData.AgentID = AgentId; logReply.AgentData.SessionID = SessionId; logReply.InventoryData = new LogoutReplyPacket.InventoryDataBlock[1]; logReply.InventoryData[0] = new LogoutReplyPacket.InventoryDataBlock(); logReply.InventoryData[0].ItemID = UUID.Zero; OutPacket(logReply, ThrottleOutPacketType.Task); } } public void SendHealth(float health) { HealthMessagePacket healthpacket = (HealthMessagePacket)PacketPool.Instance.GetPacket(PacketType.HealthMessage); healthpacket.HealthData.Health = health; OutPacket(healthpacket, ThrottleOutPacketType.Task); } public void SendAgentOnline(UUID[] agentIDs) { OnlineNotificationPacket onp = new OnlineNotificationPacket(); OnlineNotificationPacket.AgentBlockBlock[] onpb = new OnlineNotificationPacket.AgentBlockBlock[agentIDs.Length]; for (int i = 0; i < agentIDs.Length; i++) { OnlineNotificationPacket.AgentBlockBlock onpbl = new OnlineNotificationPacket.AgentBlockBlock(); onpbl.AgentID = agentIDs[i]; onpb[i] = onpbl; } onp.AgentBlock = onpb; onp.Header.Reliable = true; OutPacket(onp, ThrottleOutPacketType.Task); } public void SendAgentOffline(UUID[] agentIDs) { OfflineNotificationPacket offp = new OfflineNotificationPacket(); OfflineNotificationPacket.AgentBlockBlock[] offpb = new OfflineNotificationPacket.AgentBlockBlock[agentIDs.Length]; for (int i = 0; i < agentIDs.Length; i++) { OfflineNotificationPacket.AgentBlockBlock onpbl = new OfflineNotificationPacket.AgentBlockBlock(); onpbl.AgentID = agentIDs[i]; offpb[i] = onpbl; } offp.AgentBlock = offpb; offp.Header.Reliable = true; OutPacket(offp, ThrottleOutPacketType.Task); } public void SendFindAgent(UUID HunterID, UUID PreyID, double GlobalX, double GlobalY) { FindAgentPacket fap = new FindAgentPacket(); fap.AgentBlock.Hunter = HunterID; fap.AgentBlock.Prey = PreyID; fap.AgentBlock.SpaceIP = 0; fap.LocationBlock = new FindAgentPacket.LocationBlockBlock[1]; fap.LocationBlock[0] = new FindAgentPacket.LocationBlockBlock(); fap.LocationBlock[0].GlobalX = GlobalX; fap.LocationBlock[0].GlobalY = GlobalY; OutPacket(fap, ThrottleOutPacketType.Task); } public void SendSitResponse(UUID TargetID, Vector3 OffsetPos, Quaternion SitOrientation, bool autopilot, Vector3 CameraAtOffset, Vector3 CameraEyeOffset, bool ForceMouseLook) { AvatarSitResponsePacket avatarSitResponse = new AvatarSitResponsePacket(); avatarSitResponse.SitObject.ID = TargetID; avatarSitResponse.SitTransform.CameraAtOffset = CameraAtOffset; avatarSitResponse.SitTransform.CameraEyeOffset = CameraEyeOffset; avatarSitResponse.SitTransform.ForceMouselook = ForceMouseLook; avatarSitResponse.SitTransform.AutoPilot = autopilot; avatarSitResponse.SitTransform.SitPosition = OffsetPos; avatarSitResponse.SitTransform.SitRotation = SitOrientation; OutPacket(avatarSitResponse, ThrottleOutPacketType.Task); } public void SendAdminResponse(UUID Token, uint AdminLevel) { GrantGodlikePowersPacket respondPacket = new GrantGodlikePowersPacket(); GrantGodlikePowersPacket.GrantDataBlock gdb = new GrantGodlikePowersPacket.GrantDataBlock(); GrantGodlikePowersPacket.AgentDataBlock adb = new GrantGodlikePowersPacket.AgentDataBlock(); adb.AgentID = AgentId; adb.SessionID = SessionId; // More security gdb.GodLevel = (byte)AdminLevel; gdb.Token = Token; //respondPacket.AgentData = (GrantGodlikePowersPacket.AgentDataBlock)ablock; respondPacket.GrantData = gdb; respondPacket.AgentData = adb; OutPacket(respondPacket, ThrottleOutPacketType.Task); } public void SendGroupMembership(GroupMembershipData[] GroupMembership) { UpdateGroupMembership(GroupMembership); SendAgentGroupDataUpdate(AgentId,GroupMembership); } public void SendSelectedPartsProprieties(List parts) { /* not in use // udp part ObjectPropertiesPacket packet = (ObjectPropertiesPacket)PacketPool.Instance.GetPacket(PacketType.ObjectProperties); ObjectPropertiesPacket.ObjectDataBlock[] ObjectData = new ObjectPropertiesPacket.ObjectDataBlock[parts.Count]; int i = 0; foreach(SceneObjectPart sop in parts) ObjectData[i++] = CreateObjectPropertiesBlock(sop); packet.ObjectData = ObjectData; packet.Header.Zerocoded = true; // udp send splits this mega packets correctly // mb later will avoid that to reduce gc stress OutPacket(packet, ThrottleOutPacketType.Task, true); // caps physics part IEventQueue eq = Scene.RequestModuleInterface(); if(eq == null) return; OSDArray array = new OSDArray(); foreach(SceneObjectPart sop in parts) { OSDMap physinfo = new OSDMap(6); physinfo["LocalID"] = sop.LocalId; physinfo["Density"] = sop.Density; physinfo["Friction"] = sop.Friction; physinfo["GravityMultiplier"] = sop.GravityModifier; physinfo["Restitution"] = sop.Restitution; physinfo["PhysicsShapeType"] = (int)sop.PhysicsShapeType; array.Add(physinfo); } OSDMap llsdBody = new OSDMap(1); llsdBody.Add("ObjectData", array); eq.Enqueue(BuildEvent("ObjectPhysicsProperties", llsdBody),AgentId); */ } public void SendPartPhysicsProprieties(ISceneEntity entity) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) return; SceneObjectPart part = (SceneObjectPart)entity; if (part == null) return; uint localid = part.LocalId; byte physshapetype = part.PhysicsShapeType; float density = part.Density; float friction = part.Friction; float bounce = part.Restitution; float gravmod = part.GravityModifier; eq.partPhysicsProperties(localid, physshapetype, density, friction, bounce, gravmod,AgentId); } public void SendGroupNameReply(UUID groupLLUID, string GroupName) { UUIDGroupNameReplyPacket pack = new UUIDGroupNameReplyPacket(); UUIDGroupNameReplyPacket.UUIDNameBlockBlock[] uidnameblock = new UUIDGroupNameReplyPacket.UUIDNameBlockBlock[1]; UUIDGroupNameReplyPacket.UUIDNameBlockBlock uidnamebloc = new UUIDGroupNameReplyPacket.UUIDNameBlockBlock(); uidnamebloc.ID = groupLLUID; uidnamebloc.GroupName = Util.StringToBytes256(GroupName); uidnameblock[0] = uidnamebloc; pack.UUIDNameBlock = uidnameblock; OutPacket(pack, ThrottleOutPacketType.Task); } public void SendLandStatReply(uint reportType, uint requestFlags, uint resultCount, LandStatReportItem[] lsrpia) { LandStatReplyPacket lsrp = new LandStatReplyPacket(); // LandStatReplyPacket.RequestDataBlock lsreqdpb = new LandStatReplyPacket.RequestDataBlock(); LandStatReplyPacket.ReportDataBlock[] lsrepdba = new LandStatReplyPacket.ReportDataBlock[lsrpia.Length]; //LandStatReplyPacket.ReportDataBlock lsrepdb = new LandStatReplyPacket.ReportDataBlock(); // lsrepdb. lsrp.RequestData.ReportType = reportType; lsrp.RequestData.RequestFlags = requestFlags; lsrp.RequestData.TotalObjectCount = resultCount; for (int i = 0; i < lsrpia.Length; i++) { LandStatReplyPacket.ReportDataBlock lsrepdb = new LandStatReplyPacket.ReportDataBlock(); lsrepdb.LocationX = lsrpia[i].LocationX; lsrepdb.LocationY = lsrpia[i].LocationY; lsrepdb.LocationZ = lsrpia[i].LocationZ; lsrepdb.Score = lsrpia[i].Score; lsrepdb.TaskID = lsrpia[i].TaskID; lsrepdb.TaskLocalID = lsrpia[i].TaskLocalID; lsrepdb.TaskName = Util.StringToBytes256(lsrpia[i].TaskName); lsrepdb.OwnerName = Util.StringToBytes256(lsrpia[i].OwnerName); lsrepdba[i] = lsrepdb; } lsrp.ReportData = lsrepdba; OutPacket(lsrp, ThrottleOutPacketType.Task); } public void SendScriptRunningReply(UUID objectID, UUID itemID, bool running) { ScriptRunningReplyPacket scriptRunningReply = new ScriptRunningReplyPacket(); scriptRunningReply.Script.ObjectID = objectID; scriptRunningReply.Script.ItemID = itemID; scriptRunningReply.Script.Running = running; OutPacket(scriptRunningReply, ThrottleOutPacketType.Task); } public void SendAsset(AssetRequestToClient req) { if (req.AssetInf == null) { m_log.ErrorFormat("{0} Cannot send asset {1} ({2}), asset is null", LogHeader); return; } if (req.AssetInf.Data == null) { m_log.ErrorFormat("{0} Cannot send asset {1} ({2}), asset data is null", LogHeader, req.AssetInf.ID, req.AssetInf.Metadata.ContentType); return; } bool isWearable = false; isWearable = ((AssetType) req.AssetInf.Type == AssetType.Bodypart || (AssetType) req.AssetInf.Type == AssetType.Clothing); //m_log.Debug("sending asset " + req.RequestAssetID + ", iswearable: " + isWearable); //if (isWearable) // m_log.Debug((AssetType)req.AssetInf.Type); TransferInfoPacket Transfer = new TransferInfoPacket(); Transfer.TransferInfo.ChannelType = 2; Transfer.TransferInfo.Status = 0; Transfer.TransferInfo.TargetType = 0; if (req.AssetRequestSource == 2) { Transfer.TransferInfo.Params = new byte[20]; Array.Copy(req.RequestAssetID.GetBytes(), 0, Transfer.TransferInfo.Params, 0, 16); int assType = req.AssetInf.Type; Array.Copy(Utils.IntToBytes(assType), 0, Transfer.TransferInfo.Params, 16, 4); } else if (req.AssetRequestSource == 3) { Transfer.TransferInfo.Params = req.Params; // Transfer.TransferInfo.Params = new byte[100]; //Array.Copy(req.RequestUser.AgentId.GetBytes(), 0, Transfer.TransferInfo.Params, 0, 16); //Array.Copy(req.RequestUser.SessionId.GetBytes(), 0, Transfer.TransferInfo.Params, 16, 16); } Transfer.TransferInfo.Size = req.AssetInf.Data.Length; Transfer.TransferInfo.TransferID = req.TransferRequestID; Transfer.Header.Zerocoded = true; OutPacket(Transfer, isWearable ? ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority : ThrottleOutPacketType.Asset); if (req.NumPackets == 1) { TransferPacketPacket TransferPacket = new TransferPacketPacket(); TransferPacket.TransferData.Packet = 0; TransferPacket.TransferData.ChannelType = 2; TransferPacket.TransferData.TransferID = req.TransferRequestID; TransferPacket.TransferData.Data = req.AssetInf.Data; TransferPacket.TransferData.Status = 1; TransferPacket.Header.Zerocoded = true; OutPacket(TransferPacket, isWearable ? ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority : ThrottleOutPacketType.Asset); } else { int processedLength = 0; // int maxChunkSize = Settings.MAX_PACKET_SIZE - 100; int maxChunkSize = (int) MaxTransferBytesPerPacket; int packetNumber = 0; while (processedLength < req.AssetInf.Data.Length) { TransferPacketPacket TransferPacket = new TransferPacketPacket(); TransferPacket.TransferData.Packet = packetNumber; TransferPacket.TransferData.ChannelType = 2; TransferPacket.TransferData.TransferID = req.TransferRequestID; int chunkSize = Math.Min(req.AssetInf.Data.Length - processedLength, maxChunkSize); byte[] chunk = new byte[chunkSize]; Array.Copy(req.AssetInf.Data, processedLength, chunk, 0, chunk.Length); TransferPacket.TransferData.Data = chunk; // 0 indicates more packets to come, 1 indicates last packet if (req.AssetInf.Data.Length - processedLength > maxChunkSize) { TransferPacket.TransferData.Status = 0; } else { TransferPacket.TransferData.Status = 1; } TransferPacket.Header.Zerocoded = true; OutPacket(TransferPacket, isWearable ? ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority : ThrottleOutPacketType.Asset); processedLength += chunkSize; packetNumber++; } } } public void SendAssetNotFound(AssetRequestToClient req) { TransferInfoPacket Transfer = new TransferInfoPacket(); Transfer.TransferInfo.ChannelType = 2; Transfer.TransferInfo.Status = -2; Transfer.TransferInfo.TargetType = 0; Transfer.TransferInfo.Params = req.Params; Transfer.TransferInfo.Size = 0; Transfer.TransferInfo.TransferID = req.TransferRequestID; Transfer.Header.Zerocoded = true; OutPacket(Transfer, ThrottleOutPacketType.Asset); } public void SendTexture(AssetBase TextureAsset) { } public void SendRegionHandle(UUID regionID, ulong handle) { RegionIDAndHandleReplyPacket reply = (RegionIDAndHandleReplyPacket)PacketPool.Instance.GetPacket(PacketType.RegionIDAndHandleReply); reply.ReplyBlock.RegionID = regionID; reply.ReplyBlock.RegionHandle = handle; OutPacket(reply, ThrottleOutPacketType.Land); } public void SendParcelInfo(RegionInfo info, LandData land, UUID parcelID, uint x, uint y) { ParcelInfoReplyPacket reply = (ParcelInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.ParcelInfoReply); reply.AgentData.AgentID = m_agentId; reply.Data.ParcelID = parcelID; reply.Data.OwnerID = land.OwnerID; reply.Data.Name = Utils.StringToBytes(land.Name); if (land.Description != null && land.Description != String.Empty) reply.Data.Desc = Utils.StringToBytes(land.Description.Substring(0, land.Description.Length > 254 ? 254: land.Description.Length)); else reply.Data.Desc = new Byte[0]; reply.Data.ActualArea = land.Area; reply.Data.BillableArea = land.Area; // TODO: what is this? reply.Data.Flags = (byte)Util.ConvertAccessLevelToMaturity((byte)info.AccessLevel); if((land.Flags & (uint)ParcelFlags.ForSale) != 0) reply.Data.Flags |= (byte)((1 << 7)); Vector3 pos = land.UserLocation; if (pos.Equals(Vector3.Zero)) { pos = (land.AABBMax + land.AABBMin) * 0.5f; } reply.Data.GlobalX = info.RegionLocX + x; reply.Data.GlobalY = info.RegionLocY + y; reply.Data.GlobalZ = pos.Z; reply.Data.SimName = Utils.StringToBytes(info.RegionName); reply.Data.SnapshotID = land.SnapshotID; reply.Data.Dwell = land.Dwell; reply.Data.SalePrice = land.SalePrice; reply.Data.AuctionID = (int)land.AuctionID; OutPacket(reply, ThrottleOutPacketType.Land); } public void SendScriptTeleportRequest(string objName, string simName, Vector3 pos, Vector3 lookAt) { ScriptTeleportRequestPacket packet = (ScriptTeleportRequestPacket)PacketPool.Instance.GetPacket(PacketType.ScriptTeleportRequest); packet.Data.ObjectName = Utils.StringToBytes(objName); packet.Data.SimName = Utils.StringToBytes(simName); packet.Data.SimPosition = pos; packet.Data.LookAt = lookAt; OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirPlacesReply(UUID queryID, DirPlacesReplyData[] data) { DirPlacesReplyPacket packet = (DirPlacesReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirPlacesReply); packet.AgentData = new DirPlacesReplyPacket.AgentDataBlock(); packet.QueryData = new DirPlacesReplyPacket.QueryDataBlock[1]; packet.QueryData[0] = new DirPlacesReplyPacket.QueryDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData[0].QueryID = queryID; DirPlacesReplyPacket.QueryRepliesBlock[] replies = new DirPlacesReplyPacket.QueryRepliesBlock[0]; DirPlacesReplyPacket.StatusDataBlock[] status = new DirPlacesReplyPacket.StatusDataBlock[0]; packet.QueryReplies = replies; packet.StatusData = status; foreach (DirPlacesReplyData d in data) { int idx = replies.Length; Array.Resize(ref replies, idx + 1); Array.Resize(ref status, idx + 1); replies[idx] = new DirPlacesReplyPacket.QueryRepliesBlock(); status[idx] = new DirPlacesReplyPacket.StatusDataBlock(); replies[idx].ParcelID = d.parcelID; replies[idx].Name = Utils.StringToBytes(d.name); replies[idx].ForSale = d.forSale; replies[idx].Auction = d.auction; replies[idx].Dwell = d.dwell; status[idx].Status = d.Status; packet.QueryReplies = replies; packet.StatusData = status; if (packet.Length >= 1000) { OutPacket(packet, ThrottleOutPacketType.Task); packet = (DirPlacesReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirPlacesReply); packet.AgentData = new DirPlacesReplyPacket.AgentDataBlock(); packet.QueryData = new DirPlacesReplyPacket.QueryDataBlock[1]; packet.QueryData[0] = new DirPlacesReplyPacket.QueryDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData[0].QueryID = queryID; replies = new DirPlacesReplyPacket.QueryRepliesBlock[0]; status = new DirPlacesReplyPacket.StatusDataBlock[0]; } } if (replies.Length > 0 || data.Length == 0) OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirPeopleReply(UUID queryID, DirPeopleReplyData[] data) { DirPeopleReplyPacket packet = (DirPeopleReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirPeopleReply); packet.AgentData = new DirPeopleReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirPeopleReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirPeopleReplyPacket.QueryRepliesBlock[ data.Length]; int i = 0; foreach (DirPeopleReplyData d in data) { packet.QueryReplies[i] = new DirPeopleReplyPacket.QueryRepliesBlock(); packet.QueryReplies[i].AgentID = d.agentID; packet.QueryReplies[i].FirstName = Utils.StringToBytes(d.firstName); packet.QueryReplies[i].LastName = Utils.StringToBytes(d.lastName); packet.QueryReplies[i].Group = Utils.StringToBytes(d.group); packet.QueryReplies[i].Online = d.online; packet.QueryReplies[i].Reputation = d.reputation; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirEventsReply(UUID queryID, DirEventsReplyData[] data) { DirEventsReplyPacket packet = (DirEventsReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirEventsReply); packet.AgentData = new DirEventsReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirEventsReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirEventsReplyPacket.QueryRepliesBlock[ data.Length]; packet.StatusData = new DirEventsReplyPacket.StatusDataBlock[ data.Length]; int i = 0; foreach (DirEventsReplyData d in data) { packet.QueryReplies[i] = new DirEventsReplyPacket.QueryRepliesBlock(); packet.StatusData[i] = new DirEventsReplyPacket.StatusDataBlock(); packet.QueryReplies[i].OwnerID = d.ownerID; packet.QueryReplies[i].Name = Utils.StringToBytes(d.name); packet.QueryReplies[i].EventID = d.eventID; packet.QueryReplies[i].Date = Utils.StringToBytes(d.date); packet.QueryReplies[i].UnixTime = d.unixTime; packet.QueryReplies[i].EventFlags = d.eventFlags; packet.StatusData[i].Status = d.Status; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirGroupsReply(UUID queryID, DirGroupsReplyData[] data) { DirGroupsReplyPacket packet = (DirGroupsReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirGroupsReply); packet.AgentData = new DirGroupsReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirGroupsReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirGroupsReplyPacket.QueryRepliesBlock[ data.Length]; int i = 0; foreach (DirGroupsReplyData d in data) { packet.QueryReplies[i] = new DirGroupsReplyPacket.QueryRepliesBlock(); packet.QueryReplies[i].GroupID = d.groupID; packet.QueryReplies[i].GroupName = Utils.StringToBytes(d.groupName); packet.QueryReplies[i].Members = d.members; packet.QueryReplies[i].SearchOrder = d.searchOrder; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirClassifiedReply(UUID queryID, DirClassifiedReplyData[] data) { DirClassifiedReplyPacket packet = (DirClassifiedReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirClassifiedReply); packet.AgentData = new DirClassifiedReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirClassifiedReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirClassifiedReplyPacket.QueryRepliesBlock[ data.Length]; packet.StatusData = new DirClassifiedReplyPacket.StatusDataBlock[ data.Length]; int i = 0; foreach (DirClassifiedReplyData d in data) { packet.QueryReplies[i] = new DirClassifiedReplyPacket.QueryRepliesBlock(); packet.StatusData[i] = new DirClassifiedReplyPacket.StatusDataBlock(); packet.QueryReplies[i].ClassifiedID = d.classifiedID; packet.QueryReplies[i].Name = Utils.StringToBytes(d.name); packet.QueryReplies[i].ClassifiedFlags = d.classifiedFlags; packet.QueryReplies[i].CreationDate = d.creationDate; packet.QueryReplies[i].ExpirationDate = d.expirationDate; packet.QueryReplies[i].PriceForListing = d.price; packet.StatusData[i].Status = d.Status; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirLandReply(UUID queryID, DirLandReplyData[] data) { DirLandReplyPacket packet = (DirLandReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirLandReply); packet.AgentData = new DirLandReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirLandReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirLandReplyPacket.QueryRepliesBlock[ data.Length]; int i = 0; foreach (DirLandReplyData d in data) { packet.QueryReplies[i] = new DirLandReplyPacket.QueryRepliesBlock(); packet.QueryReplies[i].ParcelID = d.parcelID; packet.QueryReplies[i].Name = Utils.StringToBytes(d.name); packet.QueryReplies[i].Auction = d.auction; packet.QueryReplies[i].ForSale = d.forSale; packet.QueryReplies[i].SalePrice = d.salePrice; packet.QueryReplies[i].ActualArea = d.actualArea; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirPopularReply(UUID queryID, DirPopularReplyData[] data) { DirPopularReplyPacket packet = (DirPopularReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirPopularReply); packet.AgentData = new DirPopularReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.QueryData = new DirPopularReplyPacket.QueryDataBlock(); packet.QueryData.QueryID = queryID; packet.QueryReplies = new DirPopularReplyPacket.QueryRepliesBlock[ data.Length]; int i = 0; foreach (DirPopularReplyData d in data) { packet.QueryReplies[i] = new DirPopularReplyPacket.QueryRepliesBlock(); packet.QueryReplies[i].ParcelID = d.parcelID; packet.QueryReplies[i].Name = Utils.StringToBytes(d.name); packet.QueryReplies[i].Dwell = d.dwell; i++; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendEventInfoReply(EventData data) { EventInfoReplyPacket packet = (EventInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.EventInfoReply); packet.AgentData = new EventInfoReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.EventData = new EventInfoReplyPacket.EventDataBlock(); packet.EventData.EventID = data.eventID; packet.EventData.Creator = Utils.StringToBytes(data.creator); packet.EventData.Name = Utils.StringToBytes(data.name); packet.EventData.Category = Utils.StringToBytes(data.category); packet.EventData.Desc = Utils.StringToBytes(data.description); packet.EventData.Date = Utils.StringToBytes(data.date); packet.EventData.DateUTC = data.dateUTC; packet.EventData.Duration = data.duration; packet.EventData.Cover = data.cover; packet.EventData.Amount = data.amount; packet.EventData.SimName = Utils.StringToBytes(data.simName); packet.EventData.GlobalPos = new Vector3d(data.globalPos); packet.EventData.EventFlags = data.eventFlags; OutPacket(packet, ThrottleOutPacketType.Task); } public void SendOfferCallingCard(UUID srcID, UUID transactionID) { // a bit special, as this uses AgentID to store the source instead // of the destination. The destination (the receiver) goes into destID OfferCallingCardPacket p = (OfferCallingCardPacket)PacketPool.Instance.GetPacket(PacketType.OfferCallingCard); p.AgentData.AgentID = srcID; p.AgentData.SessionID = UUID.Zero; p.AgentBlock.DestID = AgentId; p.AgentBlock.TransactionID = transactionID; OutPacket(p, ThrottleOutPacketType.Task); } public void SendAcceptCallingCard(UUID transactionID) { AcceptCallingCardPacket p = (AcceptCallingCardPacket)PacketPool.Instance.GetPacket(PacketType.AcceptCallingCard); p.AgentData.AgentID = AgentId; p.AgentData.SessionID = UUID.Zero; p.FolderData = new AcceptCallingCardPacket.FolderDataBlock[1]; p.FolderData[0] = new AcceptCallingCardPacket.FolderDataBlock(); p.FolderData[0].FolderID = UUID.Zero; OutPacket(p, ThrottleOutPacketType.Task); } public void SendDeclineCallingCard(UUID transactionID) { DeclineCallingCardPacket p = (DeclineCallingCardPacket)PacketPool.Instance.GetPacket(PacketType.DeclineCallingCard); p.AgentData.AgentID = AgentId; p.AgentData.SessionID = UUID.Zero; p.TransactionBlock.TransactionID = transactionID; OutPacket(p, ThrottleOutPacketType.Task); } public void SendTerminateFriend(UUID exFriendID) { TerminateFriendshipPacket p = (TerminateFriendshipPacket)PacketPool.Instance.GetPacket(PacketType.TerminateFriendship); p.AgentData.AgentID = AgentId; p.AgentData.SessionID = SessionId; p.ExBlock.OtherID = exFriendID; OutPacket(p, ThrottleOutPacketType.Task); } public void SendAvatarGroupsReply(UUID avatarID, GroupMembershipData[] data) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) return; // message template has a GroupData field AcceptNotices ignored by viewers // and a array NewGroupData also ignored StringBuilder sb = eq.StartEvent("AvatarGroupsReply"); LLSDxmlEncode.AddArrayAndMap("AgentData", sb); LLSDxmlEncode.AddElem("AgentID", AgentId, sb); LLSDxmlEncode.AddElem("AvatarID", avatarID, sb); LLSDxmlEncode.AddEndMapAndArray(sb); if(data.Length == 0) LLSDxmlEncode.AddEmptyArray("GroupData", sb); else { LLSDxmlEncode.AddArray("GroupData", sb); foreach (GroupMembershipData m in data) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("GroupPowers", m.GroupPowers, sb); LLSDxmlEncode.AddElem("GroupTitle", m.GroupTitle, sb); LLSDxmlEncode.AddElem("GroupID",m.GroupID, sb); LLSDxmlEncode.AddElem("GroupName", m.GroupName, sb); LLSDxmlEncode.AddElem("GroupInsigniaID", m.GroupPicture, sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); } OSD ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } public void SendAgentGroupDataUpdate(UUID avatarID, GroupMembershipData[] data) { if(avatarID != AgentId) m_log.Debug("[CLIENT]: SendAgentGroupDataUpdate avatarID != AgentId"); IEventQueue eq = this.Scene.RequestModuleInterface(); if(eq != null) { eq.GroupMembershipData(avatarID,data); } else { // use UDP if no caps AgentGroupDataUpdatePacket Groupupdate = new AgentGroupDataUpdatePacket(); AgentGroupDataUpdatePacket.GroupDataBlock[] Groups = new AgentGroupDataUpdatePacket.GroupDataBlock[data.Length]; for (int i = 0; i < data.Length; i++) { AgentGroupDataUpdatePacket.GroupDataBlock Group = new AgentGroupDataUpdatePacket.GroupDataBlock(); Group.AcceptNotices = data[i].AcceptNotices; Group.Contribution = data[i].Contribution; Group.GroupID = data[i].GroupID; Group.GroupInsigniaID = data[i].GroupPicture; Group.GroupName = Util.StringToBytes256(data[i].GroupName); Group.GroupPowers = data[i].GroupPowers; Groups[i] = Group; } Groupupdate.GroupData = Groups; Groupupdate.AgentData = new AgentGroupDataUpdatePacket.AgentDataBlock(); Groupupdate.AgentData.AgentID = avatarID; OutPacket(Groupupdate, ThrottleOutPacketType.Task); } } public void SendJoinGroupReply(UUID groupID, bool success) { JoinGroupReplyPacket p = (JoinGroupReplyPacket)PacketPool.Instance.GetPacket(PacketType.JoinGroupReply); p.AgentData = new JoinGroupReplyPacket.AgentDataBlock(); p.AgentData.AgentID = AgentId; p.GroupData = new JoinGroupReplyPacket.GroupDataBlock(); p.GroupData.GroupID = groupID; p.GroupData.Success = success; OutPacket(p, ThrottleOutPacketType.Task); } public void SendEjectGroupMemberReply(UUID agentID, UUID groupID, bool success) { EjectGroupMemberReplyPacket p = (EjectGroupMemberReplyPacket)PacketPool.Instance.GetPacket(PacketType.EjectGroupMemberReply); p.AgentData = new EjectGroupMemberReplyPacket.AgentDataBlock(); p.AgentData.AgentID = agentID; p.GroupData = new EjectGroupMemberReplyPacket.GroupDataBlock(); p.GroupData.GroupID = groupID; p.EjectData = new EjectGroupMemberReplyPacket.EjectDataBlock(); p.EjectData.Success = success; OutPacket(p, ThrottleOutPacketType.Task); } public void SendLeaveGroupReply(UUID groupID, bool success) { LeaveGroupReplyPacket p = (LeaveGroupReplyPacket)PacketPool.Instance.GetPacket(PacketType.LeaveGroupReply); p.AgentData = new LeaveGroupReplyPacket.AgentDataBlock(); p.AgentData.AgentID = AgentId; p.GroupData = new LeaveGroupReplyPacket.GroupDataBlock(); p.GroupData.GroupID = groupID; p.GroupData.Success = success; OutPacket(p, ThrottleOutPacketType.Task); } public void SendAvatarClassifiedReply(UUID targetID, UUID[] classifiedID, string[] name) { if (classifiedID.Length != name.Length) return; AvatarClassifiedReplyPacket ac = (AvatarClassifiedReplyPacket)PacketPool.Instance.GetPacket( PacketType.AvatarClassifiedReply); ac.AgentData = new AvatarClassifiedReplyPacket.AgentDataBlock(); ac.AgentData.AgentID = AgentId; ac.AgentData.TargetID = targetID; ac.Data = new AvatarClassifiedReplyPacket.DataBlock[classifiedID.Length]; int i; for (i = 0; i < classifiedID.Length; i++) { ac.Data[i].ClassifiedID = classifiedID[i]; ac.Data[i].Name = Utils.StringToBytes(name[i]); } OutPacket(ac, ThrottleOutPacketType.Task); } 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) { ClassifiedInfoReplyPacket cr = (ClassifiedInfoReplyPacket)PacketPool.Instance.GetPacket( PacketType.ClassifiedInfoReply); cr.AgentData = new ClassifiedInfoReplyPacket.AgentDataBlock(); cr.AgentData.AgentID = AgentId; cr.Data = new ClassifiedInfoReplyPacket.DataBlock(); cr.Data.ClassifiedID = classifiedID; cr.Data.CreatorID = creatorID; cr.Data.CreationDate = creationDate; cr.Data.ExpirationDate = expirationDate; cr.Data.Category = category; cr.Data.Name = Utils.StringToBytes(name); cr.Data.Desc = Utils.StringToBytes(description); cr.Data.ParcelID = parcelID; cr.Data.ParentEstate = parentEstate; cr.Data.SnapshotID = snapshotID; cr.Data.SimName = Utils.StringToBytes(simName); cr.Data.PosGlobal = new Vector3d(globalPos); cr.Data.ParcelName = Utils.StringToBytes(parcelName); cr.Data.ClassifiedFlags = classifiedFlags; cr.Data.PriceForListing = price; OutPacket(cr, ThrottleOutPacketType.Task); } public void SendAgentDropGroup(UUID groupID) { AgentDropGroupPacket dg = (AgentDropGroupPacket)PacketPool.Instance.GetPacket( PacketType.AgentDropGroup); dg.AgentData = new AgentDropGroupPacket.AgentDataBlock(); dg.AgentData.AgentID = AgentId; dg.AgentData.GroupID = groupID; OutPacket(dg, ThrottleOutPacketType.Task); } public void SendAvatarNotesReply(UUID targetID, string text) { AvatarNotesReplyPacket an = (AvatarNotesReplyPacket)PacketPool.Instance.GetPacket( PacketType.AvatarNotesReply); an.AgentData = new AvatarNotesReplyPacket.AgentDataBlock(); an.AgentData.AgentID = AgentId; an.Data = new AvatarNotesReplyPacket.DataBlock(); an.Data.TargetID = targetID; an.Data.Notes = Utils.StringToBytes(text); OutPacket(an, ThrottleOutPacketType.Task); } public void SendAvatarPicksReply(UUID targetID, Dictionary picks) { AvatarPicksReplyPacket ap = (AvatarPicksReplyPacket)PacketPool.Instance.GetPacket( PacketType.AvatarPicksReply); ap.AgentData = new AvatarPicksReplyPacket.AgentDataBlock(); ap.AgentData.AgentID = AgentId; ap.AgentData.TargetID = targetID; ap.Data = new AvatarPicksReplyPacket.DataBlock[picks.Count]; int i = 0; foreach (KeyValuePair pick in picks) { ap.Data[i] = new AvatarPicksReplyPacket.DataBlock(); ap.Data[i].PickID = pick.Key; ap.Data[i].PickName = Utils.StringToBytes(pick.Value); i++; } OutPacket(ap, ThrottleOutPacketType.Task); } public void SendAvatarClassifiedReply(UUID targetID, Dictionary classifieds) { AvatarClassifiedReplyPacket ac = (AvatarClassifiedReplyPacket)PacketPool.Instance.GetPacket( PacketType.AvatarClassifiedReply); ac.AgentData = new AvatarClassifiedReplyPacket.AgentDataBlock(); ac.AgentData.AgentID = AgentId; ac.AgentData.TargetID = targetID; ac.Data = new AvatarClassifiedReplyPacket.DataBlock[classifieds.Count]; int i = 0; foreach (KeyValuePair classified in classifieds) { ac.Data[i] = new AvatarClassifiedReplyPacket.DataBlock(); ac.Data[i].ClassifiedID = classified.Key; ac.Data[i].Name = Utils.StringToBytes(classified.Value); i++; } OutPacket(ac, ThrottleOutPacketType.Task); } public void SendParcelDwellReply(int localID, UUID parcelID, float dwell) { ParcelDwellReplyPacket pd = (ParcelDwellReplyPacket)PacketPool.Instance.GetPacket( PacketType.ParcelDwellReply); pd.AgentData = new ParcelDwellReplyPacket.AgentDataBlock(); pd.AgentData.AgentID = AgentId; pd.Data = new ParcelDwellReplyPacket.DataBlock(); pd.Data.LocalID = localID; pd.Data.ParcelID = parcelID; pd.Data.Dwell = dwell; OutPacket(pd, ThrottleOutPacketType.Land); } public void SendUserInfoReply(bool imViaEmail, bool visible, string email) { UserInfoReplyPacket ur = (UserInfoReplyPacket)PacketPool.Instance.GetPacket( PacketType.UserInfoReply); string Visible = "hidden"; if (visible) Visible = "default"; ur.AgentData = new UserInfoReplyPacket.AgentDataBlock(); ur.AgentData.AgentID = AgentId; ur.UserData = new UserInfoReplyPacket.UserDataBlock(); ur.UserData.IMViaEMail = imViaEmail; ur.UserData.DirectoryVisibility = Utils.StringToBytes(Visible); ur.UserData.EMail = Utils.StringToBytes(email); OutPacket(ur, ThrottleOutPacketType.Task); } public void SendCreateGroupReply(UUID groupID, bool success, string message) { CreateGroupReplyPacket createGroupReply = (CreateGroupReplyPacket)PacketPool.Instance.GetPacket(PacketType.CreateGroupReply); createGroupReply.AgentData = new CreateGroupReplyPacket.AgentDataBlock(); createGroupReply.ReplyData = new CreateGroupReplyPacket.ReplyDataBlock(); createGroupReply.AgentData.AgentID = AgentId; createGroupReply.ReplyData.GroupID = groupID; createGroupReply.ReplyData.Success = success; createGroupReply.ReplyData.Message = Utils.StringToBytes(message); OutPacket(createGroupReply, ThrottleOutPacketType.Task); } public void SendUseCachedMuteList() { UseCachedMuteListPacket useCachedMuteList = (UseCachedMuteListPacket)PacketPool.Instance.GetPacket(PacketType.UseCachedMuteList); useCachedMuteList.AgentData = new UseCachedMuteListPacket.AgentDataBlock(); useCachedMuteList.AgentData.AgentID = AgentId; OutPacket(useCachedMuteList, ThrottleOutPacketType.Task); } public void SendEmpytMuteList() { GenericMessagePacket gmp = new GenericMessagePacket(); gmp.AgentData.AgentID = AgentId; gmp.AgentData.SessionID = m_sessionId; gmp.AgentData.TransactionID = UUID.Zero; gmp.MethodData.Method = Util.StringToBytes256("emptymutelist"); gmp.ParamList = new GenericMessagePacket.ParamListBlock[1]; gmp.ParamList[0] = new GenericMessagePacket.ParamListBlock(); gmp.ParamList[0].Parameter = new byte[0]; OutPacket(gmp, ThrottleOutPacketType.Task); } public void SendMuteListUpdate(string filename) { MuteListUpdatePacket muteListUpdate = (MuteListUpdatePacket)PacketPool.Instance.GetPacket(PacketType.MuteListUpdate); muteListUpdate.MuteData = new MuteListUpdatePacket.MuteDataBlock(); muteListUpdate.MuteData.AgentID = AgentId; muteListUpdate.MuteData.Filename = Utils.StringToBytes(filename); OutPacket(muteListUpdate, ThrottleOutPacketType.Task); } 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) { PickInfoReplyPacket pickInfoReply = (PickInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.PickInfoReply); pickInfoReply.AgentData = new PickInfoReplyPacket.AgentDataBlock(); pickInfoReply.AgentData.AgentID = AgentId; pickInfoReply.Data = new PickInfoReplyPacket.DataBlock(); pickInfoReply.Data.PickID = pickID; pickInfoReply.Data.CreatorID = creatorID; pickInfoReply.Data.TopPick = topPick; pickInfoReply.Data.ParcelID = parcelID; pickInfoReply.Data.Name = Utils.StringToBytes(name); pickInfoReply.Data.Desc = Utils.StringToBytes(desc); pickInfoReply.Data.SnapshotID = snapshotID; pickInfoReply.Data.User = Utils.StringToBytes(user); pickInfoReply.Data.OriginalName = Utils.StringToBytes(originalName); pickInfoReply.Data.SimName = Utils.StringToBytes(simName); pickInfoReply.Data.PosGlobal = new Vector3d(posGlobal); pickInfoReply.Data.SortOrder = sortOrder; pickInfoReply.Data.Enabled = enabled; OutPacket(pickInfoReply, ThrottleOutPacketType.Task); } #endregion Scene/Avatar to Client // Gesture #region Appearance/ Wearables Methods public void SendWearables(AvatarWearable[] wearables, int serial) { AgentWearablesUpdatePacket aw = (AgentWearablesUpdatePacket)PacketPool.Instance.GetPacket(PacketType.AgentWearablesUpdate); aw.AgentData.AgentID = AgentId; aw.AgentData.SerialNum = (uint)serial; aw.AgentData.SessionID = m_sessionId; int count = 0; for (int i = 0; i < wearables.Length; i++) count += wearables[i].Count; // TODO: don't create new blocks if recycling an old packet aw.WearableData = new AgentWearablesUpdatePacket.WearableDataBlock[count]; AgentWearablesUpdatePacket.WearableDataBlock awb; int idx = 0; for (int i = 0; i < wearables.Length; i++) { for (int j = 0; j < wearables[i].Count; j++) { awb = new AgentWearablesUpdatePacket.WearableDataBlock(); awb.WearableType = (byte) i; awb.AssetID = wearables[i][j].AssetID; awb.ItemID = wearables[i][j].ItemID; aw.WearableData[idx] = awb; idx++; // m_log.DebugFormat( // "[APPEARANCE]: Sending wearable item/asset {0} {1} (index {2}) for {3}", // awb.ItemID, awb.AssetID, i, Name); } } OutPacket(aw, ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority); } static private readonly byte[] AvatarAppearanceHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 158 // ID 158 (low frequency bigendian) not zeroencoded //0xff, 0xff, 0, 1, 158 // ID 158 (low frequency bigendian) zeroencoded }; public void SendAppearance(UUID targetID, byte[] visualParams, byte[] textureEntry) { // doing post zero encode, because odds of beeing bad are not that low UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(AvatarAppearanceHeader, 0, buf.Data, 0, 10); byte[] data = buf.Data; int pos = 10; //sender block targetID.ToBytes(data, pos); pos += 16; data[pos++] = 0;// is trial = false // objectdata block ie texture int len = textureEntry.Length; if (len == 0) { data[pos++] = 0; data[pos++] = 0; } else { data[pos++] = (byte)len; data[pos++] = (byte)(len >> 8); Buffer.BlockCopy(textureEntry, 0, data, pos, len); pos += len; } // visual parameters len = visualParams.Length; data[pos++] = (byte)len; if(len > 0) Buffer.BlockCopy(visualParams, 0, data, pos, len); pos += len; // no AppearanceData data[pos++] = 0; // no AppearanceHover data[pos++] = 0; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority, null, false, true); } static private readonly byte[] AvatarAnimationHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 20 // ID (high frequency) }; public void SendAnimations(UUID[] animations, int[] seqs, UUID sourceAgentId, UUID[] objectIDs) { // m_log.DebugFormat("[LLCLIENTVIEW]: Sending animations for {0} to {1}", sourceAgentId, Name); UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(AvatarAnimationHeader, 0, data, 0, 7); //agent block sourceAgentId.ToBytes(data, 7); // animations count data[23] = (byte)animations.Length; int pos = 24; //self animations if (sourceAgentId == AgentId) { List withobjects = new List(animations.Length); List noobjects = new List(animations.Length); for (int i = 0; i < animations.Length; ++i) { if (objectIDs[i] == sourceAgentId || objectIDs[i] == UUID.Zero) noobjects.Add(i); else withobjects.Add(i); } // first the ones with corresponding objects foreach (int i in withobjects) { animations[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } // then the rest foreach (int i in noobjects) { animations[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } // object ids block data[pos++] = (byte)withobjects.Count; foreach (int i in withobjects) { objectIDs[i].ToBytes(data, pos); pos += 16; } } else { for(int i = 0; i < animations.Length; ++i) { animations[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } data[pos++] = 0; // no object ids } data[pos++] = 0; // no physical avatar events buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task | ThrottleOutPacketType.HighPriority, null, false, false); } public void SendObjectAnimations(UUID[] animations, int[] seqs, UUID senderId) { // m_log.DebugFormat("[LLCLIENTVIEW]: Sending Object animations for {0} to {1}", sourceAgentId, Name); if(!m_SupportObjectAnimations) return; ObjectAnimationPacket ani = (ObjectAnimationPacket)PacketPool.Instance.GetPacket(PacketType.ObjectAnimation); // TODO: don't create new blocks if recycling an old packet ani.Sender = new ObjectAnimationPacket.SenderBlock(); ani.Sender.ID = senderId; ani.AnimationList = new ObjectAnimationPacket.AnimationListBlock[animations.Length]; for (int i = 0; i < animations.Length; ++i) { ani.AnimationList[i] = new ObjectAnimationPacket.AnimationListBlock(); ani.AnimationList[i].AnimID = animations[i]; ani.AnimationList[i].AnimSequenceID = seqs[i]; } OutPacket(ani, ThrottleOutPacketType.Task); } #endregion #region Avatar Packet/Data Sending Methods /// /// Send an ObjectUpdate packet with information about an avatar /// public void SendEntityFullUpdateImmediate(ISceneEntity ent) { if (ent == null || (!(ent is ScenePresence) && !(ent is SceneObjectPart))) return; UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(objectUpdateHeader, 0, buf.Data, 0, 7); LLUDPZeroEncoder zc = new LLUDPZeroEncoder(buf.Data); zc.Position = 7; zc.AddUInt64(m_scene.RegionInfo.RegionHandle); zc.AddUInt16(Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f)); zc.AddByte(1); // block count ThrottleOutPacketType ptype = ThrottleOutPacketType.Task; if (ent is ScenePresence) { CreateAvatarUpdateBlock(ent as ScenePresence, zc); ptype |= ThrottleOutPacketType.HighPriority; } else CreatePrimUpdateBlock(ent as SceneObjectPart, (ScenePresence)SceneAgent, zc); buf.DataLength = zc.Finish(); m_udpServer.SendUDPPacket(m_udpClient, buf, ptype , null, false, false); } public void SendEntityTerseUpdateImmediate(ISceneEntity ent) { if (ent == null) return; UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); //setup header and regioninfo block Buffer.BlockCopy(terseUpdateHeader, 0, buf.Data, 0, 7); if (ent is ScenePresence) Utils.UInt64ToBytesSafepos(((ScenePresence)ent).RegionHandle, buf.Data, 7); else Utils.UInt64ToBytesSafepos(m_scene.RegionInfo.RegionHandle, buf.Data, 7); Utils.UInt16ToBytes(Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f), buf.Data, 15); buf.Data[17] = 1; int pos = 18; CreateImprovedTerseBlock(ent, buf.Data, ref pos, false); buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); } static private readonly byte[] CoarseLocationUpdateHeader = new byte[] { 0, // no acks plz 0, 0, 0, 0, // sequence number 0, // extra 0xff, 6 // ID 6 (medium frequency) }; public void SendCoarseLocationUpdate(List users, List CoarseLocations) { // We don't need to update inactive clients. if (!IsActive) return; int totalLocations = Math.Min(CoarseLocations.Count, 60); int totalAgents = Math.Min(users.Count, 60); if(totalAgents > totalLocations) totalAgents = totalLocations; int selfindex = -1; int preyindex = -1; UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(CoarseLocationUpdateHeader, 0, buf.Data, 0, 8); byte[] data = buf.Data; data[8] = (byte)totalLocations; int pos = 9; for (int i = 0; i < totalLocations; ++i) { data[pos++] = (byte)CoarseLocations[i].X; data[pos++] = (byte)CoarseLocations[i].Y; data[pos++] = CoarseLocations[i].Z > 1024 ? (byte)0 : (byte)(CoarseLocations[i].Z * 0.25f); if (i < totalAgents) { if (users[i] == AgentId) selfindex = i; //if (users[i] == PreyId) // preyindex = -1; } } Utils.Int16ToBytes((short)selfindex, data, pos); pos += 2; Utils.Int16ToBytes((short)preyindex, data, pos); pos += 2; data[pos++] = (byte)totalAgents; for (int i = 0; i < totalAgents; ++i) { users[i].ToBytes(data, pos); pos += 16; } buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, false); } #endregion Avatar Packet/Data Sending Methods #region Primitive Packet/Data Sending Methods /// /// Generate one of the object update packets based on PrimUpdateFlags /// and broadcast the packet to clients /// public void SendEntityUpdate(ISceneEntity entity, PrimUpdateFlags updateFlags) { /* if (entity.UUID == m_agentId && !updateFlags.HasFlag(PrimUpdateFlags.FullUpdate)) { ImprovedTerseObjectUpdatePacket packet = (ImprovedTerseObjectUpdatePacket)PacketPool.Instance.GetPacket(PacketType.ImprovedTerseObjectUpdate); packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; packet.RegionData.TimeDilation = Utils.FloatToUInt16(1, 0.0f, 1.0f); packet.ObjectData = new ImprovedTerseObjectUpdatePacket.ObjectDataBlock[1]; packet.ObjectData[0] = CreateImprovedTerseBlock(entity, false); OutPacket(packet, ThrottleOutPacketType.Unknown, true); return; } */ if (entity is SceneObjectPart) { SceneObjectPart p = (SceneObjectPart)entity; SceneObjectGroup g = p.ParentGroup; if (g.HasPrivateAttachmentPoint && g.OwnerID != AgentId) return; // Don't send updates for other people's HUDs if((updateFlags ^ PrimUpdateFlags.SendInTransit) == 0) { List partIDs = (new List {p.LocalId}); lock (m_entityProps.SyncRoot) m_entityProps.Remove(partIDs); lock (m_entityUpdates.SyncRoot) m_entityUpdates.Remove(partIDs); return; } } //double priority = m_prioritizer.GetUpdatePriority(this, entity); uint priority = m_prioritizer.GetUpdatePriority(this, entity); lock (m_entityUpdates.SyncRoot) m_entityUpdates.Enqueue(priority, new EntityUpdate(entity, updateFlags)); } /// /// 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. /// 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 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. /// private void ResendPrimUpdates(List updates, OutgoingPacket oPacket) { // m_log.WarnFormat("[CLIENT] resending prim updates {0}, packet sequence number {1}", updates[0].UpdateTime, oPacket.SequenceNumber); // 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); // 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); // We're not going to worry about interlock yet since its not currently critical that this total count // is 100% correct m_udpServer.PacketsResentCount++; foreach (EntityUpdate update in updates) ResendPrimUpdate(update); } static private readonly byte[] objectUpdateHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, 0, 0, 0, 0, // sequence number 0, // extra 12 // ID (high frequency) }; static private readonly byte[] terseUpdateHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, // zero code is not as spec 0, 0, 0, 0, // sequence number 0, // extra 15 // ID (high frequency) }; private void ProcessEntityUpdates(int maxUpdatesBytes) { if (!IsActive) return; ScenePresence mysp = (ScenePresence)SceneAgent; if (mysp == null) return; // List compressedUpdateBlocks = null; List objectUpdates = null; // List compressedUpdates = null; List terseUpdates = null; List ObjectAnimationUpdates = null; // Check to see if this is a flush if (maxUpdatesBytes <= 0) { maxUpdatesBytes = Int32.MaxValue; } EntityUpdate update; bool doCulling = m_scene.ObjectsCullingByDistance; float cullingrange = 64.0f; Vector3 mypos = Vector3.Zero; //bool orderedDequeue = m_scene.UpdatePrioritizationScheme == UpdatePrioritizationSchemes.SimpleAngularDistance; bool orderedDequeue = false; // temporary off HashSet GroupsNeedFullUpdate = new HashSet(); if (doCulling) { cullingrange = mysp.DrawDistance + m_scene.ReprioritizationDistance + 16f; mypos = mysp.AbsolutePosition; } while (maxUpdatesBytes > 0) { if (!IsActive) return; lock (m_entityUpdates.SyncRoot) { if(orderedDequeue) { if (!m_entityUpdates.TryOrderedDequeue(out update)) break; } else { if (!m_entityUpdates.TryDequeue(out update)) break; } } PrimUpdateFlags updateFlags = (PrimUpdateFlags)update.Flags; if(updateFlags.HasFlag(PrimUpdateFlags.Kill)) { m_killRecord.Add(update.Entity.LocalId); maxUpdatesBytes -= 30; continue; } if (update.Entity is SceneObjectPart) { SceneObjectPart part = (SceneObjectPart)update.Entity; SceneObjectGroup grp = part.ParentGroup; if (grp.inTransit && !update.Flags.HasFlag(PrimUpdateFlags.SendInTransit)) continue; if (grp.IsDeleted) { // Don't send updates for objects that have been marked deleted. // Instead send another kill object, because the first one may have gotten // into a race condition if (part == grp.RootPart && !m_killRecord.Contains(grp.LocalId)) { m_killRecord.Add(grp.LocalId); maxUpdatesBytes -= 30; } continue; } if (grp.IsAttachment) { // animated attachments are nasty if not supported by viewer if(!m_SupportObjectAnimations && grp.RootPart.Shape.MeshFlagEntry) continue; // Someone else's HUD, why are we getting these? if (grp.OwnerID != AgentId && grp.HasPrivateAttachmentPoint) continue; // if owner gone don't update it to anyone ScenePresence sp; if (!m_scene.TryGetScenePresence(part.OwnerID, out sp)) continue; // On vehicle crossing, the attachments are received // while the avatar is still a child. Don't send // updates here because the LocalId has not yet // been updated and the viewer will derender the // attachments until the avatar becomes root. if (sp.IsChildAgent) continue; // It's an attachment of a valid avatar, but // doesn't seem to be attached, skip List atts = sp.GetAttachments(); bool found = false; foreach (SceneObjectGroup att in atts) { if (att == grp) { found = true; break; } } if (!found) continue; if (m_disableFacelights) { if (grp.RootPart.Shape.State != (byte)AttachmentPoint.LeftHand && grp.RootPart.Shape.State != (byte)AttachmentPoint.RightHand) { part.Shape.LightEntry = false; } } } else if (doCulling) { if(GroupsNeedFullUpdate.Contains(grp)) continue; bool inViewGroups = false; lock(GroupsInView) inViewGroups = GroupsInView.Contains(grp); if(!inViewGroups) { Vector3 partpos = grp.getCenterOffset(); float dpos = (partpos - mypos).LengthSquared(); float maxview = grp.GetBoundsRadius() + cullingrange; if (dpos > maxview * maxview) continue; GroupsNeedFullUpdate.Add(grp); continue; } } } else if (update.Entity is ScenePresence) { ScenePresence presence = (ScenePresence)update.Entity; if (presence.IsDeleted) continue; // If ParentUUID is not UUID.Zero and ParentID is 0, this // avatar is in the process of crossing regions while // sat on an object. In this state, we don't want any // updates because they will visually orbit the avatar. // Update will be forced once crossing is completed anyway. if (presence.ParentUUID != UUID.Zero && presence.ParentID == 0) continue; } else // what is this update ? continue; #region UpdateFlags to packet type conversion // bool canUseCompressed = true; if (update.Entity is SceneObjectPart) { if (m_SupportObjectAnimations && updateFlags.HasFlag(PrimUpdateFlags.Animations)) { SceneObjectPart sop = (SceneObjectPart)update.Entity; if ( sop.Animations != null) { if(ObjectAnimationUpdates == null) ObjectAnimationUpdates = new List(); ObjectAnimationUpdates.Add(sop); maxUpdatesBytes -= 20 * sop.Animations.Count + 24; } } } else { // canUseCompressed = false; } updateFlags &= PrimUpdateFlags.FullUpdate; // clear other control bits already handled if(updateFlags == PrimUpdateFlags.None) continue; /* const PrimUpdateFlags canNotUseCompressedMask = PrimUpdateFlags.Velocity | PrimUpdateFlags.Acceleration | PrimUpdateFlags.CollisionPlane | PrimUpdateFlags.Joint; if ((updateFlags & canNotUseCompressedMask) != 0) { canUseCompressed = false; } */ const PrimUpdateFlags canNotUseImprovedMask = ~( PrimUpdateFlags.AttachmentPoint | PrimUpdateFlags.Position | PrimUpdateFlags.Rotation | PrimUpdateFlags.Velocity | PrimUpdateFlags.Acceleration | PrimUpdateFlags.AngularVelocity | PrimUpdateFlags.CollisionPlane | PrimUpdateFlags.Textures ); #endregion UpdateFlags to packet type conversion #region Block Construction // TODO: Remove this once we can build compressed updates /* if (canUseCompressed) { ObjectUpdateCompressedPacket.ObjectDataBlock ablock = CreateCompressedUpdateBlock((SceneObjectPart)update.Entity, updateFlags); compressedUpdateBlocks.Add(ablock); compressedUpdates.Value.Add(update); maxUpdatesBytes -= ablock.Length; } else if (canUseImproved) */ if ((updateFlags & canNotUseImprovedMask) == 0) { if (terseUpdates == null) { terseUpdates = new List(); maxUpdatesBytes -= 18; } terseUpdates.Add(update); if (update.Entity is ScenePresence) maxUpdatesBytes -= 63; // no texture entry else { if ((updateFlags & PrimUpdateFlags.Textures) == 0) maxUpdatesBytes -= 47; else maxUpdatesBytes -= 150; // aprox } } else { if (update.Entity is ScenePresence) maxUpdatesBytes -= 150; // crude estimation else maxUpdatesBytes -= 300; if(objectUpdates == null) { objectUpdates = new List(); maxUpdatesBytes -= 18; } objectUpdates.Add(update); } #endregion Block Construction } #region Packet Sending ushort timeDilation; if (!IsActive) return; timeDilation = Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f); if(objectUpdates != null) { int blocks = objectUpdates.Count; List tau = new List(30); UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(objectUpdateHeader, 0, buf.Data, 0, 7); LLUDPZeroEncoder zc = new LLUDPZeroEncoder(buf.Data); zc.Position = 7; zc.AddUInt64(m_scene.RegionInfo.RegionHandle); zc.AddUInt16(Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f)); zc.AddByte(1); // tmp block count int countposition = zc.Position - 1; int lastpos = 0; int lastzc = 0; int count = 0; foreach (EntityUpdate eu in objectUpdates) { lastpos = zc.Position; lastzc = zc.ZeroCount; if (eu.Entity is ScenePresence) CreateAvatarUpdateBlock((ScenePresence)eu.Entity, zc); else CreatePrimUpdateBlock((SceneObjectPart)eu.Entity, mysp, zc); if (zc.Position < LLUDPServer.MAXPAYLOAD) { tau.Add(eu); ++count; --blocks; } else if (blocks > 0) { // we need more packets UDPPacketBuffer newbuf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(buf.Data, 0, newbuf.Data, 0, countposition); // start is the same buf.Data[countposition] = (byte)count; // get pending zeros at cut point if(lastzc > 0) { buf.Data[lastpos++] = 0; buf.Data[lastpos++] = (byte)lastzc; } buf.DataLength = lastpos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, false); buf = newbuf; zc.Data = buf.Data; zc.ZeroCount = 0; zc.Position = countposition + 1; // im lazy now, just do last again if (eu.Entity is ScenePresence) CreateAvatarUpdateBlock((ScenePresence)eu.Entity, zc); else CreatePrimUpdateBlock((SceneObjectPart)eu.Entity, mysp, zc); tau = new List(30); tau.Add(eu); count = 1; --blocks; } } if (count > 0) { buf.Data[countposition] = (byte)count; buf.DataLength = zc.Finish(); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, false); } } /* if (compressedUpdateBlocks != null) { ObjectUpdateCompressedPacket packet = (ObjectUpdateCompressedPacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdateCompressed); packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; packet.RegionData.TimeDilation = timeDilation; packet.ObjectData = compressedUpdateBlocks.ToArray(); compressedUpdateBlocks.Clear(); OutPacket(packet, ThrottleOutPacketType.Task, true, delegate(OutgoingPacket oPacket) { ResendPrimUpdates(compressedUpdates, oPacket); }); } */ if (terseUpdates != null) { int blocks = terseUpdates.Count; List tau = new List(30); UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); //setup header and regioninfo block Buffer.BlockCopy(terseUpdateHeader, 0, buf.Data, 0, 7); Utils.UInt64ToBytesSafepos(m_scene.RegionInfo.RegionHandle, buf.Data, 7); Utils.UInt16ToBytes(timeDilation, buf.Data, 15); int pos = 18; int lastpos = 0; int count = 0; foreach (EntityUpdate eu in terseUpdates) { lastpos = pos; CreateImprovedTerseBlock(eu.Entity, buf.Data, ref pos, (eu.Flags & PrimUpdateFlags.Textures) != 0); if (pos < LLUDPServer.MAXPAYLOAD) { tau.Add(eu); ++count; --blocks; } else if (blocks > 0) { // we need more packets UDPPacketBuffer newbuf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(buf.Data, 0, newbuf.Data, 0, 17); // start is the same // copy what we done in excess int extralen = pos - lastpos; if(extralen > 0) Buffer.BlockCopy(buf.Data, lastpos, newbuf.Data, 18, extralen); pos = 18 + extralen; buf.Data[17] = (byte)count; buf.DataLength = lastpos; // zero encode is not as spec m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, true); tau = new List(30); tau.Add(eu); count = 1; --blocks; buf = newbuf; } } if (count > 0) { buf.Data[17] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, true); } } if (ObjectAnimationUpdates != null) { foreach (SceneObjectPart sop in ObjectAnimationUpdates) { if (sop.Animations == null) continue; SceneObjectGroup sog = sop.ParentGroup; if (sog == null || sog.IsDeleted) continue; SceneObjectPart root = sog.RootPart; if (root == null || root.Shape == null || !root.Shape.MeshFlagEntry) continue; UUID[] ids = null; int[] seqs = null; int count = sop.GetAnimations(out ids, out seqs); ObjectAnimationPacket ani = (ObjectAnimationPacket)PacketPool.Instance.GetPacket(PacketType.ObjectAnimation); ani.Sender = new ObjectAnimationPacket.SenderBlock(); ani.Sender.ID = sop.UUID; ani.AnimationList = new ObjectAnimationPacket.AnimationListBlock[count]; for(int i = 0; i< count; i++) { ani.AnimationList[i] = new ObjectAnimationPacket.AnimationListBlock(); ani.AnimationList[i].AnimID = ids[i]; ani.AnimationList[i].AnimSequenceID = seqs[i]; } OutPacket(ani, ThrottleOutPacketType.Task, true); } } #endregion Packet Sending #region Handle deleted objects if (m_killRecord.Count > 0) { SendKillObject(m_killRecord); m_killRecord.Clear(); } if(GroupsNeedFullUpdate.Count > 0) { foreach(SceneObjectGroup grp in GroupsNeedFullUpdate) { lock (GroupsInView) GroupsInView.Add(grp); foreach (SceneObjectPart p in grp.Parts) SendEntityUpdate(p, PrimUpdateFlags.CancelKill); } } #endregion } // hack.. dont use /* public void SendPartFullUpdate(ISceneEntity ent, uint? parentID) { if (ent is SceneObjectPart) { SceneObjectPart part = (SceneObjectPart)ent; ObjectUpdatePacket packet = (ObjectUpdatePacket)PacketPool.Instance.GetPacket(PacketType.ObjectUpdate); packet.RegionData.RegionHandle = m_scene.RegionInfo.RegionHandle; packet.RegionData.TimeDilation = Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f); packet.ObjectData = new ObjectUpdatePacket.ObjectDataBlock[1]; ObjectUpdatePacket.ObjectDataBlock blk = CreatePrimUpdateBlock(part, mysp); if (parentID.HasValue) { blk.ParentID = parentID.Value; } packet.ObjectData[0] = blk; OutPacket(packet, ThrottleOutPacketType.Task, true); } // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sent {0} updates in ProcessEntityUpdates() for {1} {2} in {3}", // updatesThisCall, Name, SceneAgent.IsChildAgent ? "child" : "root", Scene.Name); // } */ public void ReprioritizeUpdates() { lock (m_entityUpdates.SyncRoot) m_entityUpdates.Reprioritize(UpdatePriorityHandler); CheckGroupsInView(); } private bool CheckGroupsInViewBusy = false; public void CheckGroupsInView() { bool doCulling = m_scene.ObjectsCullingByDistance; if(!doCulling) return; if (!IsActive) return; if (CheckGroupsInViewBusy) return; ScenePresence mysp = (ScenePresence)SceneAgent; if (mysp == null || mysp.IsDeleted) return; CheckGroupsInViewBusy = true; float cullingrange = mysp.DrawDistance + m_scene.ReprioritizationDistance + 16f; Vector3 mypos = mysp.AbsolutePosition; HashSet NewGroupsInView = new HashSet(); HashSet GroupsNeedFullUpdate = new HashSet(); List kills = new List(); EntityBase[] entities = m_scene.Entities.GetEntities(); foreach (EntityBase e in entities) { if (!IsActive) return; if (e != null && e is SceneObjectGroup) { SceneObjectGroup grp = (SceneObjectGroup)e; if(grp.IsDeleted || grp.IsAttachment) continue; bool inviewgroups; lock (GroupsInView) inviewgroups = GroupsInView.Contains(grp); Vector3 grppos = grp.getCenterOffset(); float dpos = (grppos - mypos).LengthSquared(); float maxview = grp.GetBoundsRadius() + cullingrange; if (dpos > maxview * maxview) { if(inviewgroups) kills.Add(grp); } else { if(!inviewgroups) GroupsNeedFullUpdate.Add(grp); NewGroupsInView.Add(grp); } } } lock(GroupsInView) GroupsInView = NewGroupsInView; if (kills.Count > 0) { List partIDs = new List(); foreach(SceneObjectGroup grp in kills) { SendEntityUpdate(grp.RootPart, PrimUpdateFlags.Kill); foreach(SceneObjectPart p in grp.Parts) { if(p != grp.RootPart) partIDs.Add(p.LocalId); } } kills.Clear(); if(partIDs.Count > 0) { lock (m_entityProps.SyncRoot) m_entityProps.Remove(partIDs); lock (m_entityUpdates.SyncRoot) m_entityUpdates.Remove(partIDs); } } if(GroupsNeedFullUpdate.Count > 0) { foreach(SceneObjectGroup grp in GroupsNeedFullUpdate) { foreach(SceneObjectPart p in grp.Parts) SendEntityUpdate(p, PrimUpdateFlags.CancelKill); } } CheckGroupsInViewBusy = false; } private bool UpdatePriorityHandler(ref uint priority, ISceneEntity entity) { if (entity == null) return false; priority = m_prioritizer.GetUpdatePriority(this, entity); return true; } public void FlushPrimUpdates() { m_log.WarnFormat("[CLIENT]: Flushing prim updates to " + m_firstName + " " + m_lastName); while (m_entityUpdates.Count > 0) ProcessEntityUpdates(-1); } #endregion Primitive Packet/Data Sending Methods // These are used to implement an adaptive backoff in the number // of updates converted to packets. Since we don't want packets // to sit in the queue with old data, only convert enough updates // to packets that can be sent in 30ms. void HandleQueueEmpty(ThrottleOutPacketTypeFlags categories) { if(m_scene == null) return; if ((categories & ThrottleOutPacketTypeFlags.Task) != 0) { int maxUpdateBytes = m_udpClient.GetCatBytesCanSend(ThrottleOutPacketType.Task, 30); if (m_entityUpdates.Count > 0) ProcessEntityUpdates(maxUpdateBytes); if (m_entityProps.Count > 0) ProcessEntityPropertyRequests(maxUpdateBytes); } if ((categories & ThrottleOutPacketTypeFlags.Texture) != 0) ImageManager.ProcessImageQueue(m_udpServer.TextureSendLimit); } internal bool HandleHasUpdates(ThrottleOutPacketTypeFlags categories) { if ((categories & ThrottleOutPacketTypeFlags.Task) != 0) { if (m_entityUpdates.Count > 0) return true; if (m_entityProps.Count > 0) return true; } if ((categories & ThrottleOutPacketTypeFlags.Texture) != 0) { if (ImageManager.HasUpdates()) return true; } return false; } public void SendAssetUploadCompleteMessage(sbyte AssetType, bool Success, UUID AssetFullID) { AssetUploadCompletePacket newPack = new AssetUploadCompletePacket(); newPack.AssetBlock.Type = AssetType; newPack.AssetBlock.Success = Success; newPack.AssetBlock.UUID = AssetFullID; newPack.Header.Zerocoded = true; OutPacket(newPack, ThrottleOutPacketType.Asset); } public void SendXferRequest(ulong XferID, short AssetType, UUID vFileID, byte FilePath, byte[] FileName) { RequestXferPacket newPack = new RequestXferPacket(); newPack.XferID.ID = XferID; newPack.XferID.VFileType = AssetType; newPack.XferID.VFileID = vFileID; newPack.XferID.FilePath = FilePath; newPack.XferID.Filename = FileName; newPack.Header.Zerocoded = true; OutPacket(newPack, ThrottleOutPacketType.Asset); } public void SendConfirmXfer(ulong xferID, uint PacketID) { ConfirmXferPacketPacket newPack = new ConfirmXferPacketPacket(); newPack.XferID.ID = xferID; newPack.XferID.Packet = PacketID; newPack.Header.Zerocoded = true; OutPacket(newPack, ThrottleOutPacketType.Asset); } public void SendInitiateDownload(string simFileName, string clientFileName) { InitiateDownloadPacket newPack = new InitiateDownloadPacket(); newPack.AgentData.AgentID = AgentId; newPack.FileData.SimFilename = Utils.StringToBytes(simFileName); newPack.FileData.ViewerFilename = Utils.StringToBytes(clientFileName); OutPacket(newPack, ThrottleOutPacketType.Asset); } public void SendImageFirstPart( ushort numParts, UUID ImageUUID, uint ImageSize, byte[] ImageData, byte imageCodec) { ImageDataPacket im = new ImageDataPacket(); im.ImageID.Packets = numParts; im.ImageID.ID = ImageUUID; if (ImageSize > 0) im.ImageID.Size = ImageSize; im.ImageData.Data = ImageData; im.ImageID.Codec = imageCodec; im.Header.Zerocoded = true; OutPacket(im, ThrottleOutPacketType.Texture); } public void SendImageNextPart(ushort partNumber, UUID imageUuid, byte[] imageData) { ImagePacketPacket im = new ImagePacketPacket(); im.ImageID.Packet = partNumber; im.ImageID.ID = imageUuid; im.ImageData.Data = imageData; OutPacket(im, ThrottleOutPacketType.Texture); } public void SendImageNotFound(UUID imageid) { ImageNotInDatabasePacket notFoundPacket = (ImageNotInDatabasePacket)PacketPool.Instance.GetPacket(PacketType.ImageNotInDatabase); notFoundPacket.ImageID.ID = imageid; OutPacket(notFoundPacket, ThrottleOutPacketType.Texture); } public void SendShutdownConnectionNotice() { OutPacket(PacketPool.Instance.GetPacket(PacketType.DisableSimulator), ThrottleOutPacketType.Unknown); } public void SendSimStats(SimStats stats) { SimStatsPacket pack = new SimStatsPacket(); pack.Region = new SimStatsPacket.RegionBlock(); pack.Region.RegionX = stats.RegionX; pack.Region.RegionY = stats.RegionY; pack.Region.RegionFlags = stats.RegionFlags; pack.Region.ObjectCapacity = stats.ObjectCapacity; //pack.Region = //stats.RegionBlock; pack.Stat = stats.StatsBlock; pack.Header.Reliable = false; pack.RegionInfo = new SimStatsPacket.RegionInfoBlock[0]; OutPacket(pack, ThrottleOutPacketType.Task); } private class ObjectPropertyUpdate : EntityUpdate { internal bool SendFamilyProps; internal bool SendObjectProps; public ObjectPropertyUpdate(ISceneEntity entity, uint flags, bool sendfam, bool sendobj) : base(entity,(PrimUpdateFlags)flags) { SendFamilyProps = sendfam; SendObjectProps = sendobj; } public void Update(ObjectPropertyUpdate update) { SendFamilyProps = SendFamilyProps || update.SendFamilyProps; SendObjectProps = SendObjectProps || update.SendObjectProps; // other properties may need to be updated by base class base.Update(update); } } public void SendObjectPropertiesFamilyData(ISceneEntity entity, uint requestFlags) { uint priority = 0; // time based ordering only lock (m_entityProps.SyncRoot) 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, 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); // 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); // We're not going to worry about interlock yet since its not currently critical that this total count // is 100% correct m_udpServer.PacketsResentCount++; foreach (ObjectPropertyUpdate update in updates) ResendPropertyUpdate(update); } public void SendObjectPropertiesReply(ISceneEntity entity) { uint priority = 0; // time based ordering only lock (m_entityProps.SyncRoot) m_entityProps.Enqueue(priority, new ObjectPropertyUpdate(entity,0,false,true)); } static private readonly byte[] ObjectPropertyUpdateHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 9 // ID (medium frequency) }; static private readonly byte[] ObjectFamilyUpdateHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 10 // ID (medium frequency) }; private void ProcessEntityPropertyRequests(int maxUpdateBytes) { List objectPropertiesUpdates = null; List objectPropertiesFamilyUpdates = null; List needPhysics = null; // bool orderedDequeue = m_scene.UpdatePrioritizationScheme == UpdatePrioritizationSchemes.SimpleAngularDistance; bool orderedDequeue = false; // for now EntityUpdate iupdate; while (maxUpdateBytes > 0) { lock (m_entityProps.SyncRoot) { if(orderedDequeue) { if (!m_entityProps.TryOrderedDequeue(out iupdate)) break; } else { if (!m_entityProps.TryDequeue(out iupdate)) break; } } ObjectPropertyUpdate update = (ObjectPropertyUpdate)iupdate; if (update.SendFamilyProps) { if (update.Entity is SceneObjectPart) { SceneObjectPart sop = (SceneObjectPart)update.Entity; if(objectPropertiesFamilyUpdates == null) objectPropertiesFamilyUpdates = new List(); objectPropertiesFamilyUpdates.Add(update); maxUpdateBytes -= 100; } } if (update.SendObjectProps) { if (update.Entity is SceneObjectPart) { SceneObjectPart sop = (SceneObjectPart)update.Entity; if(needPhysics == null) needPhysics = new List(); needPhysics.Add(sop); if(objectPropertiesUpdates == null) objectPropertiesUpdates = new List(); objectPropertiesUpdates.Add(update); maxUpdateBytes -= 200; // aprox } } } if (objectPropertiesUpdates != null) { int blocks = objectPropertiesUpdates.Count; //List tau = new List(30); UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(ObjectPropertyUpdateHeader, 0, buf.Data, 0, 8); LLUDPZeroEncoder zc = new LLUDPZeroEncoder(buf.Data); zc.Position = 8; zc.AddByte(1); // tmp block count int countposition = zc.Position - 1; int lastpos = 0; int lastzc = 0; int count = 0; foreach (EntityUpdate eu in objectPropertiesUpdates) { lastpos = zc.Position; lastzc = zc.ZeroCount; CreateObjectPropertiesBlock((SceneObjectPart)eu.Entity, zc); if (zc.Position < LLUDPServer.MAXPAYLOAD) { //tau.Add(eu); ++count; --blocks; } else if (blocks > 0) { // we need more packets UDPPacketBuffer newbuf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(buf.Data, 0, newbuf.Data, 0, countposition); // start is the same buf.Data[countposition] = (byte)count; // get pending zeros at cut point if (lastzc > 0) { buf.Data[lastpos++] = 0; buf.Data[lastpos++] = (byte)lastzc; } buf.DataLength = lastpos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, // delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, false); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, false); buf = newbuf; zc.Data = buf.Data; zc.ZeroCount = 0; zc.Position = countposition + 1; // im lazy now, just do last again CreateObjectPropertiesBlock((SceneObjectPart)eu.Entity, zc); //tau = new List(30); //tau.Add(eu); count = 1; --blocks; } } if (count > 0) { buf.Data[countposition] = (byte)count; buf.DataLength = zc.Finish(); //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, // delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, false); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, false); } } if (objectPropertiesFamilyUpdates != null) { foreach (EntityUpdate eu in objectPropertiesFamilyUpdates) { UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(ObjectFamilyUpdateHeader, 0, buf.Data, 0, 8); LLUDPZeroEncoder zc = new LLUDPZeroEncoder(buf.Data); zc.Position = 8; CreateObjectPropertiesFamilyBlock((SceneObjectPart)eu.Entity, eu.Flags, zc); buf.DataLength = zc.Finish(); //List tau = new List(1); //tau.Add(new ObjectPropertyUpdate((ISceneEntity) eu, (uint)eu.Flags, true, false)); //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, // delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, false, false); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, false); } } if (needPhysics != null) { IEventQueue eq = Scene.RequestModuleInterface(); if(eq != null) { StringBuilder sb = eq.StartEvent("ObjectPhysicsProperties"); LLSDxmlEncode.AddArray("ObjectData", sb); foreach (SceneObjectPart sop in needPhysics) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("LocalID",(int)sop.LocalId, sb); LLSDxmlEncode.AddElem("Density", sop.Density, sb); LLSDxmlEncode.AddElem("Friction", sop.Friction, sb); LLSDxmlEncode.AddElem("GravityMultiplier", sop.GravityModifier, sb); LLSDxmlEncode.AddElem("Restitution", sop.Restitution, sb); LLSDxmlEncode.AddElem("PhysicsShapeType", (int)sop.PhysicsShapeType, sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); OSDllsdxml ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } } } private void CreateObjectPropertiesFamilyBlock(SceneObjectPart sop, PrimUpdateFlags requestFlags, LLUDPZeroEncoder zc) { SceneObjectPart root = sop.ParentGroup.RootPart; zc.AddUInt((uint)requestFlags); zc.AddUUID(sop.UUID); if (sop.OwnerID == sop.GroupID) zc.AddZeros(16); else zc.AddUUID(sop.OwnerID); zc.AddUUID(sop.GroupID); zc.AddUInt(root.BaseMask); zc.AddUInt(root.OwnerMask); zc.AddUInt(root.GroupMask); zc.AddUInt(root.EveryoneMask); zc.AddUInt(root.NextOwnerMask); zc.AddZeros(4); // int ownership cost //sale info block zc.AddByte(root.ObjectSaleType); zc.AddInt(root.SalePrice); zc.AddUInt(sop.Category); //Category zc.AddUUID(sop.LastOwnerID); //name zc.AddShortString(sop.Name, 64); //Description zc.AddShortString(sop.Description, 128); } private void CreateObjectPropertiesBlock(SceneObjectPart sop, LLUDPZeroEncoder zc) { SceneObjectPart root = sop.ParentGroup.RootPart; zc.AddUUID(sop.UUID); zc.AddUUID(sop.CreatorID); if (sop.OwnerID == sop.GroupID) zc.AddZeros(16); else zc.AddUUID(sop.OwnerID); zc.AddUUID(sop.GroupID); zc.AddUInt64((ulong)sop.CreationDate * 1000000UL); zc.AddUInt(root.BaseMask); zc.AddUInt(root.OwnerMask); zc.AddUInt(root.GroupMask); zc.AddUInt(root.EveryoneMask); zc.AddUInt(root.NextOwnerMask); zc.AddZeros(4); // int ownership cost //sale info block zc.AddByte(root.ObjectSaleType); zc.AddInt(root.SalePrice); //aggregated perms we may will need to fix this zc.AddByte(0); //AggregatePerms zc.AddByte(0); //AggregatePermTextures; zc.AddByte(0); //AggregatePermTexturesOwner //inventory info zc.AddUInt(sop.Category); //Category zc.AddInt16((short)sop.InventorySerial); zc.AddUUID(sop.FromUserInventoryItemID); zc.AddUUID(UUID.Zero); //FolderID zc.AddUUID(UUID.Zero); //FromTaskID zc.AddUUID(sop.LastOwnerID); //name zc.AddShortString(sop.Name, 64); //Description zc.AddShortString(sop.Description, 128); // touch name zc.AddShortString(root.TouchName, 9, 37); // sit name zc.AddShortString(root.SitName, 9, 37); //texture ids block // still not sending, not clear the impact on viewers, if any. // does seem redundant // to send we will need proper list of face texture ids without having to unpack texture entry all the time zc.AddZeros(1); } #region Estate Data Sending Methods private static bool convertParamStringToBool(byte[] field) { string s = Utils.BytesToString(field); if (s == "1" || s.ToLower() == "y" || s.ToLower() == "yes" || s.ToLower() == "t" || s.ToLower() == "true") { return true; } return false; } public void SendEstateList(UUID invoice, int code, UUID[] Data, uint estateID) { int TotalnumberIDs = Data.Length; int numberIDs; int IDIndex = 0; do { if(TotalnumberIDs > 63) numberIDs = 63; else numberIDs = TotalnumberIDs; TotalnumberIDs -= numberIDs; EstateOwnerMessagePacket packet = new EstateOwnerMessagePacket(); packet.AgentData.TransactionID = UUID.Random(); packet.AgentData.AgentID = AgentId; packet.AgentData.SessionID = SessionId; packet.MethodData.Invoice = invoice; packet.MethodData.Method = Utils.StringToBytes("setaccess"); EstateOwnerMessagePacket.ParamListBlock[] returnblock = new EstateOwnerMessagePacket.ParamListBlock[6 + numberIDs]; for (int i = 0; i < (6 + numberIDs); i++) { returnblock[i] = new EstateOwnerMessagePacket.ParamListBlock(); } returnblock[0].Parameter = Utils.StringToBytes(estateID.ToString()); returnblock[1].Parameter = Utils.StringToBytes(code.ToString()); if((code & 1) != 0) // allowagents returnblock[2].Parameter = Utils.StringToBytes(numberIDs.ToString()); else returnblock[2].Parameter = Utils.StringToBytes("0"); if((code & 2) != 0) // groups returnblock[3].Parameter = Utils.StringToBytes(numberIDs.ToString()); else returnblock[3].Parameter = Utils.StringToBytes("0"); if((code & 4) != 0) // bans returnblock[4].Parameter = Utils.StringToBytes(numberIDs.ToString()); else returnblock[4].Parameter = Utils.StringToBytes("0"); if((code & 8) != 0) // managers returnblock[5].Parameter = Utils.StringToBytes(numberIDs.ToString()); else returnblock[5].Parameter = Utils.StringToBytes("0"); int j = 6; for (int i = 0; i < numberIDs; i++) { returnblock[j].Parameter = Data[IDIndex].GetBytes(); j++; IDIndex++; } packet.ParamList = returnblock; packet.Header.Reliable = true; OutPacket(packet, ThrottleOutPacketType.Task); } while (TotalnumberIDs > 0); } public void SendBannedUserList(UUID invoice, EstateBan[] bl, uint estateID) { List BannedUsers = new List(); for (int i = 0; i < bl.Length; i++) { if (bl[i] == null) continue; if (bl[i].BannedUserID == UUID.Zero) continue; BannedUsers.Add(bl[i].BannedUserID); } SendEstateList(invoice, 4, BannedUsers.ToArray(), estateID); } public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) { RegionInfoPacket rinfopack = new RegionInfoPacket(); RegionInfoPacket.RegionInfoBlock rinfoblk = new RegionInfoPacket.RegionInfoBlock(); rinfopack.AgentData.AgentID = AgentId; rinfopack.AgentData.SessionID = SessionId; rinfoblk.BillableFactor = args.billableFactor; rinfoblk.EstateID = args.estateID; rinfoblk.MaxAgents = args.maxAgents; rinfoblk.ObjectBonusFactor = args.objectBonusFactor; rinfoblk.ParentEstateID = args.parentEstateID; rinfoblk.PricePerMeter = args.pricePerMeter; rinfoblk.RedirectGridX = args.redirectGridX; rinfoblk.RedirectGridY = args.redirectGridY; rinfoblk.RegionFlags = args.regionFlags; rinfoblk.SimAccess = args.simAccess; rinfoblk.SunHour = args.sunHour; rinfoblk.TerrainLowerLimit = args.terrainLowerLimit; rinfoblk.TerrainRaiseLimit = args.terrainRaiseLimit; rinfoblk.UseEstateSun = args.useEstateSun; rinfoblk.WaterHeight = args.waterHeight; rinfoblk.SimName = Utils.StringToBytes(args.simName); rinfopack.RegionInfo2 = new RegionInfoPacket.RegionInfo2Block(); rinfopack.RegionInfo2.HardMaxAgents = uint.MaxValue; rinfopack.RegionInfo2.HardMaxObjects = uint.MaxValue; rinfopack.RegionInfo2.MaxAgents32 = uint.MaxValue; rinfopack.RegionInfo2.ProductName = Util.StringToBytes256(args.regionType); rinfopack.RegionInfo2.ProductSKU = Utils.EmptyBytes; rinfopack.HasVariableBlocks = true; rinfopack.RegionInfo = rinfoblk; rinfopack.AgentData = new RegionInfoPacket.AgentDataBlock(); rinfopack.AgentData.AgentID = AgentId; rinfopack.AgentData.SessionID = SessionId; rinfopack.RegionInfo3 = new RegionInfoPacket.RegionInfo3Block[0]; OutPacket(rinfopack, ThrottleOutPacketType.Task); } public void SendEstateCovenantInformation(UUID covenant) { // m_log.DebugFormat("[LLCLIENTVIEW]: Sending estate covenant asset id of {0} to {1}", covenant, Name); EstateCovenantReplyPacket einfopack = new EstateCovenantReplyPacket(); EstateCovenantReplyPacket.DataBlock edata = new EstateCovenantReplyPacket.DataBlock(); edata.CovenantID = covenant; edata.CovenantTimestamp = (uint) m_scene.RegionInfo.RegionSettings.CovenantChangedDateTime; edata.EstateOwnerID = m_scene.RegionInfo.EstateSettings.EstateOwner; edata.EstateName = Utils.StringToBytes(m_scene.RegionInfo.EstateSettings.EstateName); einfopack.Data = edata; OutPacket(einfopack, ThrottleOutPacketType.Task); } public void SendDetailedEstateData( UUID invoice, string estateName, uint estateID, uint parentEstate, uint estateFlags, uint sunPosition, UUID covenant, uint covenantChanged, string abuseEmail, UUID estateOwner) { // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending detailed estate data to {0} with covenant asset id {1}", Name, covenant); EstateOwnerMessagePacket packet = new EstateOwnerMessagePacket(); packet.MethodData.Invoice = invoice; packet.AgentData.TransactionID = UUID.Random(); packet.MethodData.Method = Utils.StringToBytes("estateupdateinfo"); EstateOwnerMessagePacket.ParamListBlock[] returnblock = new EstateOwnerMessagePacket.ParamListBlock[10]; for (int i = 0; i < 10; i++) { returnblock[i] = new EstateOwnerMessagePacket.ParamListBlock(); } //Sending Estate Settings returnblock[0].Parameter = Utils.StringToBytes(estateName); returnblock[1].Parameter = Utils.StringToBytes(estateOwner.ToString()); returnblock[2].Parameter = Utils.StringToBytes(estateID.ToString()); returnblock[3].Parameter = Utils.StringToBytes(estateFlags.ToString()); returnblock[4].Parameter = Utils.StringToBytes(sunPosition.ToString()); returnblock[5].Parameter = Utils.StringToBytes(parentEstate.ToString()); returnblock[6].Parameter = Utils.StringToBytes(covenant.ToString()); returnblock[7].Parameter = Utils.StringToBytes(covenantChanged.ToString()); returnblock[8].Parameter = Utils.StringToBytes("1"); // what is this? returnblock[9].Parameter = Utils.StringToBytes(abuseEmail); packet.ParamList = returnblock; //m_log.Debug("[ESTATE]: SIM--->" + packet.ToString()); OutPacket(packet, ThrottleOutPacketType.Task); } public void SendTelehubInfo(UUID ObjectID, string ObjectName, Vector3 ObjectPos, Quaternion ObjectRot, List SpawnPoint) { TelehubInfoPacket packet = (TelehubInfoPacket)PacketPool.Instance.GetPacket(PacketType.TelehubInfo); packet.TelehubBlock.ObjectID = ObjectID; packet.TelehubBlock.ObjectName = Utils.StringToBytes(ObjectName); packet.TelehubBlock.TelehubPos = ObjectPos; packet.TelehubBlock.TelehubRot = ObjectRot; packet.SpawnPointBlock = new TelehubInfoPacket.SpawnPointBlockBlock[SpawnPoint.Count]; for (int n = 0; n < SpawnPoint.Count; n++) { packet.SpawnPointBlock[n] = new TelehubInfoPacket.SpawnPointBlockBlock{SpawnPointPos = SpawnPoint[n]}; } OutPacket(packet, ThrottleOutPacketType.Task); } #endregion #region Land Data Sending Methods public void SendLandParcelOverlay(byte[] data, int sequence_id) { ParcelOverlayPacket packet = (ParcelOverlayPacket)PacketPool.Instance.GetPacket(PacketType.ParcelOverlay); packet.ParcelData.Data = data; packet.ParcelData.SequenceID = sequence_id; // OutPacket(packet, ThrottleOutPacketType.Task); OutPacket(packet, ThrottleOutPacketType.Land); } public void SendLandProperties( int sequence_id, bool snap_selection, int request_result, ILandObject lo, float simObjectBonusFactor, int parcelObjectCapacity, int simObjectCapacity, uint regionFlags) { // m_log.DebugFormat("[LLCLIENTVIEW]: Sending land properties for {0} to {1}", lo.LandData.GlobalID, Name); IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) { m_log.Warn("[LLCLIENTVIEW]: No EQ Interface when sending parcel data."); return; } LandData landData = lo.LandData; IPrimCounts pc = lo.PrimCounts; StringBuilder sb = eq.StartEvent("ParcelProperties"); LLSDxmlEncode.AddArrayAndMap("ParcelData", sb); LLSDxmlEncode.AddElem("LocalID", landData.LocalID, sb); LLSDxmlEncode.AddElem("AABBMax", landData.AABBMax, sb); LLSDxmlEncode.AddElem("AABBMin", landData.AABBMin, sb); LLSDxmlEncode.AddElem("Area", landData.Area, sb); LLSDxmlEncode.AddElem("AuctionID", (int)landData.AuctionID, sb); LLSDxmlEncode.AddElem("AuthBuyerID", landData.AuthBuyerID, sb); LLSDxmlEncode.AddElem("Bitmap", landData.Bitmap, sb); LLSDxmlEncode.AddElem("Category", (int)landData.Category, sb); LLSDxmlEncode.AddElem("ClaimDate", Util.ToDateTime(landData.ClaimDate), sb); LLSDxmlEncode.AddElem("ClaimPrice", landData.ClaimPrice, sb); LLSDxmlEncode.AddElem("Desc", landData.Description, sb); LLSDxmlEncode.AddElem("ParcelFlags", landData.Flags, sb); LLSDxmlEncode.AddElem("GroupID", landData.GroupID, sb); LLSDxmlEncode.AddElem("GroupPrims", pc.Group, sb); LLSDxmlEncode.AddElem("IsGroupOwned", landData.IsGroupOwned, sb); LLSDxmlEncode.AddElem("LandingType", (int)landData.LandingType, sb); if (landData.Area > 0) LLSDxmlEncode.AddElem("MaxPrims", parcelObjectCapacity, sb); else LLSDxmlEncode.AddElem("MaxPrims", (int)0, sb); LLSDxmlEncode.AddElem("MediaID", landData.MediaID, sb); LLSDxmlEncode.AddElem("MediaURL", landData.MediaURL, sb); LLSDxmlEncode.AddElem("MediaAutoScale", landData.MediaAutoScale != 0, sb); LLSDxmlEncode.AddElem("MusicURL", landData.MusicURL, sb); LLSDxmlEncode.AddElem("Name", landData.Name, sb); LLSDxmlEncode.AddElem("OtherCleanTime", landData.OtherCleanTime, sb); LLSDxmlEncode.AddElem("OtherCount", (int)0 , sb); //TODO LLSDxmlEncode.AddElem("OtherPrims", pc.Others, sb); LLSDxmlEncode.AddElem("OwnerID", landData.OwnerID, sb); LLSDxmlEncode.AddElem("OwnerPrims", pc.Owner, sb); LLSDxmlEncode.AddElem("ParcelPrimBonus", simObjectBonusFactor, sb); LLSDxmlEncode.AddElem("PassHours", landData.PassHours, sb); LLSDxmlEncode.AddElem("PassPrice", landData.PassPrice, sb); LLSDxmlEncode.AddElem("PublicCount", (int)0, sb); //TODO LLSDxmlEncode.AddElem("RegionDenyAnonymous", (regionFlags & (uint)RegionFlags.DenyAnonymous) != 0, sb); //LLSDxmlEncode.AddElem("RegionDenyIdentified", (regionFlags & (uint)RegionFlags.DenyIdentified) != 0, sb); LLSDxmlEncode.AddElem("RegionDenyIdentified", false, sb); //LLSDxmlEncode.AddElem("RegionDenyTransacted", (regionFlags & (uint)RegionFlags.DenyTransacted) != 0, sb); LLSDxmlEncode.AddElem("RegionDenyTransacted", false, sb); LLSDxmlEncode.AddElem("RegionPushOverride", (regionFlags & (uint)RegionFlags.RestrictPushObject) != 0, sb); LLSDxmlEncode.AddElem("RentPrice", (int) 0, sb);; LLSDxmlEncode.AddElem("RequestResult", request_result, sb); LLSDxmlEncode.AddElem("SalePrice", landData.SalePrice, sb); LLSDxmlEncode.AddElem("SelectedPrims", pc.Selected, sb); LLSDxmlEncode.AddElem("SelfCount", (int)0, sb); //TODO LLSDxmlEncode.AddElem("SequenceID", sequence_id, sb); if (landData.SimwideArea > 0) LLSDxmlEncode.AddElem("SimWideMaxPrims", lo.GetSimulatorMaxPrimCount(), sb); else LLSDxmlEncode.AddElem("SimWideMaxPrims", (int)0, sb); LLSDxmlEncode.AddElem("SimWideTotalPrims", pc.Simulator, sb); LLSDxmlEncode.AddElem("SnapSelection", snap_selection, sb); LLSDxmlEncode.AddElem("SnapshotID", landData.SnapshotID, sb); LLSDxmlEncode.AddElem("Status", (int)landData.Status, sb); LLSDxmlEncode.AddElem("TotalPrims", pc.Total, sb); LLSDxmlEncode.AddElem("UserLocation", landData.UserLocation, sb); LLSDxmlEncode.AddElem("UserLookAt", landData.UserLookAt, sb); LLSDxmlEncode.AddElem("SeeAVs", landData.SeeAVs, sb); LLSDxmlEncode.AddElem("AnyAVSounds", landData.AnyAVSounds, sb); LLSDxmlEncode.AddElem("GroupAVSounds", landData.GroupAVSounds, sb); LLSDxmlEncode.AddEndMapAndArray(sb); LLSDxmlEncode.AddArrayAndMap("MediaData", sb); LLSDxmlEncode.AddElem("MediaDesc", landData.MediaDescription, sb); LLSDxmlEncode.AddElem("MediaHeight", landData.MediaHeight, sb); LLSDxmlEncode.AddElem("MediaWidth", landData.MediaWidth, sb); LLSDxmlEncode.AddElem("MediaLoop", landData.MediaLoop, sb); LLSDxmlEncode.AddElem("MediaType", landData.MediaType, sb); LLSDxmlEncode.AddElem("ObscureMedia", landData.ObscureMedia, sb); LLSDxmlEncode.AddElem("ObscureMusic", landData.ObscureMusic, sb); LLSDxmlEncode.AddEndMapAndArray(sb); LLSDxmlEncode.AddArrayAndMap("AgeVerificationBlock", sb); //LLSDxmlEncode.AddElem("RegionDenyAgeUnverified", (regionFlags & (uint)RegionFlags.DenyAgeUnverified) != 0, sb); LLSDxmlEncode.AddElem("RegionDenyAgeUnverified", false, sb); LLSDxmlEncode.AddEndMapAndArray(sb); OSDllsdxml ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } public void SendLandAccessListData(List accessList, uint accessFlag, int localLandID) { ParcelAccessListReplyPacket replyPacket = (ParcelAccessListReplyPacket)PacketPool.Instance.GetPacket(PacketType.ParcelAccessListReply); replyPacket.Data.AgentID = AgentId; replyPacket.Data.Flags = accessFlag; replyPacket.Data.LocalID = localLandID; replyPacket.Data.SequenceID = 0; List list = new List(); foreach (LandAccessEntry entry in accessList) { ParcelAccessListReplyPacket.ListBlock block = new ParcelAccessListReplyPacket.ListBlock(); block.Flags = accessFlag; block.ID = entry.AgentID; block.Time = entry.Expires; list.Add(block); } replyPacket.List = list.ToArray(); replyPacket.Header.Zerocoded = true; OutPacket(replyPacket, ThrottleOutPacketType.Task); } public void SendForceClientSelectObjects(List ObjectIDs) { // m_log.DebugFormat("[LLCLIENTVIEW] sending select with {0} objects", ObjectIDs.Count); bool firstCall = true; const int MAX_OBJECTS_PER_PACKET = 251; ForceObjectSelectPacket pack = (ForceObjectSelectPacket)PacketPool.Instance.GetPacket(PacketType.ForceObjectSelect); ForceObjectSelectPacket.DataBlock[] data; while (ObjectIDs.Count > 0) { if (firstCall) { pack._Header.ResetList = true; firstCall = false; } else { pack._Header.ResetList = false; } if (ObjectIDs.Count > MAX_OBJECTS_PER_PACKET) { data = new ForceObjectSelectPacket.DataBlock[MAX_OBJECTS_PER_PACKET]; } else { data = new ForceObjectSelectPacket.DataBlock[ObjectIDs.Count]; } int i; for (i = 0; i < MAX_OBJECTS_PER_PACKET && ObjectIDs.Count > 0; i++) { data[i] = new ForceObjectSelectPacket.DataBlock(); data[i].LocalID = Convert.ToUInt32(ObjectIDs[0]); ObjectIDs.RemoveAt(0); } pack.Data = data; pack.Header.Zerocoded = true; OutPacket(pack, ThrottleOutPacketType.Task); } } public void SendCameraConstraint(Vector4 ConstraintPlane) { CameraConstraintPacket cpack = (CameraConstraintPacket)PacketPool.Instance.GetPacket(PacketType.CameraConstraint); cpack.CameraCollidePlane = new CameraConstraintPacket.CameraCollidePlaneBlock(); cpack.CameraCollidePlane.Plane = ConstraintPlane; //m_log.DebugFormat("[CLIENTVIEW]: Constraint {0}", ConstraintPlane); OutPacket(cpack, ThrottleOutPacketType.Task); } public void SendLandObjectOwners(LandData land, List groups, Dictionary ownersAndCount) { int notifyCount = ownersAndCount.Count; ParcelObjectOwnersReplyPacket pack = (ParcelObjectOwnersReplyPacket)PacketPool.Instance.GetPacket(PacketType.ParcelObjectOwnersReply); if (notifyCount > 0) { // if (notifyCount > 32) // { // m_log.InfoFormat( // "[LAND]: More than {0} avatars own prims on this parcel. Only sending back details of first {0}" // + " - a developer might want to investigate whether this is a hard limit", 32); // // notifyCount = 32; // } ParcelObjectOwnersReplyPacket.DataBlock[] dataBlock = new ParcelObjectOwnersReplyPacket.DataBlock[notifyCount]; int num = 0; foreach (UUID owner in ownersAndCount.Keys) { dataBlock[num] = new ParcelObjectOwnersReplyPacket.DataBlock(); dataBlock[num].Count = ownersAndCount[owner]; if (land.GroupID == owner || groups.Contains(owner)) dataBlock[num].IsGroupOwned = true; dataBlock[num].OnlineStatus = true; //TODO: fix me later dataBlock[num].OwnerID = owner; num++; if (num >= notifyCount) { break; } } pack.Data = dataBlock; } else { pack.Data = new ParcelObjectOwnersReplyPacket.DataBlock[0]; } pack.Header.Zerocoded = true; this.OutPacket(pack, ThrottleOutPacketType.Task); } #endregion #region Helper Methods private void ClampVectorForUint(ref Vector3 v, float max) { float a,b; a = Math.Abs(v.X); b = Math.Abs(v.Y); if(b > a) a = b; b= Math.Abs(v.Z); if(b > a) a = b; if (a > max) { a = max / a; v.X *= a; v.Y *= a; v.Z *= a; } } protected void CreateImprovedTerseBlock(ISceneEntity entity, byte[] data, ref int pos, bool includeTexture) { #region ScenePresence/SOP Handling bool avatar = (entity is ScenePresence); uint localID = entity.LocalId; uint attachPoint; Vector4 collisionPlane; Vector3 position, velocity, acceleration, angularVelocity; Quaternion rotation; byte datasize; byte[] te = null; if (avatar) { ScenePresence presence = (ScenePresence)entity; position = presence.OffsetPosition; velocity = presence.Velocity; acceleration = Vector3.Zero; rotation = presence.Rotation; // tpvs can only see rotations around Z in some cases if (!presence.Flying && !presence.IsSatOnObject) { rotation.X = 0f; rotation.Y = 0f; } rotation.Normalize(); angularVelocity = presence.AngularVelocity; // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending terse update to {0} with position {1} in {2}", Name, presence.OffsetPosition, m_scene.Name); attachPoint = presence.State; collisionPlane = presence.CollisionPlane; datasize = 60; } else { SceneObjectPart part = (SceneObjectPart)entity; attachPoint = part.ParentGroup.AttachmentPoint; attachPoint = ((attachPoint % 16) * 16 + (attachPoint / 16)); // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending attachPoint {0} for {1} {2} to {3}", // attachPoint, part.Name, part.LocalId, Name); collisionPlane = Vector4.Zero; position = part.RelativePosition; velocity = part.Velocity; acceleration = part.Acceleration; angularVelocity = part.AngularVelocity; rotation = part.RotationOffset; datasize = 44; if(includeTexture) te = part.Shape.TextureEntry; } #endregion ScenePresence/SOP Handling //object block size data[pos++] = datasize; // LocalID Utils.UIntToBytes(localID, data, pos); pos += 4; data[pos++] = (byte)attachPoint; // Avatar/CollisionPlane if (avatar) { data[pos++] = 1; if (collisionPlane == Vector4.Zero) collisionPlane = Vector4.UnitW; //m_log.DebugFormat("CollisionPlane: {0}",collisionPlane); collisionPlane.ToBytes(data, pos); pos += 16; } else { data[pos++] = 0; } // Position position.ToBytes(data, pos); pos += 12; // Velocity ClampVectorForUint(ref velocity, 128f); Utils.FloatToUInt16Bytes(velocity.X, 128.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(velocity.Y, 128.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(velocity.Z, 128.0f, data, pos); pos += 2; // Acceleration ClampVectorForUint(ref acceleration, 64f); Utils.FloatToUInt16Bytes(acceleration.X, 64.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(acceleration.Y, 64.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(acceleration.Z, 64.0f, data, pos); pos += 2; // Rotation Utils.FloatToUInt16Bytes(rotation.X, 1.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(rotation.Y, 1.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(rotation.Z, 1.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(rotation.W, 1.0f, data, pos); pos += 2; // Angular Velocity ClampVectorForUint(ref angularVelocity, 64f); Utils.FloatToUInt16Bytes(angularVelocity.X, 64.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(angularVelocity.Y, 64.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(angularVelocity.Z, 64.0f, data, pos); pos += 2; // texture entry block size if(te == null) { data[pos++] = 0; data[pos++] = 0; } else { int len = te.Length & 0x7fff; int totlen = len + 4; data[pos++] = (byte)totlen; data[pos++] = (byte)(totlen >> 8); data[pos++] = (byte)len; // wtf ??? data[pos++] = (byte)(len >> 8); data[pos++] = 0; data[pos++] = 0; Buffer.BlockCopy(te, 0, data, pos, len); pos += len; } // total size 63 or 47 + (texture size + 4) } protected ObjectUpdatePacket.ObjectDataBlock CreateAvatarUpdateBlock(ScenePresence data) { Quaternion rotation = data.Rotation; // tpvs can only see rotations around Z in some cases if(!data.Flying && !data.IsSatOnObject) { rotation.X = 0f; rotation.Y = 0f; } rotation.Normalize(); // m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending full update to {0} with pos {1}, vel {2} in {3}", Name, data.OffsetPosition, data.Velocity, m_scene.Name); byte[] objectData = new byte[76]; //Vector3 velocity = Vector3.Zero; Vector3 acceleration = Vector3.Zero; Vector3 angularvelocity = Vector3.Zero; data.CollisionPlane.ToBytes(objectData, 0); data.OffsetPosition.ToBytes(objectData, 16); data.Velocity.ToBytes(objectData, 28); acceleration.ToBytes(objectData, 40); rotation.ToBytes(objectData, 52); angularvelocity.ToBytes(objectData, 64); ObjectUpdatePacket.ObjectDataBlock update = new ObjectUpdatePacket.ObjectDataBlock(); update.Data = Utils.EmptyBytes; update.ExtraParams = Utils.EmptyBytes; update.FullID = data.UUID; update.ID = data.LocalId; update.Material = (byte)Material.Flesh; update.MediaURL = Utils.EmptyBytes; update.NameValue = Utils.StringToBytes("FirstName STRING RW SV " + data.Firstname + "\nLastName STRING RW SV " + data.Lastname + "\nTitle STRING RW SV " + data.Grouptitle); update.ObjectData = objectData; SceneObjectPart parentPart = data.ParentPart; if (parentPart != null) update.ParentID = parentPart.ParentGroup.LocalId; else update.ParentID = 0; update.PathCurve = 16; update.PathScaleX = 100; update.PathScaleY = 100; update.PCode = (byte)PCode.Avatar; update.ProfileCurve = 1; update.PSBlock = Utils.EmptyBytes; update.Scale = data.Appearance.AvatarSize; // update.Scale.Z -= 0.2f; update.Text = Utils.EmptyBytes; update.TextColor = new byte[4]; // Don't send texture anim for avatars - this has no meaning for them. update.TextureAnim = Utils.EmptyBytes; // Don't send texture entry for avatars here - this is accomplished via the AvatarAppearance packet update.TextureEntry = Utils.EmptyBytes; update.UpdateFlags = 0; return update; } protected void CreateAvatarUpdateBlock(ScenePresence data, byte[] dest, ref int pos) { Quaternion rotation = data.Rotation; // tpvs can only see rotations around Z in some cases if (!data.Flying && !data.IsSatOnObject) { rotation.X = 0f; rotation.Y = 0f; } rotation.Normalize(); //Vector3 velocity = Vector3.Zero; //Vector3 acceleration = Vector3.Zero; //Vector3 angularvelocity = Vector3.Zero; Utils.UIntToBytesSafepos(data.LocalId, dest, pos); pos += 4; dest[pos++] = 0; // state data.UUID.ToBytes(dest, pos); pos += 16; Utils.UIntToBytesSafepos(0 , dest, pos); pos += 4; // crc dest[pos++] = (byte)PCode.Avatar; dest[pos++] = (byte)Material.Flesh; dest[pos++] = 0; // clickaction data.Appearance.AvatarSize.ToBytes(dest, pos); pos += 12; // objectdata block dest[pos++] = 76; data.CollisionPlane.ToBytes(dest, pos); pos += 16; data.OffsetPosition.ToBytes(dest, pos); pos += 12; data.Velocity.ToBytes(dest, pos); pos += 12; //acceleration.ToBytes(dest, pos); pos += 12; Array.Clear(dest, pos, 12); pos += 12; rotation.ToBytes(dest, pos); pos += 12; //angularvelocity.ToBytes(dest, pos); pos += 12; Array.Clear(dest, pos, 12); pos += 12; SceneObjectPart parentPart = data.ParentPart; if (parentPart != null) { Utils.UIntToBytesSafepos(parentPart.ParentGroup.LocalId, dest, pos); pos += 4; } else { // Utils.UIntToBytesSafepos(0, dest, pos); // pos += 4; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; } //Utils.UIntToBytesSafepos(0, dest, pos); pos += 4; //update flags dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; //pbs dest[pos++] = 16; dest[pos++] = 1; //Utils.UInt16ToBytes(0, dest, pos); pos += 2; //Utils.UInt16ToBytes(0, dest, pos); pos += 2; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 100; dest[pos++] = 100; // rest of pbs is 0 (15), texture entry (2) and texture anim (1) const int pbszeros = 15 + 2 + 1; Array.Clear(dest, pos, pbszeros); pos += pbszeros; //NameValue byte[] nv = Utils.StringToBytes("FirstName STRING RW SV " + data.Firstname + "\nLastName STRING RW SV " + data.Lastname + "\nTitle STRING RW SV " + data.Grouptitle); int len = nv.Length; dest[pos++] = (byte)len; dest[pos++] = (byte)(len >> 8); Buffer.BlockCopy(nv, 0, dest, pos, len); pos += len; // data(2), text(1), text color(4), media url(1), PBblock(1), ExtramParams(1), // sound id(16), sound owner(16) gain (4), flags (1), radius (4) // jointtype(1) joint pivot(12) joint offset(12) const int lastzeros = 2 + 1 + 4 + 1 + 1 + 1 + 16 + 16 + 4 + 1 + 4 + 1 + 12 + 12; Array.Clear(dest, pos, lastzeros); pos += lastzeros; } protected void CreateAvatarUpdateBlock(ScenePresence data, LLUDPZeroEncoder zc) { Quaternion rotation = data.Rotation; // tpvs can only see rotations around Z in some cases if (!data.Flying && !data.IsSatOnObject) { rotation.X = 0f; rotation.Y = 0f; } rotation.Normalize(); zc.AddUInt(data.LocalId); zc.AddByte(0); zc.AddUUID(data.UUID); zc.AddZeros(4); // crc unused zc.AddByte((byte)PCode.Avatar); zc.AddByte((byte)Material.Flesh); zc.AddByte(0); // clickaction zc.AddVector3(data.Appearance.AvatarSize); // objectdata block zc.AddByte(76); // fixed avatar block size zc.AddVector4(data.CollisionPlane); zc.AddVector3(data.OffsetPosition); zc.AddVector3(data.Velocity); //zc.AddVector3(acceleration); zc.AddZeros(12); zc.AddNormQuat(rotation); //zc.AddVector3(angularvelocity); zc.AddZeros(12); SceneObjectPart parentPart = data.ParentPart; if (parentPart != null) zc.AddUInt(parentPart.ParentGroup.LocalId); else zc.AddZeros(4); zc.AddZeros(4); //update flags //pbs volume data 23 //texture entry 2 //texture anim (1) const int pbszeros = 23 + 2 + 1; zc.AddZeros(pbszeros); //NameValue byte[] nv = Utils.StringToBytes("FirstName STRING RW SV " + data.Firstname + "\nLastName STRING RW SV " + data.Lastname + "\nTitle STRING RW SV " + data.Grouptitle); int len = nv.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddBytes(nv, len); // data(2), text(1), text color(4), media url(1), PBblock(1), ExtramParams(1), // sound id(16), sound owner(16) gain (4), flags (1), radius (4) // jointtype(1) joint pivot(12) joint offset(12) const int lastzeros = 2 + 1 + 4 + 1 + 1 + 1 + 16 + 16 + 4 + 1 + 4 + 1 + 12 + 12; zc.AddZeros(lastzeros); } protected void CreatePrimUpdateBlock(SceneObjectPart part, ScenePresence sp, LLUDPZeroEncoder zc) { // prepare data #region PrimFlags // prim/update flags PrimFlags primflags = (PrimFlags)m_scene.Permissions.GenerateClientFlags(part, sp); // Don't send the CreateSelected flag to everyone primflags &= ~PrimFlags.CreateSelected; if (sp.UUID == part.OwnerID) { if (part.CreateSelected) { // Only send this flag once, then unset it primflags |= PrimFlags.CreateSelected; part.CreateSelected = false; } } #endregion PrimFlags // data block byte[] data = null; byte state = part.Shape.State; PCode pcode = (PCode)part.Shape.PCode; //vegetation is special so just do it inline if(pcode == PCode.Grass || pcode == PCode.Tree || pcode == PCode.NewTree) { zc.AddUInt(part.LocalId); zc.AddByte(state); // state zc.AddUUID(part.UUID); zc.AddZeros(4); // crc unused zc.AddByte((byte)pcode); // material 1 // clickaction 1 zc.AddZeros(2); zc.AddVector3(part.Shape.Scale); // objectdata block zc.AddByte(60); // fixed object block size zc.AddVector3(part.RelativePosition); if (pcode == PCode.Grass) zc.AddZeros(48); else { zc.AddZeros(24); Quaternion rot = part.RotationOffset; rot.Normalize(); zc.AddNormQuat(rot); zc.AddZeros(12); } zc.AddUInt(part.ParentID); zc.AddUInt((uint)primflags); //update flags /* if (pcode == PCode.Grass) { //pbs volume data 23 //texture entry 2 //texture anim 1 //name value 2 // data 1 // text 5 // media url 1 // particle system 1 // Extraparams 1 // sound id 16 // ownwer 16 // sound gain 4 // sound flags 1 // sound radius 4 // jointtype 1 // joint pivot 12 // joint offset 12 zc.AddZeros(23 + 2 + 1 + 2 + 1 + 5 + 1 + 1 + 1 + 16 + 16 + 4 + 1 + 4 + 1 + 12 + 12); return; } */ //pbs volume data 23 //texture entry 2 //texture anim 1 //name value 2 zc.AddZeros(23 + 2 + 1 + 2); //data: the tree type zc.AddByte(1); zc.AddZeros(1); zc.AddByte(state); // text 5 // media url 1 // particle system 1 // Extraparams 1 // sound id 16 // ownwer 16 // sound gain 4 // sound flags 1 // sound radius 4 // jointtype 1 // joint pivot 12 // joint offset 12 zc.AddZeros(5 + 1 + 1 + 1 + 16 + 16 + 4 + 1 + 4 + 1 + 12 + 12); return; } //NameValue and state byte[] nv = null; if (part.ParentGroup.IsAttachment) { if (part.IsRoot) nv = Util.StringToBytes256("AttachItemID STRING RW SV " + part.ParentGroup.FromItemID); int st = (int)part.ParentGroup.AttachmentPoint; state = (byte)(((st & 0xf0) >> 4) + ((st & 0x0f) << 4)); ; } // filter out mesh faces hack ushort profileBegin = part.Shape.ProfileBegin; ushort profileHollow = part.Shape.ProfileHollow; byte profileCurve = part.Shape.ProfileCurve; byte pathScaleY = part.Shape.PathScaleY; if (part.Shape.SculptType == (byte)SculptType.Mesh) // filter out hack { profileCurve = (byte)(part.Shape.ProfileCurve & 0x0f); // fix old values that confused viewers if (profileBegin == 1) profileBegin = 9375; if (profileHollow == 1) profileHollow = 27500; // fix torus hole size Y that also confuse some viewers if (profileCurve == (byte)ProfileShape.Circle && pathScaleY < 150) pathScaleY = 150; } // do encode the things zc.AddUInt(part.LocalId); zc.AddByte(state); // state zc.AddUUID(part.UUID); zc.AddZeros(4); // crc unused zc.AddByte((byte)pcode); zc.AddByte(part.Material); zc.AddByte(part.ClickAction); // clickaction zc.AddVector3(part.Shape.Scale); // objectdata block zc.AddByte(60); // fixed object block size zc.AddVector3(part.RelativePosition); zc.AddVector3(part.Velocity); zc.AddVector3(part.Acceleration); Quaternion rotation = part.RotationOffset; rotation.Normalize(); zc.AddNormQuat(rotation); zc.AddVector3(part.AngularVelocity); zc.AddUInt(part.ParentID); zc.AddUInt((uint)primflags); //update flags //pbs zc.AddByte(part.Shape.PathCurve); zc.AddByte(profileCurve); zc.AddUInt16(part.Shape.PathBegin); zc.AddUInt16(part.Shape.PathEnd); zc.AddByte(part.Shape.PathScaleX); zc.AddByte(pathScaleY); zc.AddByte(part.Shape.PathShearX); zc.AddByte(part.Shape.PathShearY); zc.AddByte((byte)part.Shape.PathTwist); zc.AddByte((byte)part.Shape.PathTwistBegin); zc.AddByte((byte)part.Shape.PathRadiusOffset); zc.AddByte((byte)part.Shape.PathTaperX); zc.AddByte((byte)part.Shape.PathTaperY); zc.AddByte(part.Shape.PathRevolutions); zc.AddByte((byte)part.Shape.PathSkew); zc.AddUInt16(profileBegin); zc.AddUInt16(part.Shape.ProfileEnd); zc.AddUInt16(profileHollow); // texture byte[] tentry = part.Shape.TextureEntry; if (tentry == null) zc.AddZeros(2); else { int len = tentry.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddBytes(tentry, len); } // texture animation byte[] tanim = part.TextureAnimation; if (tanim == null) zc.AddZeros(1); else { int len = tanim.Length; zc.AddByte((byte)len); zc.AddBytes(tanim, len); } //NameValue if(nv == null) zc.AddZeros(2); else { int len = nv.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddBytes(nv, len); } // data if (data == null) zc.AddZeros(2); else { int len = data.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddBytes(data, len); } //text if (part.Text == null || part.Text.Length == 0) zc.AddZeros(5); else { zc.AddShortString(part.Text, 255); //textcolor byte[] tc = part.GetTextColor().GetBytes(false); zc.AddBytes(tc, 4); } //media url if (part.MediaUrl == null || part.MediaUrl.Length == 0) zc.AddZeros(1); else zc.AddShortString(part.MediaUrl, 255); bool hasps = false; //particle system byte[] ps = part.ParticleSystem; if (ps == null || ps.Length < 1) zc.AddZeros(1); else { int len = ps.Length; zc.AddByte((byte)len); zc.AddBytes(ps, len); hasps = true; } //Extraparams byte[] ep = part.Shape.ExtraParams; if (ep == null || ep.Length < 2) zc.AddZeros(1); else { int len = ep.Length; zc.AddByte((byte)len); zc.AddBytes(ep, len); } bool hassound = part.Sound != UUID.Zero || part.SoundFlags != 0; if (hassound) zc.AddUUID(part.Sound); else zc.AddZeros(16); if (hassound || hasps) zc.AddUUID(part.OwnerID); else zc.AddZeros(16); if (hassound) { zc.AddFloat((float)part.SoundGain); zc.AddByte(part.SoundFlags); zc.AddFloat((float)part.SoundRadius); } else zc.AddZeros(9); // jointtype(1) joint pivot(12) joint offset(12) const int lastzeros = 1 + 12 + 12; zc.AddZeros(lastzeros); } protected ObjectUpdateCompressedPacket.ObjectDataBlock CreateCompressedUpdateBlock(SceneObjectPart part, PrimUpdateFlags updateFlags) { // TODO: Implement this return null; } public void SendNameReply(UUID profileId, string firstname, string lastname) { UUIDNameReplyPacket packet = (UUIDNameReplyPacket)PacketPool.Instance.GetPacket(PacketType.UUIDNameReply); // TODO: don't create new blocks if recycling an old packet packet.UUIDNameBlock = new UUIDNameReplyPacket.UUIDNameBlockBlock[1]; packet.UUIDNameBlock[0] = new UUIDNameReplyPacket.UUIDNameBlockBlock(); packet.UUIDNameBlock[0].ID = profileId; packet.UUIDNameBlock[0].FirstName = Util.StringToBytes256(firstname); packet.UUIDNameBlock[0].LastName = Util.StringToBytes256(lastname); OutPacket(packet, ThrottleOutPacketType.Task); } public Dictionary GetGroupPowers() { lock(m_groupPowers) { return new Dictionary(m_groupPowers); } } public void SetGroupPowers(Dictionary powers) { lock(m_groupPowers) { m_groupPowers.Clear(); m_groupPowers = powers; } } public ulong GetGroupPowers(UUID groupID) { lock(m_groupPowers) { if (m_groupPowers.ContainsKey(groupID)) return m_groupPowers[groupID]; } return 0; } #endregion /// /// This is a different way of processing packets then ProcessInPacket /// protected virtual void RegisterLocalPacketHandlers() { AddLocalPacketHandler(PacketType.LogoutRequest, HandleLogout); // If AgentUpdate is ever handled asynchronously, then we will also need to construct a new AgentUpdateArgs // for each AgentUpdate packet. AddLocalPacketHandler(PacketType.AgentUpdate, HandleAgentUpdate, false); AddLocalPacketHandler(PacketType.ViewerEffect, HandleViewerEffect, false); AddLocalPacketHandler(PacketType.VelocityInterpolateOff, HandleVelocityInterpolateOff, false); AddLocalPacketHandler(PacketType.VelocityInterpolateOn, HandleVelocityInterpolateOn, false); AddLocalPacketHandler(PacketType.AgentCachedTexture, HandleAgentTextureCached, false); AddLocalPacketHandler(PacketType.MultipleObjectUpdate, HandleMultipleObjUpdate, false); AddLocalPacketHandler(PacketType.MoneyTransferRequest, HandleMoneyTransferRequest, false); AddLocalPacketHandler(PacketType.ParcelBuy, HandleParcelBuyRequest, false); AddLocalPacketHandler(PacketType.UUIDGroupNameRequest, HandleUUIDGroupNameRequest); AddLocalPacketHandler(PacketType.ObjectGroup, HandleObjectGroupRequest); AddLocalPacketHandler(PacketType.GenericMessage, HandleGenericMessage); AddLocalPacketHandler(PacketType.AvatarPropertiesRequest, HandleAvatarPropertiesRequest); AddLocalPacketHandler(PacketType.ChatFromViewer, HandleChatFromViewer); AddLocalPacketHandler(PacketType.AvatarPropertiesUpdate, HandlerAvatarPropertiesUpdate); AddLocalPacketHandler(PacketType.ScriptDialogReply, HandlerScriptDialogReply); AddLocalPacketHandler(PacketType.ImprovedInstantMessage, HandlerImprovedInstantMessage); AddLocalPacketHandler(PacketType.AcceptFriendship, HandlerAcceptFriendship); AddLocalPacketHandler(PacketType.DeclineFriendship, HandlerDeclineFriendship); AddLocalPacketHandler(PacketType.TerminateFriendship, HandlerTerminateFriendship); AddLocalPacketHandler(PacketType.RezObject, HandlerRezObject); AddLocalPacketHandler(PacketType.DeRezObject, HandlerDeRezObject); AddLocalPacketHandler(PacketType.RezRestoreToWorld, HandlerRezRestoreToWorld); AddLocalPacketHandler(PacketType.ModifyLand, HandlerModifyLand); AddLocalPacketHandler(PacketType.RegionHandshakeReply, HandlerRegionHandshakeReply, false); AddLocalPacketHandler(PacketType.AgentWearablesRequest, HandlerAgentWearablesRequest); AddLocalPacketHandler(PacketType.AgentSetAppearance, HandlerAgentSetAppearance); AddLocalPacketHandler(PacketType.AgentIsNowWearing, HandlerAgentIsNowWearing); AddLocalPacketHandler(PacketType.RezSingleAttachmentFromInv, HandlerRezSingleAttachmentFromInv); AddLocalPacketHandler(PacketType.RezMultipleAttachmentsFromInv, HandleRezMultipleAttachmentsFromInv); AddLocalPacketHandler(PacketType.DetachAttachmentIntoInv, HandleDetachAttachmentIntoInv); AddLocalPacketHandler(PacketType.ObjectAttach, HandleObjectAttach); AddLocalPacketHandler(PacketType.ObjectDetach, HandleObjectDetach); AddLocalPacketHandler(PacketType.ObjectDrop, HandleObjectDrop); AddLocalPacketHandler(PacketType.SetAlwaysRun, HandleSetAlwaysRun, false); AddLocalPacketHandler(PacketType.CompleteAgentMovement, HandleCompleteAgentMovement); AddLocalPacketHandler(PacketType.AgentAnimation, HandleAgentAnimation, false); AddLocalPacketHandler(PacketType.AgentRequestSit, HandleAgentRequestSit); AddLocalPacketHandler(PacketType.AgentSit, HandleAgentSit); AddLocalPacketHandler(PacketType.SoundTrigger, HandleSoundTrigger); AddLocalPacketHandler(PacketType.AvatarPickerRequest, HandleAvatarPickerRequest); AddLocalPacketHandler(PacketType.AgentDataUpdateRequest, HandleAgentDataUpdateRequest); AddLocalPacketHandler(PacketType.UserInfoRequest, HandleUserInfoRequest); AddLocalPacketHandler(PacketType.UpdateUserInfo, HandleUpdateUserInfo); AddLocalPacketHandler(PacketType.SetStartLocationRequest, HandleSetStartLocationRequest); AddLocalPacketHandler(PacketType.AgentThrottle, HandleAgentThrottle, false); AddLocalPacketHandler(PacketType.AgentPause, HandleAgentPause, false); AddLocalPacketHandler(PacketType.AgentResume, HandleAgentResume, false); AddLocalPacketHandler(PacketType.ForceScriptControlRelease, HandleForceScriptControlRelease); AddLocalPacketHandler(PacketType.ObjectLink, HandleObjectLink); AddLocalPacketHandler(PacketType.ObjectDelink, HandleObjectDelink); AddLocalPacketHandler(PacketType.ObjectAdd, HandleObjectAdd); AddLocalPacketHandler(PacketType.ObjectShape, HandleObjectShape); AddLocalPacketHandler(PacketType.ObjectExtraParams, HandleObjectExtraParams); AddLocalPacketHandler(PacketType.ObjectDuplicate, HandleObjectDuplicate); AddLocalPacketHandler(PacketType.RequestMultipleObjects, HandleRequestMultipleObjects); AddLocalPacketHandler(PacketType.ObjectSelect, HandleObjectSelect); AddLocalPacketHandler(PacketType.ObjectDeselect, HandleObjectDeselect); AddLocalPacketHandler(PacketType.ObjectPosition, HandleObjectPosition); AddLocalPacketHandler(PacketType.ObjectScale, HandleObjectScale); AddLocalPacketHandler(PacketType.ObjectRotation, HandleObjectRotation); AddLocalPacketHandler(PacketType.ObjectFlagUpdate, HandleObjectFlagUpdate); // Handle ObjectImage (TextureEntry) updates synchronously, since when updating multiple prim faces at once, // some clients will send out a separate ObjectImage packet for each face AddLocalPacketHandler(PacketType.ObjectImage, HandleObjectImage, false); AddLocalPacketHandler(PacketType.ObjectGrab, HandleObjectGrab, false); AddLocalPacketHandler(PacketType.ObjectGrabUpdate, HandleObjectGrabUpdate, false); AddLocalPacketHandler(PacketType.ObjectDeGrab, HandleObjectDeGrab); AddLocalPacketHandler(PacketType.ObjectSpinStart, HandleObjectSpinStart, false); AddLocalPacketHandler(PacketType.ObjectSpinUpdate, HandleObjectSpinUpdate, false); AddLocalPacketHandler(PacketType.ObjectSpinStop, HandleObjectSpinStop, false); AddLocalPacketHandler(PacketType.ObjectDescription, HandleObjectDescription, false); AddLocalPacketHandler(PacketType.ObjectName, HandleObjectName, false); AddLocalPacketHandler(PacketType.ObjectPermissions, HandleObjectPermissions, false); AddLocalPacketHandler(PacketType.Undo, HandleUndo, false); AddLocalPacketHandler(PacketType.UndoLand, HandleLandUndo, false); AddLocalPacketHandler(PacketType.Redo, HandleRedo, false); AddLocalPacketHandler(PacketType.ObjectDuplicateOnRay, HandleObjectDuplicateOnRay); AddLocalPacketHandler(PacketType.RequestObjectPropertiesFamily, HandleRequestObjectPropertiesFamily, false); AddLocalPacketHandler(PacketType.ObjectIncludeInSearch, HandleObjectIncludeInSearch); AddLocalPacketHandler(PacketType.ScriptAnswerYes, HandleScriptAnswerYes, false); AddLocalPacketHandler(PacketType.ObjectClickAction, HandleObjectClickAction, false); AddLocalPacketHandler(PacketType.ObjectMaterial, HandleObjectMaterial, false); AddLocalPacketHandler(PacketType.RequestImage, HandleRequestImage, false); AddLocalPacketHandler(PacketType.TransferRequest, HandleTransferRequest, false); AddLocalPacketHandler(PacketType.AssetUploadRequest, HandleAssetUploadRequest); AddLocalPacketHandler(PacketType.RequestXfer, HandleRequestXfer); AddLocalPacketHandler(PacketType.SendXferPacket, HandleSendXferPacket); AddLocalPacketHandler(PacketType.ConfirmXferPacket, HandleConfirmXferPacket); AddLocalPacketHandler(PacketType.AbortXfer, HandleAbortXfer); AddLocalPacketHandler(PacketType.CreateInventoryFolder, HandleCreateInventoryFolder); AddLocalPacketHandler(PacketType.UpdateInventoryFolder, HandleUpdateInventoryFolder); AddLocalPacketHandler(PacketType.MoveInventoryFolder, HandleMoveInventoryFolder); AddLocalPacketHandler(PacketType.CreateInventoryItem, HandleCreateInventoryItem); AddLocalPacketHandler(PacketType.LinkInventoryItem, HandleLinkInventoryItem); AddLocalPacketHandler(PacketType.FetchInventory, HandleFetchInventory); AddLocalPacketHandler(PacketType.FetchInventoryDescendents, HandleFetchInventoryDescendents); AddLocalPacketHandler(PacketType.PurgeInventoryDescendents, HandlePurgeInventoryDescendents); AddLocalPacketHandler(PacketType.UpdateInventoryItem, HandleUpdateInventoryItem); AddLocalPacketHandler(PacketType.CopyInventoryItem, HandleCopyInventoryItem); AddLocalPacketHandler(PacketType.MoveInventoryItem, HandleMoveInventoryItem); AddLocalPacketHandler(PacketType.RemoveInventoryItem, HandleRemoveInventoryItem); AddLocalPacketHandler(PacketType.RemoveInventoryFolder, HandleRemoveInventoryFolder); AddLocalPacketHandler(PacketType.RemoveInventoryObjects, HandleRemoveInventoryObjects); AddLocalPacketHandler(PacketType.RequestTaskInventory, HandleRequestTaskInventory); AddLocalPacketHandler(PacketType.UpdateTaskInventory, HandleUpdateTaskInventory); AddLocalPacketHandler(PacketType.RemoveTaskInventory, HandleRemoveTaskInventory); AddLocalPacketHandler(PacketType.MoveTaskInventory, HandleMoveTaskInventory); AddLocalPacketHandler(PacketType.RezScript, HandleRezScript); AddLocalPacketHandler(PacketType.MapLayerRequest, HandleMapLayerRequest); AddLocalPacketHandler(PacketType.MapBlockRequest, HandleMapBlockRequest); AddLocalPacketHandler(PacketType.MapNameRequest, HandleMapNameRequest); AddLocalPacketHandler(PacketType.TeleportLandmarkRequest, HandleTeleportLandmarkRequest); AddLocalPacketHandler(PacketType.TeleportCancel, HandleTeleportCancel); AddLocalPacketHandler(PacketType.TeleportLocationRequest, HandleTeleportLocationRequest); AddLocalPacketHandler(PacketType.UUIDNameRequest, HandleUUIDNameRequest, false); AddLocalPacketHandler(PacketType.RegionHandleRequest, HandleRegionHandleRequest, false); AddLocalPacketHandler(PacketType.ParcelInfoRequest, HandleParcelInfoRequest); AddLocalPacketHandler(PacketType.ParcelAccessListRequest, HandleParcelAccessListRequest, false); AddLocalPacketHandler(PacketType.ParcelAccessListUpdate, HandleParcelAccessListUpdate, false); AddLocalPacketHandler(PacketType.ParcelPropertiesRequest, HandleParcelPropertiesRequest, false); AddLocalPacketHandler(PacketType.ParcelDivide, HandleParcelDivide); AddLocalPacketHandler(PacketType.ParcelJoin, HandleParcelJoin); AddLocalPacketHandler(PacketType.ParcelPropertiesUpdate, HandleParcelPropertiesUpdate); AddLocalPacketHandler(PacketType.ParcelSelectObjects, HandleParcelSelectObjects); AddLocalPacketHandler(PacketType.ParcelObjectOwnersRequest, HandleParcelObjectOwnersRequest); AddLocalPacketHandler(PacketType.ParcelGodForceOwner, HandleParcelGodForceOwner); AddLocalPacketHandler(PacketType.ParcelRelease, HandleParcelRelease); AddLocalPacketHandler(PacketType.ParcelReclaim, HandleParcelReclaim); AddLocalPacketHandler(PacketType.ParcelReturnObjects, HandleParcelReturnObjects); AddLocalPacketHandler(PacketType.ParcelSetOtherCleanTime, HandleParcelSetOtherCleanTime); AddLocalPacketHandler(PacketType.LandStatRequest, HandleLandStatRequest); AddLocalPacketHandler(PacketType.ParcelDwellRequest, HandleParcelDwellRequest); AddLocalPacketHandler(PacketType.EstateOwnerMessage, HandleEstateOwnerMessage); AddLocalPacketHandler(PacketType.RequestRegionInfo, HandleRequestRegionInfo, false); AddLocalPacketHandler(PacketType.EstateCovenantRequest, HandleEstateCovenantRequest); AddLocalPacketHandler(PacketType.RequestGodlikePowers, HandleRequestGodlikePowers); AddLocalPacketHandler(PacketType.GodKickUser, HandleGodKickUser); AddLocalPacketHandler(PacketType.MoneyBalanceRequest, HandleMoneyBalanceRequest); AddLocalPacketHandler(PacketType.EconomyDataRequest, HandleEconomyDataRequest); AddLocalPacketHandler(PacketType.RequestPayPrice, HandleRequestPayPrice); AddLocalPacketHandler(PacketType.ObjectSaleInfo, HandleObjectSaleInfo); AddLocalPacketHandler(PacketType.ObjectBuy, HandleObjectBuy); AddLocalPacketHandler(PacketType.GetScriptRunning, HandleGetScriptRunning); AddLocalPacketHandler(PacketType.SetScriptRunning, HandleSetScriptRunning); AddLocalPacketHandler(PacketType.ScriptReset, HandleScriptReset); AddLocalPacketHandler(PacketType.ActivateGestures, HandleActivateGestures); AddLocalPacketHandler(PacketType.DeactivateGestures, HandleDeactivateGestures); AddLocalPacketHandler(PacketType.ObjectOwner, HandleObjectOwner); AddLocalPacketHandler(PacketType.AgentFOV, HandleAgentFOV, false); AddLocalPacketHandler(PacketType.ViewerStats, HandleViewerStats); AddLocalPacketHandler(PacketType.MapItemRequest, HandleMapItemRequest, false); AddLocalPacketHandler(PacketType.TransferAbort, HandleTransferAbort, false); AddLocalPacketHandler(PacketType.MuteListRequest, HandleMuteListRequest, false); AddLocalPacketHandler(PacketType.UseCircuitCode, HandleUseCircuitCode); AddLocalPacketHandler(PacketType.CreateNewOutfitAttachments, HandleCreateNewOutfitAttachments); AddLocalPacketHandler(PacketType.AgentHeightWidth, HandleAgentHeightWidth, false); AddLocalPacketHandler(PacketType.InventoryDescendents, HandleInventoryDescendents); AddLocalPacketHandler(PacketType.DirPlacesQuery, HandleDirPlacesQuery); AddLocalPacketHandler(PacketType.DirFindQuery, HandleDirFindQuery); AddLocalPacketHandler(PacketType.DirLandQuery, HandleDirLandQuery); AddLocalPacketHandler(PacketType.DirPopularQuery, HandleDirPopularQuery); AddLocalPacketHandler(PacketType.DirClassifiedQuery, HandleDirClassifiedQuery); AddLocalPacketHandler(PacketType.EventInfoRequest, HandleEventInfoRequest); AddLocalPacketHandler(PacketType.OfferCallingCard, HandleOfferCallingCard); AddLocalPacketHandler(PacketType.AcceptCallingCard, HandleAcceptCallingCard); AddLocalPacketHandler(PacketType.DeclineCallingCard, HandleDeclineCallingCard); AddLocalPacketHandler(PacketType.ActivateGroup, HandleActivateGroup); AddLocalPacketHandler(PacketType.GroupTitlesRequest, HandleGroupTitlesRequest); AddLocalPacketHandler(PacketType.GroupProfileRequest, HandleGroupProfileRequest); AddLocalPacketHandler(PacketType.GroupMembersRequest, HandleGroupMembersRequest); AddLocalPacketHandler(PacketType.GroupRoleDataRequest, HandleGroupRoleDataRequest); AddLocalPacketHandler(PacketType.GroupRoleMembersRequest, HandleGroupRoleMembersRequest); AddLocalPacketHandler(PacketType.CreateGroupRequest, HandleCreateGroupRequest); AddLocalPacketHandler(PacketType.UpdateGroupInfo, HandleUpdateGroupInfo); AddLocalPacketHandler(PacketType.SetGroupAcceptNotices, HandleSetGroupAcceptNotices); AddLocalPacketHandler(PacketType.GroupTitleUpdate, HandleGroupTitleUpdate); AddLocalPacketHandler(PacketType.ParcelDeedToGroup, HandleParcelDeedToGroup); AddLocalPacketHandler(PacketType.GroupNoticesListRequest, HandleGroupNoticesListRequest); AddLocalPacketHandler(PacketType.GroupNoticeRequest, HandleGroupNoticeRequest); AddLocalPacketHandler(PacketType.GroupRoleUpdate, HandleGroupRoleUpdate); AddLocalPacketHandler(PacketType.GroupRoleChanges, HandleGroupRoleChanges); AddLocalPacketHandler(PacketType.JoinGroupRequest, HandleJoinGroupRequest); AddLocalPacketHandler(PacketType.LeaveGroupRequest, HandleLeaveGroupRequest); AddLocalPacketHandler(PacketType.EjectGroupMemberRequest, HandleEjectGroupMemberRequest); AddLocalPacketHandler(PacketType.InviteGroupRequest, HandleInviteGroupRequest); AddLocalPacketHandler(PacketType.StartLure, HandleStartLure); AddLocalPacketHandler(PacketType.TeleportLureRequest, HandleTeleportLureRequest); AddLocalPacketHandler(PacketType.ClassifiedInfoRequest, HandleClassifiedInfoRequest); AddLocalPacketHandler(PacketType.ClassifiedInfoUpdate, HandleClassifiedInfoUpdate); AddLocalPacketHandler(PacketType.ClassifiedDelete, HandleClassifiedDelete); AddLocalPacketHandler(PacketType.ClassifiedGodDelete, HandleClassifiedGodDelete); AddLocalPacketHandler(PacketType.EventGodDelete, HandleEventGodDelete); AddLocalPacketHandler(PacketType.EventNotificationAddRequest, HandleEventNotificationAddRequest); AddLocalPacketHandler(PacketType.EventNotificationRemoveRequest, HandleEventNotificationRemoveRequest); AddLocalPacketHandler(PacketType.RetrieveInstantMessages, HandleRetrieveInstantMessages); AddLocalPacketHandler(PacketType.PickDelete, HandlePickDelete); AddLocalPacketHandler(PacketType.PickGodDelete, HandlePickGodDelete); AddLocalPacketHandler(PacketType.PickInfoUpdate, HandlePickInfoUpdate); AddLocalPacketHandler(PacketType.AvatarNotesUpdate, HandleAvatarNotesUpdate); AddLocalPacketHandler(PacketType.AvatarInterestsUpdate, HandleAvatarInterestsUpdate); AddLocalPacketHandler(PacketType.GrantUserRights, HandleGrantUserRights); AddLocalPacketHandler(PacketType.PlacesQuery, HandlePlacesQuery); AddLocalPacketHandler(PacketType.UpdateMuteListEntry, HandleUpdateMuteListEntry); AddLocalPacketHandler(PacketType.RemoveMuteListEntry, HandleRemoveMuteListEntry); AddLocalPacketHandler(PacketType.UserReport, HandleUserReport); AddLocalPacketHandler(PacketType.FindAgent, HandleFindAgent); AddLocalPacketHandler(PacketType.TrackAgent, HandleTrackAgent); AddLocalPacketHandler(PacketType.GodUpdateRegionInfo, HandleGodUpdateRegionInfoUpdate); AddLocalPacketHandler(PacketType.GodlikeMessage, HandleGodlikeMessage); AddLocalPacketHandler(PacketType.StateSave, HandleSaveStatePacket); AddLocalPacketHandler(PacketType.GroupAccountDetailsRequest, HandleGroupAccountDetailsRequest); AddLocalPacketHandler(PacketType.GroupAccountSummaryRequest, HandleGroupAccountSummaryRequest); AddLocalPacketHandler(PacketType.GroupAccountTransactionsRequest, HandleGroupTransactionsDetailsRequest); AddLocalPacketHandler(PacketType.FreezeUser, HandleFreezeUser); AddLocalPacketHandler(PacketType.EjectUser, HandleEjectUser); AddLocalPacketHandler(PacketType.ParcelBuyPass, HandleParcelBuyPass); AddLocalPacketHandler(PacketType.ParcelGodMarkAsContent, HandleParcelGodMarkAsContent); AddLocalPacketHandler(PacketType.GroupActiveProposalsRequest, HandleGroupActiveProposalsRequest); AddLocalPacketHandler(PacketType.GroupVoteHistoryRequest, HandleGroupVoteHistoryRequest); AddLocalPacketHandler(PacketType.SimWideDeletes, HandleSimWideDeletes); AddLocalPacketHandler(PacketType.SendPostcard, HandleSendPostcard); AddLocalPacketHandler(PacketType.ChangeInventoryItemFlags, HandleChangeInventoryItemFlags); AddLocalPacketHandler(PacketType.RevokePermissions, HandleRevokePermissions); AddGenericPacketHandler("autopilot", HandleAutopilot); } #region Packet Handlers public int TotalAgentUpdates { get; set; } #region Scene/Avatar // Threshold for body rotation to be a significant agent update // use the abs of cos private const float QDELTABody = 1.0f - 0.00001f; private const float QDELTAHead = 1.0f - 0.00001f; // Threshold for camera rotation to be a significant agent update private const float VDELTA = 0.01f; /// /// This checks the update significance against the last update made. /// /// Can only be called by one thread at a time /// /// public bool CheckAgentUpdateSignificance(AgentUpdatePacket.AgentDataBlock x) { return CheckAgentMovementUpdateSignificance(x) || CheckAgentCameraUpdateSignificance(x); } /// /// This checks the movement/state update significance against the last update made. /// /// Can only be called by one thread at a time /// /// private bool CheckAgentMovementUpdateSignificance(AgentUpdatePacket.AgentDataBlock x) { if( (x.ControlFlags != m_thisAgentUpdateArgs.ControlFlags) // significant if control flags changed // || ((x.ControlFlags & (uint)AgentManager.ControlFlags.AGENT_CONTROL_FLY) != 0 && // (x.ControlFlags & 0x3f8dfff) != 0) // we need to rotate the av on fly || x.ControlFlags != (byte)AgentManager.ControlFlags.NONE// actually all movement controls need to pass || (x.Flags != m_thisAgentUpdateArgs.Flags) // significant if Flags changed || (x.State != m_thisAgentUpdateArgs.State) // significant if Stats changed || (Math.Abs(x.Far - m_thisAgentUpdateArgs.Far) >= 32) // significant if far distance changed ) return true; float qdelta = Math.Abs(Quaternion.Dot(x.BodyRotation, m_thisAgentUpdateArgs.BodyRotation)); if(qdelta < QDELTABody) // significant if body rotation above(below cos) threshold return true; return false; } /// /// This checks the camera update significance against the last update made. /// /// Can only be called by one thread at a time /// /// private bool CheckAgentCameraUpdateSignificance(AgentUpdatePacket.AgentDataBlock x) { if(Math.Abs(x.CameraCenter.X - m_thisAgentUpdateArgs.CameraCenter.X) > VDELTA || Math.Abs(x.CameraCenter.Y - m_thisAgentUpdateArgs.CameraCenter.Y) > VDELTA || Math.Abs(x.CameraCenter.Z - m_thisAgentUpdateArgs.CameraCenter.Z) > VDELTA || Math.Abs(x.CameraAtAxis.X - m_thisAgentUpdateArgs.CameraAtAxis.X) > VDELTA || Math.Abs(x.CameraAtAxis.Y - m_thisAgentUpdateArgs.CameraAtAxis.Y) > VDELTA || // Math.Abs(x.CameraAtAxis.Z - m_thisAgentUpdateArgs.CameraAtAxis.Z) > VDELTA || Math.Abs(x.CameraLeftAxis.X - m_thisAgentUpdateArgs.CameraLeftAxis.X) > VDELTA || Math.Abs(x.CameraLeftAxis.Y - m_thisAgentUpdateArgs.CameraLeftAxis.Y) > VDELTA || // Math.Abs(x.CameraLeftAxis.Z - m_thisAgentUpdateArgs.CameraLeftAxis.Z) > VDELTA || Math.Abs(x.CameraUpAxis.X - m_thisAgentUpdateArgs.CameraUpAxis.X) > VDELTA || Math.Abs(x.CameraUpAxis.Y - m_thisAgentUpdateArgs.CameraUpAxis.Y) > VDELTA // Math.Abs(x.CameraLeftAxis.Z - m_thisAgentUpdateArgs.CameraLeftAxis.Z) > VDELTA || ) return true; return false; } private bool HandleAgentUpdate(IClientAPI sender, Packet packet) { if(OnAgentUpdate == null) { PacketPool.Instance.ReturnPacket(packet); return false; } AgentUpdatePacket agentUpdate = (AgentUpdatePacket)packet; AgentUpdatePacket.AgentDataBlock x = agentUpdate.AgentData; if (x.AgentID != AgentId || x.SessionID != SessionId) { PacketPool.Instance.ReturnPacket(packet); return false; } uint seq = packet.Header.Sequence; TotalAgentUpdates++; // dont let ignored updates pollute this throttles if(SceneAgent == null || SceneAgent.IsChildAgent || SceneAgent.IsInTransit || seq <= m_thisAgentUpdateArgs.lastpacketSequence ) { // throttle reset is done at MoveAgentIntoRegion() // called by scenepresence on completemovement PacketPool.Instance.ReturnPacket(packet); return true; } m_thisAgentUpdateArgs.lastpacketSequence = seq; if (OnPreAgentUpdate != null) OnPreAgentUpdate(this, m_thisAgentUpdateArgs); bool movement; bool camera; double now = Util.GetTimeStampMS(); if(now - m_thisAgentUpdateArgs.lastUpdateTS > 500.0) // at least 2 per sec { movement = true; camera = true; } else { movement = CheckAgentMovementUpdateSignificance(x); camera = CheckAgentCameraUpdateSignificance(x); } // Was there a significant movement/state change? if (movement) { m_thisAgentUpdateArgs.BodyRotation = x.BodyRotation; m_thisAgentUpdateArgs.ControlFlags = x.ControlFlags; m_thisAgentUpdateArgs.Far = x.Far; m_thisAgentUpdateArgs.Flags = x.Flags; m_thisAgentUpdateArgs.HeadRotation = x.HeadRotation; m_thisAgentUpdateArgs.State = x.State; m_thisAgentUpdateArgs.NeedsCameraCollision = !camera; if (OnAgentUpdate != null) OnAgentUpdate(this, m_thisAgentUpdateArgs); } // Was there a significant camera(s) change? if (camera) { m_thisAgentUpdateArgs.CameraAtAxis = x.CameraAtAxis; m_thisAgentUpdateArgs.CameraCenter = x.CameraCenter; m_thisAgentUpdateArgs.CameraLeftAxis = x.CameraLeftAxis; m_thisAgentUpdateArgs.CameraUpAxis = x.CameraUpAxis; m_thisAgentUpdateArgs.NeedsCameraCollision = true; if (OnAgentCameraUpdate != null) OnAgentCameraUpdate(this, m_thisAgentUpdateArgs); } if(movement && camera) m_thisAgentUpdateArgs.lastUpdateTS = now; PacketPool.Instance.ReturnPacket(packet); return true; } private bool HandleMoneyTransferRequest(IClientAPI sender, Packet Pack) { MoneyTransferRequestPacket money = (MoneyTransferRequestPacket)Pack; // validate the agent owns the agentID and sessionID if (money.MoneyData.SourceID == sender.AgentId && money.AgentData.AgentID == sender.AgentId && money.AgentData.SessionID == sender.SessionId) { MoneyTransferRequest handlerMoneyTransferRequest = OnMoneyTransferRequest; if (handlerMoneyTransferRequest != null) { handlerMoneyTransferRequest(money.MoneyData.SourceID, money.MoneyData.DestID, money.MoneyData.Amount, money.MoneyData.TransactionType, Util.FieldToString(money.MoneyData.Description)); } return true; } return false; } private bool HandleParcelGodMarkAsContent(IClientAPI client, Packet Packet) { ParcelGodMarkAsContentPacket ParcelGodMarkAsContent = (ParcelGodMarkAsContentPacket)Packet; if(SessionId != ParcelGodMarkAsContent.AgentData.SessionID || AgentId != ParcelGodMarkAsContent.AgentData.AgentID) return false; ParcelGodMark ParcelGodMarkAsContentHandler = OnParcelGodMark; if (ParcelGodMarkAsContentHandler != null) { ParcelGodMarkAsContentHandler(this, ParcelGodMarkAsContent.AgentData.AgentID, ParcelGodMarkAsContent.ParcelData.LocalID); return true; } return false; } private bool HandleFreezeUser(IClientAPI client, Packet Packet) { FreezeUserPacket FreezeUser = (FreezeUserPacket)Packet; if(SessionId != FreezeUser.AgentData.SessionID || AgentId != FreezeUser.AgentData.AgentID) return false; FreezeUserUpdate FreezeUserHandler = OnParcelFreezeUser; if (FreezeUserHandler != null) { FreezeUserHandler(this, FreezeUser.AgentData.AgentID, FreezeUser.Data.Flags, FreezeUser.Data.TargetID); return true; } return false; } private bool HandleEjectUser(IClientAPI client, Packet Packet) { EjectUserPacket EjectUser = (EjectUserPacket)Packet; if(SessionId != EjectUser.AgentData.SessionID || AgentId != EjectUser.AgentData.AgentID) return false; EjectUserUpdate EjectUserHandler = OnParcelEjectUser; if (EjectUserHandler != null) { EjectUserHandler(this, EjectUser.AgentData.AgentID, EjectUser.Data.Flags, EjectUser.Data.TargetID); return true; } return false; } private bool HandleParcelBuyPass(IClientAPI client, Packet Packet) { ParcelBuyPassPacket ParcelBuyPass = (ParcelBuyPassPacket)Packet; if(SessionId != ParcelBuyPass.AgentData.SessionID || AgentId != ParcelBuyPass.AgentData.AgentID) return false; ParcelBuyPass ParcelBuyPassHandler = OnParcelBuyPass; if (ParcelBuyPassHandler != null) { ParcelBuyPassHandler(this, ParcelBuyPass.AgentData.AgentID, ParcelBuyPass.ParcelData.LocalID); return true; } return false; } private bool HandleParcelBuyRequest(IClientAPI sender, Packet Pack) { ParcelBuyPacket parcel = (ParcelBuyPacket)Pack; if (parcel.AgentData.AgentID == AgentId && parcel.AgentData.SessionID == SessionId) { ParcelBuy handlerParcelBuy = OnParcelBuy; if (handlerParcelBuy != null) { handlerParcelBuy(parcel.AgentData.AgentID, parcel.Data.GroupID, parcel.Data.Final, parcel.Data.IsGroupOwned, parcel.Data.RemoveContribution, parcel.Data.LocalID, parcel.ParcelData.Area, parcel.ParcelData.Price, false); } return true; } return false; } private bool HandleUUIDGroupNameRequest(IClientAPI sender, Packet Pack) { ScenePresence sp = (ScenePresence)SceneAgent; if(sp == null || sp.IsDeleted || (sp.IsInTransit && !sp.IsInLocalTransit)) return true; UUIDGroupNameRequestPacket upack = (UUIDGroupNameRequestPacket)Pack; for (int i = 0; i < upack.UUIDNameBlock.Length; i++) { UUIDNameRequest handlerUUIDGroupNameRequest = OnUUIDGroupNameRequest; if (handlerUUIDGroupNameRequest != null) { handlerUUIDGroupNameRequest(upack.UUIDNameBlock[i].ID, this); } } return true; } public bool HandleGenericMessage(IClientAPI sender, Packet pack) { GenericMessagePacket gmpack = (GenericMessagePacket)pack; if (m_genericPacketHandlers.Count == 0) return false; if (gmpack.AgentData.SessionID != SessionId) return false; GenericMessage handlerGenericMessage = null; string method = Util.FieldToString(gmpack.MethodData.Method).ToLower().Trim(); if (m_genericPacketHandlers.TryGetValue(method, out handlerGenericMessage)) { List msg = new List(); List msgBytes = new List(); if (handlerGenericMessage != null) { foreach (GenericMessagePacket.ParamListBlock block in gmpack.ParamList) { msg.Add(Util.FieldToString(block.Parameter)); msgBytes.Add(block.Parameter); } try { if (OnBinaryGenericMessage != null) { OnBinaryGenericMessage(this, method, msgBytes.ToArray()); } handlerGenericMessage(sender, method, msg); return true; } catch (Exception e) { m_log.ErrorFormat( "[LLCLIENTVIEW]: Exeception when handling generic message {0}{1}", e.Message, e.StackTrace); } } } //m_log.Debug("[LLCLIENTVIEW]: Not handling GenericMessage with method-type of: " + method); return false; } public bool HandleObjectGroupRequest(IClientAPI sender, Packet Pack) { ObjectGroupPacket ogpack = (ObjectGroupPacket)Pack; if (ogpack.AgentData.SessionID != SessionId) return false; RequestObjectPropertiesFamily handlerObjectGroupRequest = OnObjectGroupRequest; if (handlerObjectGroupRequest != null) { for (int i = 0; i < ogpack.ObjectData.Length; i++) { handlerObjectGroupRequest(this, ogpack.AgentData.GroupID, ogpack.ObjectData[i].ObjectLocalID, UUID.Zero); } } return true; } private bool HandleViewerEffect(IClientAPI sender, Packet Pack) { ViewerEffectPacket viewer = (ViewerEffectPacket)Pack; if (viewer.AgentData.SessionID != SessionId) return false; ViewerEffectEventHandler handlerViewerEffect = OnViewerEffect; if (handlerViewerEffect != null) { int length = viewer.Effect.Length; List args = new List(length); for (int i = 0; i < length; i++) { //copy the effects block arguments into the event handler arg. ViewerEffectEventHandlerArg argument = new ViewerEffectEventHandlerArg(); argument.AgentID = viewer.Effect[i].AgentID; argument.Color = viewer.Effect[i].Color; argument.Duration = viewer.Effect[i].Duration; argument.ID = viewer.Effect[i].ID; argument.Type = viewer.Effect[i].Type; argument.TypeData = viewer.Effect[i].TypeData; args.Add(argument); } handlerViewerEffect(sender, args); } return true; } private bool HandleVelocityInterpolateOff(IClientAPI sender, Packet Pack) { VelocityInterpolateOffPacket p = (VelocityInterpolateOffPacket)Pack; if (p.AgentData.SessionID != SessionId || p.AgentData.AgentID != AgentId) return true; m_VelocityInterpolate = false; return true; } private bool HandleVelocityInterpolateOn(IClientAPI sender, Packet Pack) { VelocityInterpolateOnPacket p = (VelocityInterpolateOnPacket)Pack; if (p.AgentData.SessionID != SessionId || p.AgentData.AgentID != AgentId) return true; m_VelocityInterpolate = true; return true; } private bool HandleAvatarPropertiesRequest(IClientAPI sender, Packet Pack) { AvatarPropertiesRequestPacket avatarProperties = (AvatarPropertiesRequestPacket)Pack; #region Packet Session and User Check if (avatarProperties.AgentData.SessionID != SessionId || avatarProperties.AgentData.AgentID != AgentId) return true; #endregion RequestAvatarProperties handlerRequestAvatarProperties = OnRequestAvatarProperties; if (handlerRequestAvatarProperties != null) { handlerRequestAvatarProperties(this, avatarProperties.AgentData.AvatarID); } return true; } private bool HandleChatFromViewer(IClientAPI sender, Packet Pack) { ChatFromViewerPacket inchatpack = (ChatFromViewerPacket)Pack; #region Packet Session and User Check if (inchatpack.AgentData.SessionID != SessionId || inchatpack.AgentData.AgentID != AgentId) return true; #endregion string fromName = String.Empty; //ClientAvatar.firstname + " " + ClientAvatar.lastname; byte[] message = inchatpack.ChatData.Message; byte type = inchatpack.ChatData.Type; Vector3 fromPos = new Vector3(); // ClientAvatar.Pos; // UUID fromAgentID = AgentId; int channel = inchatpack.ChatData.Channel; if (OnChatFromClient != null) { OSChatMessage args = new OSChatMessage(); args.Channel = channel; args.From = fromName; args.Message = Utils.BytesToString(message); args.Type = (ChatTypeEnum)type; args.Position = fromPos; args.Scene = Scene; args.Sender = this; args.SenderUUID = this.AgentId; ChatMessage handlerChatFromClient = OnChatFromClient; if (handlerChatFromClient != null) handlerChatFromClient(this, args); } return true; } private bool HandlerAvatarPropertiesUpdate(IClientAPI sender, Packet Pack) { AvatarPropertiesUpdatePacket avatarProps = (AvatarPropertiesUpdatePacket)Pack; #region Packet Session and User Check if (avatarProps.AgentData.SessionID != SessionId || avatarProps.AgentData.AgentID != AgentId) return true; #endregion UpdateAvatarProperties handlerUpdateAvatarProperties = OnUpdateAvatarProperties; if (handlerUpdateAvatarProperties != null) { AvatarPropertiesUpdatePacket.PropertiesDataBlock Properties = avatarProps.PropertiesData; UserProfileData UserProfile = new UserProfileData(); UserProfile.ID = AgentId; UserProfile.AboutText = Utils.BytesToString(Properties.AboutText); UserProfile.FirstLifeAboutText = Utils.BytesToString(Properties.FLAboutText); UserProfile.FirstLifeImage = Properties.FLImageID; UserProfile.Image = Properties.ImageID; UserProfile.ProfileUrl = Utils.BytesToString(Properties.ProfileURL); UserProfile.UserFlags &= ~3; UserProfile.UserFlags |= Properties.AllowPublish ? 1 : 0; UserProfile.UserFlags |= Properties.MaturePublish ? 2 : 0; handlerUpdateAvatarProperties(this, UserProfile); } return true; } private bool HandlerScriptDialogReply(IClientAPI sender, Packet Pack) { ScriptDialogReplyPacket rdialog = (ScriptDialogReplyPacket)Pack; //m_log.DebugFormat("[CLIENT]: Received ScriptDialogReply from {0}", rdialog.Data.ObjectID); #region Packet Session and User Check if (rdialog.AgentData.SessionID != SessionId || rdialog.AgentData.AgentID != AgentId) return true; #endregion int ch = rdialog.Data.ChatChannel; byte[] msg = rdialog.Data.ButtonLabel; if (OnChatFromClient != null) { OSChatMessage args = new OSChatMessage(); args.Channel = ch; args.From = String.Empty; args.Message = Utils.BytesToString(msg); args.Type = ChatTypeEnum.Region; //Behaviour in SL is that the response can be heard from any distance args.Position = new Vector3(); args.Scene = Scene; args.Sender = this; ChatMessage handlerChatFromClient2 = OnChatFromClient; if (handlerChatFromClient2 != null) handlerChatFromClient2(this, args); } return true; } private bool HandlerImprovedInstantMessage(IClientAPI sender, Packet Pack) { ImprovedInstantMessagePacket msgpack = (ImprovedInstantMessagePacket)Pack; #region Packet Session and User Check if (msgpack.AgentData.SessionID != SessionId || msgpack.AgentData.AgentID != AgentId) return true; #endregion string IMfromName = Util.FieldToString(msgpack.MessageBlock.FromAgentName); string IMmessage = Utils.BytesToString(msgpack.MessageBlock.Message); ImprovedInstantMessage handlerInstantMessage = OnInstantMessage; if (handlerInstantMessage != null) { GridInstantMessage im = new GridInstantMessage(Scene, msgpack.AgentData.AgentID, IMfromName, msgpack.MessageBlock.ToAgentID, msgpack.MessageBlock.Dialog, msgpack.MessageBlock.FromGroup, IMmessage, msgpack.MessageBlock.ID, msgpack.MessageBlock.Offline != 0 ? true : false, msgpack.MessageBlock.Position, msgpack.MessageBlock.BinaryBucket, true); handlerInstantMessage(this, im); } return true; } private bool HandlerAcceptFriendship(IClientAPI sender, Packet Pack) { AcceptFriendshipPacket afriendpack = (AcceptFriendshipPacket)Pack; #region Packet Session and User Check if (afriendpack.AgentData.SessionID != SessionId || afriendpack.AgentData.AgentID != AgentId) return true; #endregion // My guess is this is the folder to stick the calling card into List callingCardFolders = new List(); UUID transactionID = afriendpack.TransactionBlock.TransactionID; for (int fi = 0; fi < afriendpack.FolderData.Length; fi++) { callingCardFolders.Add(afriendpack.FolderData[fi].FolderID); } FriendActionDelegate handlerApproveFriendRequest = OnApproveFriendRequest; if (handlerApproveFriendRequest != null) { handlerApproveFriendRequest(this, transactionID, callingCardFolders); } return true; } private bool HandlerDeclineFriendship(IClientAPI sender, Packet Pack) { DeclineFriendshipPacket dfriendpack = (DeclineFriendshipPacket)Pack; #region Packet Session and User Check if (dfriendpack.AgentData.SessionID != SessionId || dfriendpack.AgentData.AgentID != AgentId) return true; #endregion if (OnDenyFriendRequest != null) { OnDenyFriendRequest(this, dfriendpack.TransactionBlock.TransactionID, null); } return true; } private bool HandlerTerminateFriendship(IClientAPI sender, Packet Pack) { TerminateFriendshipPacket tfriendpack = (TerminateFriendshipPacket)Pack; #region Packet Session and User Check if (tfriendpack.AgentData.SessionID != SessionId || tfriendpack.AgentData.AgentID != AgentId) return true; #endregion UUID exFriendID = tfriendpack.ExBlock.OtherID; FriendshipTermination TerminateFriendshipHandler = OnTerminateFriendship; if (TerminateFriendshipHandler != null) { TerminateFriendshipHandler(this, exFriendID); return true; } return false; } private bool HandleFindAgent(IClientAPI client, Packet Packet) { FindAgentPacket FindAgent = (FindAgentPacket)Packet; FindAgentUpdate FindAgentHandler = OnFindAgent; if (FindAgentHandler != null) { FindAgentHandler(this,FindAgent.AgentBlock.Hunter,FindAgent.AgentBlock.Prey); return true; } return false; } private bool HandleTrackAgent(IClientAPI client, Packet Packet) { TrackAgentPacket TrackAgent = (TrackAgentPacket)Packet; TrackAgentUpdate TrackAgentHandler = OnTrackAgent; if (TrackAgentHandler != null) { TrackAgentHandler(this, TrackAgent.AgentData.AgentID, TrackAgent.TargetData.PreyID); return true; } return false; } private bool HandlerRezObject(IClientAPI sender, Packet Pack) { RezObjectPacket rezPacket = (RezObjectPacket)Pack; #region Packet Session and User Check if (rezPacket.AgentData.SessionID != SessionId || rezPacket.AgentData.AgentID != AgentId) return true; #endregion RezObject handlerRezObject = OnRezObject; if (handlerRezObject != null) { UUID rezGroupID = rezPacket.AgentData.GroupID; if(!IsGroupMember(rezGroupID)) rezGroupID = UUID.Zero; handlerRezObject(this, rezPacket.InventoryData.ItemID, rezGroupID, rezPacket.RezData.RayEnd, rezPacket.RezData.RayStart, rezPacket.RezData.RayTargetID, rezPacket.RezData.BypassRaycast, rezPacket.RezData.RayEndIsIntersection, rezPacket.RezData.RezSelected, rezPacket.RezData.RemoveItem, rezPacket.RezData.FromTaskID); } return true; } private class DeRezObjectInfo { public int count; public List objectids; } private Dictionary m_DeRezObjectDelayed = new Dictionary(); private bool HandlerDeRezObject(IClientAPI sender, Packet Pack) { DeRezObject handlerDeRezObject = OnDeRezObject; if (handlerDeRezObject == null) return true; DeRezObjectPacket DeRezPacket = (DeRezObjectPacket)Pack; #region Packet Session and User Check if (DeRezPacket.AgentData.SessionID != SessionId || DeRezPacket.AgentData.AgentID != AgentId) return true; #endregion List deRezIDs; DeRezAction action = (DeRezAction)DeRezPacket.AgentBlock.Destination; int numberPackets = DeRezPacket.AgentBlock.PacketCount; int curPacket = DeRezPacket.AgentBlock.PacketNumber; UUID id = DeRezPacket.AgentBlock.TransactionID; if (numberPackets > 1) { DeRezObjectInfo info; if (!m_DeRezObjectDelayed.TryGetValue(id, out info)) { deRezIDs = new List(); info = new DeRezObjectInfo(); info.count = 0; info.objectids = deRezIDs; m_DeRezObjectDelayed[id] = info; } else { deRezIDs = info.objectids; } foreach (DeRezObjectPacket.ObjectDataBlock data in DeRezPacket.ObjectData) { deRezIDs.Add(data.ObjectLocalID); } info.count++; if (info.count < numberPackets) return true; m_DeRezObjectDelayed.Remove(id); info.objectids = null; } else { deRezIDs = new List(); foreach (DeRezObjectPacket.ObjectDataBlock data in DeRezPacket.ObjectData) { deRezIDs.Add(data.ObjectLocalID); } } if (handlerDeRezObject != null) handlerDeRezObject(this, deRezIDs, DeRezPacket.AgentBlock.GroupID, action, DeRezPacket.AgentBlock.DestinationID); return true; } private bool HandlerRezRestoreToWorld(IClientAPI sender, Packet Pack) { RezRestoreToWorldPacket restore = (RezRestoreToWorldPacket)Pack; #region Packet Session and User Check if (restore.AgentData.SessionID != SessionId || restore.AgentData.AgentID != AgentId) return true; #endregion RezRestoreToWorld handlerRezRestoreToWorld = OnRezRestoreToWorld; if (handlerRezRestoreToWorld != null) handlerRezRestoreToWorld(this, restore.InventoryData.ItemID); return true; } private bool HandlerModifyLand(IClientAPI sender, Packet Pack) { ModifyLandPacket modify = (ModifyLandPacket)Pack; #region Packet Session and User Check if (modify.AgentData.SessionID != SessionId || modify.AgentData.AgentID != AgentId) return true; #endregion //m_log.Info("[LAND]: LAND:" + modify.ToString()); if (modify.ParcelData.Length > 0) { // Note: the ModifyTerrain event handler sends out updated packets before the end of this event. Therefore, // a simple boolean value should work and perhaps queue up just a few terrain patch packets at the end of the edit. if (OnModifyTerrain != null) { for (int i = 0; i < modify.ParcelData.Length; i++) { ModifyTerrain handlerModifyTerrain = OnModifyTerrain; if (handlerModifyTerrain != null) { handlerModifyTerrain(AgentId, modify.ModifyBlock.Height, modify.ModifyBlock.Seconds, modify.ModifyBlock.BrushSize, modify.ModifyBlock.Action, modify.ParcelData[i].North, modify.ParcelData[i].West, modify.ParcelData[i].South, modify.ParcelData[i].East, AgentId); } } } } return true; } private bool HandlerRegionHandshakeReply(IClientAPI sender, Packet Pack) { Action handlerRegionHandShakeReply = OnRegionHandShakeReply; if (handlerRegionHandShakeReply != null) { handlerRegionHandShakeReply(this); } return true; } private bool HandlerAgentWearablesRequest(IClientAPI sender, Packet Pack) { GenericCall1 handlerRequestWearables = OnRequestWearables; if (handlerRequestWearables != null) { handlerRequestWearables(sender); } Action handlerRequestAvatarsData = OnRequestAvatarsData; if (handlerRequestAvatarsData != null) { handlerRequestAvatarsData(this); } return true; } private bool HandlerAgentSetAppearance(IClientAPI sender, Packet Pack) { AgentSetAppearancePacket appear = (AgentSetAppearancePacket)Pack; #region Packet Session and User Check if (appear.AgentData.SessionID != SessionId || appear.AgentData.AgentID != AgentId) return true; #endregion SetAppearance handlerSetAppearance = OnSetAppearance; if (handlerSetAppearance != null) { // Temporarily protect ourselves from the mantis #951 failure. // However, we could do this for several other handlers where a failure isn't terminal // for the client session anyway, in order to protect ourselves against bad code in plugins Vector3 avSize = appear.AgentData.Size; try { byte[] visualparams = new byte[appear.VisualParam.Length]; for (int i = 0; i < appear.VisualParam.Length; i++) visualparams[i] = appear.VisualParam[i].ParamValue; //var b = appear.WearableData[0]; Primitive.TextureEntry te = null; if (appear.ObjectData.TextureEntry.Length > 1) te = new Primitive.TextureEntry(appear.ObjectData.TextureEntry, 0, appear.ObjectData.TextureEntry.Length); WearableCacheItem[] cacheitems = new WearableCacheItem[appear.WearableData.Length]; for (int i=0; i> rezlist = new List>(); foreach (RezMultipleAttachmentsFromInvPacket.ObjectDataBlock obj in ((RezMultipleAttachmentsFromInvPacket)Pack).ObjectData) rezlist.Add(new KeyValuePair(obj.ItemID, obj.AttachmentPt)); handlerRezMultipleAttachments(this, rezlist); } return true; } private bool HandleDetachAttachmentIntoInv(IClientAPI sender, Packet Pack) { UUIDNameRequest handlerDetachAttachmentIntoInv = OnDetachAttachmentIntoInv; if (handlerDetachAttachmentIntoInv != null) { DetachAttachmentIntoInvPacket detachtoInv = (DetachAttachmentIntoInvPacket)Pack; #region Packet Session and User Check // UNSUPPORTED ON THIS PACKET #endregion UUID itemID = detachtoInv.ObjectData.ItemID; // UUID ATTACH_agentID = detachtoInv.ObjectData.AgentID; handlerDetachAttachmentIntoInv(itemID, this); } return true; } private bool HandleObjectAttach(IClientAPI sender, Packet Pack) { if (OnObjectAttach != null) { ObjectAttachPacket att = (ObjectAttachPacket)Pack; #region Packet Session and User Check if (att.AgentData.SessionID != SessionId || att.AgentData.AgentID != AgentId) return true; #endregion ObjectAttach handlerObjectAttach = OnObjectAttach; if (handlerObjectAttach != null) { if (att.ObjectData.Length > 0) { handlerObjectAttach(this, att.ObjectData[0].ObjectLocalID, att.AgentData.AttachmentPoint, false); } } } return true; } private bool HandleObjectDetach(IClientAPI sender, Packet Pack) { ObjectDetachPacket dett = (ObjectDetachPacket)Pack; #region Packet Session and User Check if (dett.AgentData.SessionID != SessionId || dett.AgentData.AgentID != AgentId) return true; #endregion for (int j = 0; j < dett.ObjectData.Length; j++) { uint obj = dett.ObjectData[j].ObjectLocalID; ObjectDeselect handlerObjectDetach = OnObjectDetach; if (handlerObjectDetach != null) { handlerObjectDetach(obj, this); } } return true; } private bool HandleObjectDrop(IClientAPI sender, Packet Pack) { ObjectDropPacket dropp = (ObjectDropPacket)Pack; #region Packet Session and User Check if (dropp.AgentData.SessionID != SessionId || dropp.AgentData.AgentID != AgentId) return true; #endregion for (int j = 0; j < dropp.ObjectData.Length; j++) { uint obj = dropp.ObjectData[j].ObjectLocalID; ObjectDrop handlerObjectDrop = OnObjectDrop; if (handlerObjectDrop != null) { handlerObjectDrop(obj, this); } } return true; } private bool HandleSetAlwaysRun(IClientAPI sender, Packet Pack) { SetAlwaysRunPacket run = (SetAlwaysRunPacket)Pack; #region Packet Session and User Check if (run.AgentData.SessionID != SessionId || run.AgentData.AgentID != AgentId) return true; #endregion SetAlwaysRun handlerSetAlwaysRun = OnSetAlwaysRun; if (handlerSetAlwaysRun != null) handlerSetAlwaysRun(this, run.AgentData.AlwaysRun); return true; } private bool HandleCompleteAgentMovement(IClientAPI sender, Packet Pack) { m_log.DebugFormat("[LLClientView] HandleCompleteAgentMovement"); Action handlerCompleteMovementToRegion = OnCompleteMovementToRegion; if (handlerCompleteMovementToRegion != null) { handlerCompleteMovementToRegion(sender, true); } else m_log.Debug("HandleCompleteAgentMovement NULL handler"); handlerCompleteMovementToRegion = null; return true; } private bool HandleAgentAnimation(IClientAPI sender, Packet Pack) { AgentAnimationPacket AgentAni = (AgentAnimationPacket)Pack; #region Packet Session and User Check if (AgentAni.AgentData.SessionID != SessionId || AgentAni.AgentData.AgentID != AgentId) return true; #endregion ChangeAnim handlerChangeAnim = null; for (int i = 0; i < AgentAni.AnimationList.Length; i++) { handlerChangeAnim = OnChangeAnim; if (handlerChangeAnim != null) { handlerChangeAnim(AgentAni.AnimationList[i].AnimID, AgentAni.AnimationList[i].StartAnim, false); } } handlerChangeAnim = OnChangeAnim; if (handlerChangeAnim != null) { handlerChangeAnim(UUID.Zero, false, true); } return true; } private bool HandleAgentRequestSit(IClientAPI sender, Packet Pack) { if (OnAgentRequestSit != null) { AgentRequestSitPacket agentRequestSit = (AgentRequestSitPacket)Pack; #region Packet Session and User Check if (agentRequestSit.AgentData.SessionID != SessionId || agentRequestSit.AgentData.AgentID != AgentId) return true; #endregion if (SceneAgent.IsChildAgent) { SendCantSitBecauseChildAgentResponse(); return true; } AgentRequestSit handlerAgentRequestSit = OnAgentRequestSit; if (handlerAgentRequestSit != null) handlerAgentRequestSit(this, agentRequestSit.AgentData.AgentID, agentRequestSit.TargetObject.TargetID, agentRequestSit.TargetObject.Offset); } return true; } private bool HandleAgentSit(IClientAPI sender, Packet Pack) { if (OnAgentSit != null) { AgentSitPacket agentSit = (AgentSitPacket)Pack; #region Packet Session and User Check if (agentSit.AgentData.SessionID != SessionId || agentSit.AgentData.AgentID != AgentId) return true; #endregion if (SceneAgent.IsChildAgent) { SendCantSitBecauseChildAgentResponse(); return true; } AgentSit handlerAgentSit = OnAgentSit; if (handlerAgentSit != null) { OnAgentSit(this, agentSit.AgentData.AgentID); } } return true; } /// /// Used when a child agent gets a sit response which should not be fulfilled. /// private void SendCantSitBecauseChildAgentResponse() { SendAlertMessage("Try moving closer. Can't sit on object because it is not in the same region as you."); } private bool HandleSoundTrigger(IClientAPI sender, Packet Pack) { SoundTriggerPacket soundTriggerPacket = (SoundTriggerPacket)Pack; #region Packet Session and User Check #endregion SoundTrigger handlerSoundTrigger = OnSoundTrigger; if (handlerSoundTrigger != null) { // UUIDS are sent as zeroes by the client, substitute agent's id handlerSoundTrigger(soundTriggerPacket.SoundData.SoundID, AgentId, AgentId, AgentId, soundTriggerPacket.SoundData.Gain, soundTriggerPacket.SoundData.Position, soundTriggerPacket.SoundData.Handle); } return true; } private bool HandleAvatarPickerRequest(IClientAPI sender, Packet Pack) { AvatarPickerRequestPacket avRequestQuery = (AvatarPickerRequestPacket)Pack; #region Packet Session and User Check if (avRequestQuery.AgentData.SessionID != SessionId || avRequestQuery.AgentData.AgentID != AgentId) return true; #endregion AvatarPickerRequestPacket.AgentDataBlock Requestdata = avRequestQuery.AgentData; AvatarPickerRequestPacket.DataBlock querydata = avRequestQuery.Data; //m_log.Debug("Agent Sends:" + Utils.BytesToString(querydata.Name)); AvatarPickerRequest handlerAvatarPickerRequest = OnAvatarPickerRequest; if (handlerAvatarPickerRequest != null) { handlerAvatarPickerRequest(this, Requestdata.AgentID, Requestdata.QueryID, Utils.BytesToString(querydata.Name)); } return true; } private bool HandleAgentDataUpdateRequest(IClientAPI sender, Packet Pack) { AgentDataUpdateRequestPacket avRequestDataUpdatePacket = (AgentDataUpdateRequestPacket)Pack; #region Packet Session and User Check if (avRequestDataUpdatePacket.AgentData.SessionID != SessionId || avRequestDataUpdatePacket.AgentData.AgentID != AgentId) return true; #endregion FetchInventory handlerAgentDataUpdateRequest = OnAgentDataUpdateRequest; if (handlerAgentDataUpdateRequest != null) { handlerAgentDataUpdateRequest(this, avRequestDataUpdatePacket.AgentData.AgentID, avRequestDataUpdatePacket.AgentData.SessionID); } return true; } private bool HandleUserInfoRequest(IClientAPI sender, Packet Pack) { UserInfoRequest handlerUserInfoRequest = OnUserInfoRequest; if (handlerUserInfoRequest != null) { handlerUserInfoRequest(this); } else { SendUserInfoReply(false, true, ""); } return true; } private bool HandleUpdateUserInfo(IClientAPI sender, Packet Pack) { UpdateUserInfoPacket updateUserInfo = (UpdateUserInfoPacket)Pack; #region Packet Session and User Check if (updateUserInfo.AgentData.SessionID != SessionId || updateUserInfo.AgentData.AgentID != AgentId) return true; #endregion UpdateUserInfo handlerUpdateUserInfo = OnUpdateUserInfo; if (handlerUpdateUserInfo != null) { bool visible = true; string DirectoryVisibility = Utils.BytesToString(updateUserInfo.UserData.DirectoryVisibility); if (DirectoryVisibility == "hidden") visible = false; handlerUpdateUserInfo( updateUserInfo.UserData.IMViaEMail, visible, this); } return true; } private bool HandleSetStartLocationRequest(IClientAPI sender, Packet Pack) { SetStartLocationRequestPacket avSetStartLocationRequestPacket = (SetStartLocationRequestPacket)Pack; #region Packet Session and User Check if (avSetStartLocationRequestPacket.AgentData.SessionID != SessionId || avSetStartLocationRequestPacket.AgentData.AgentID != AgentId) return true; #endregion if (avSetStartLocationRequestPacket.AgentData.AgentID == AgentId && avSetStartLocationRequestPacket.AgentData.SessionID == SessionId) { // Linden Client limitation.. if (avSetStartLocationRequestPacket.StartLocationData.LocationPos.X == 255.5f || avSetStartLocationRequestPacket.StartLocationData.LocationPos.Y == 255.5f) { ScenePresence avatar = null; if (((Scene)m_scene).TryGetScenePresence(AgentId, out avatar)) { if (avSetStartLocationRequestPacket.StartLocationData.LocationPos.X == 255.5f) { avSetStartLocationRequestPacket.StartLocationData.LocationPos.X = avatar.AbsolutePosition.X; } if (avSetStartLocationRequestPacket.StartLocationData.LocationPos.Y == 255.5f) { avSetStartLocationRequestPacket.StartLocationData.LocationPos.Y = avatar.AbsolutePosition.Y; } } } TeleportLocationRequest handlerSetStartLocationRequest = OnSetStartLocationRequest; if (handlerSetStartLocationRequest != null) { handlerSetStartLocationRequest(this, 0, avSetStartLocationRequestPacket.StartLocationData.LocationPos, avSetStartLocationRequestPacket.StartLocationData.LocationLookAt, avSetStartLocationRequestPacket.StartLocationData.LocationID); } } return true; } private bool HandleAgentThrottle(IClientAPI sender, Packet Pack) { AgentThrottlePacket atpack = (AgentThrottlePacket)Pack; #region Packet Session and User Check if (atpack.AgentData.SessionID != SessionId || atpack.AgentData.AgentID != AgentId) return true; #endregion m_udpClient.SetThrottles(atpack.Throttle.Throttles); GenericCall2 handler = OnUpdateThrottles; if (handler != null) { handler(); } return true; } private bool HandleAgentPause(IClientAPI sender, Packet Pack) { m_udpClient.IsPaused = true; return true; } private bool HandleAgentResume(IClientAPI sender, Packet Pack) { m_udpClient.IsPaused = false; m_udpServer.SendPing(m_udpClient); return true; } private bool HandleForceScriptControlRelease(IClientAPI sender, Packet Pack) { ForceReleaseControls handlerForceReleaseControls = OnForceReleaseControls; if (handlerForceReleaseControls != null) { handlerForceReleaseControls(this, AgentId); } return true; } #endregion Scene/Avatar #region Objects/m_sceneObjects private bool HandleObjectLink(IClientAPI sender, Packet Pack) { ObjectLinkPacket link = (ObjectLinkPacket)Pack; #region Packet Session and User Check if (link.AgentData.SessionID != SessionId || link.AgentData.AgentID != AgentId) return true; #endregion uint parentprimid = 0; List childrenprims = new List(); if (link.ObjectData.Length > 1) { parentprimid = link.ObjectData[0].ObjectLocalID; for (int i = 1; i < link.ObjectData.Length; i++) { childrenprims.Add(link.ObjectData[i].ObjectLocalID); } } LinkObjects handlerLinkObjects = OnLinkObjects; if (handlerLinkObjects != null) { handlerLinkObjects(this, parentprimid, childrenprims); } return true; } private bool HandleObjectDelink(IClientAPI sender, Packet Pack) { ObjectDelinkPacket delink = (ObjectDelinkPacket)Pack; #region Packet Session and User Check if (delink.AgentData.SessionID != SessionId || delink.AgentData.AgentID != AgentId) return true; #endregion // It appears the prim at index 0 is not always the root prim (for // instance, when one prim of a link set has been edited independently // of the others). Therefore, we'll pass all the ids onto the delink // method for it to decide which is the root. List prims = new List(); for (int i = 0; i < delink.ObjectData.Length; i++) { prims.Add(delink.ObjectData[i].ObjectLocalID); } DelinkObjects handlerDelinkObjects = OnDelinkObjects; if (handlerDelinkObjects != null) { handlerDelinkObjects(prims, this); } return true; } private bool HandleObjectAdd(IClientAPI sender, Packet Pack) { if (OnAddPrim != null) { ObjectAddPacket addPacket = (ObjectAddPacket)Pack; #region Packet Session and User Check if (addPacket.AgentData.SessionID != SessionId || addPacket.AgentData.AgentID != AgentId) return true; #endregion PrimitiveBaseShape shape = GetShapeFromAddPacket(addPacket); // m_log.Info("[REZData]: " + addPacket.ToString()); //BypassRaycast: 1 //RayStart: <69.79469, 158.2652, 98.40343> //RayEnd: <61.97724, 141.995, 92.58341> //RayTargetID: 00000000-0000-0000-0000-000000000000 //Check to see if adding the prim is allowed; useful for any module wanting to restrict the //object from rezing initially AddNewPrim handlerAddPrim = OnAddPrim; if (handlerAddPrim != null) handlerAddPrim(AgentId, addPacket.AgentData.GroupID, addPacket.ObjectData.RayEnd, addPacket.ObjectData.Rotation, shape, addPacket.ObjectData.BypassRaycast, addPacket.ObjectData.RayStart, addPacket.ObjectData.RayTargetID, addPacket.ObjectData.RayEndIsIntersection); } return true; } private bool HandleObjectShape(IClientAPI sender, Packet Pack) { ObjectShapePacket shapePacket = (ObjectShapePacket)Pack; #region Packet Session and User Check if (shapePacket.AgentData.SessionID != SessionId || shapePacket.AgentData.AgentID != AgentId) return true; #endregion UpdateShape handlerUpdatePrimShape = null; for (int i = 0; i < shapePacket.ObjectData.Length; i++) { handlerUpdatePrimShape = OnUpdatePrimShape; if (handlerUpdatePrimShape != null) { UpdateShapeArgs shapeData = new UpdateShapeArgs(); shapeData.ObjectLocalID = shapePacket.ObjectData[i].ObjectLocalID; shapeData.PathBegin = shapePacket.ObjectData[i].PathBegin; shapeData.PathCurve = shapePacket.ObjectData[i].PathCurve; shapeData.PathEnd = shapePacket.ObjectData[i].PathEnd; shapeData.PathRadiusOffset = shapePacket.ObjectData[i].PathRadiusOffset; shapeData.PathRevolutions = shapePacket.ObjectData[i].PathRevolutions; shapeData.PathScaleX = shapePacket.ObjectData[i].PathScaleX; shapeData.PathScaleY = shapePacket.ObjectData[i].PathScaleY; shapeData.PathShearX = shapePacket.ObjectData[i].PathShearX; shapeData.PathShearY = shapePacket.ObjectData[i].PathShearY; shapeData.PathSkew = shapePacket.ObjectData[i].PathSkew; shapeData.PathTaperX = shapePacket.ObjectData[i].PathTaperX; shapeData.PathTaperY = shapePacket.ObjectData[i].PathTaperY; shapeData.PathTwist = shapePacket.ObjectData[i].PathTwist; shapeData.PathTwistBegin = shapePacket.ObjectData[i].PathTwistBegin; shapeData.ProfileBegin = shapePacket.ObjectData[i].ProfileBegin; shapeData.ProfileCurve = shapePacket.ObjectData[i].ProfileCurve; shapeData.ProfileEnd = shapePacket.ObjectData[i].ProfileEnd; shapeData.ProfileHollow = shapePacket.ObjectData[i].ProfileHollow; handlerUpdatePrimShape(m_agentId, shapePacket.ObjectData[i].ObjectLocalID, shapeData); } } return true; } private bool HandleObjectExtraParams(IClientAPI sender, Packet Pack) { ObjectExtraParamsPacket extraPar = (ObjectExtraParamsPacket)Pack; #region Packet Session and User Check if (extraPar.AgentData.SessionID != SessionId || extraPar.AgentData.AgentID != AgentId) return true; #endregion ObjectExtraParams handlerUpdateExtraParams = OnUpdateExtraParams; if (handlerUpdateExtraParams != null) { for (int i = 0; i < extraPar.ObjectData.Length; i++) { handlerUpdateExtraParams(m_agentId, extraPar.ObjectData[i].ObjectLocalID, extraPar.ObjectData[i].ParamType, extraPar.ObjectData[i].ParamInUse, extraPar.ObjectData[i].ParamData); } } return true; } private bool HandleObjectDuplicate(IClientAPI sender, Packet Pack) { ObjectDuplicatePacket dupe = (ObjectDuplicatePacket)Pack; #region Packet Session and User Check if (dupe.AgentData.SessionID != SessionId || dupe.AgentData.AgentID != AgentId) return true; #endregion // ObjectDuplicatePacket.AgentDataBlock AgentandGroupData = dupe.AgentData; ObjectDuplicate handlerObjectDuplicate = null; handlerObjectDuplicate = OnObjectDuplicate; if (handlerObjectDuplicate != null) { for (int i = 0; i < dupe.ObjectData.Length; i++) { UUID rezGroupID = dupe.AgentData.GroupID; if(!IsGroupMember(rezGroupID)) rezGroupID = UUID.Zero; handlerObjectDuplicate(dupe.ObjectData[i].ObjectLocalID, dupe.SharedData.Offset, dupe.SharedData.DuplicateFlags, AgentId, rezGroupID); } } return true; } private bool HandleRequestMultipleObjects(IClientAPI sender, Packet Pack) { RequestMultipleObjectsPacket incomingRequest = (RequestMultipleObjectsPacket)Pack; #region Packet Session and User Check if (incomingRequest.AgentData.SessionID != SessionId || incomingRequest.AgentData.AgentID != AgentId) return true; #endregion ObjectRequest handlerObjectRequest = null; for (int i = 0; i < incomingRequest.ObjectData.Length; i++) { handlerObjectRequest = OnObjectRequest; if (handlerObjectRequest != null) { handlerObjectRequest(incomingRequest.ObjectData[i].ID, this); } } return true; } private bool HandleObjectSelect(IClientAPI sender, Packet Pack) { ObjectSelectPacket incomingselect = (ObjectSelectPacket)Pack; #region Packet Session and User Check if (incomingselect.AgentData.SessionID != SessionId || incomingselect.AgentData.AgentID != AgentId) return true; #endregion List thisSelection = new List(); ObjectSelect handlerObjectSelect = null; uint objID; handlerObjectSelect = OnObjectSelect; if (handlerObjectSelect != null) { for (int i = 0; i < incomingselect.ObjectData.Length; i++) { objID = incomingselect.ObjectData[i].ObjectLocalID; thisSelection.Add(objID); } handlerObjectSelect(thisSelection, this); } return true; } private bool HandleObjectDeselect(IClientAPI sender, Packet Pack) { ObjectDeselectPacket incomingdeselect = (ObjectDeselectPacket)Pack; #region Packet Session and User Check if (incomingdeselect.AgentData.SessionID != SessionId || incomingdeselect.AgentData.AgentID != AgentId) return true; #endregion ObjectDeselect handlerObjectDeselect = null; uint objID; for (int i = 0; i < incomingdeselect.ObjectData.Length; i++) { objID = incomingdeselect.ObjectData[i].ObjectLocalID; handlerObjectDeselect = OnObjectDeselect; if (handlerObjectDeselect != null) { OnObjectDeselect(objID, this); } } return true; } private bool HandleObjectPosition(IClientAPI sender, Packet Pack) { // DEPRECATED: but till libsecondlife removes it, people will use it ObjectPositionPacket position = (ObjectPositionPacket)Pack; #region Packet Session and User Check if (position.AgentData.SessionID != SessionId || position.AgentData.AgentID != AgentId) return true; #endregion for (int i = 0; i < position.ObjectData.Length; i++) { UpdateVector handlerUpdateVector = OnUpdatePrimGroupPosition; if (handlerUpdateVector != null) handlerUpdateVector(position.ObjectData[i].ObjectLocalID, position.ObjectData[i].Position, this); } return true; } private bool HandleObjectScale(IClientAPI sender, Packet Pack) { // DEPRECATED: but till libsecondlife removes it, people will use it ObjectScalePacket scale = (ObjectScalePacket)Pack; #region Packet Session and User Check if (scale.AgentData.SessionID != SessionId || scale.AgentData.AgentID != AgentId) return true; #endregion for (int i = 0; i < scale.ObjectData.Length; i++) { UpdateVector handlerUpdatePrimGroupScale = OnUpdatePrimGroupScale; if (handlerUpdatePrimGroupScale != null) handlerUpdatePrimGroupScale(scale.ObjectData[i].ObjectLocalID, scale.ObjectData[i].Scale, this); } return true; } private bool HandleObjectRotation(IClientAPI sender, Packet Pack) { // DEPRECATED: but till libsecondlife removes it, people will use it ObjectRotationPacket rotation = (ObjectRotationPacket)Pack; #region Packet Session and User Check if (rotation.AgentData.SessionID != SessionId || rotation.AgentData.AgentID != AgentId) return true; #endregion for (int i = 0; i < rotation.ObjectData.Length; i++) { UpdatePrimRotation handlerUpdatePrimRotation = OnUpdatePrimGroupRotation; if (handlerUpdatePrimRotation != null) handlerUpdatePrimRotation(rotation.ObjectData[i].ObjectLocalID, rotation.ObjectData[i].Rotation, this); } return true; } private bool HandleObjectFlagUpdate(IClientAPI sender, Packet Pack) { ObjectFlagUpdatePacket flags = (ObjectFlagUpdatePacket)Pack; #region Packet Session and User Check if (flags.AgentData.SessionID != SessionId || flags.AgentData.AgentID != AgentId) return true; #endregion UpdatePrimFlags handlerUpdatePrimFlags = OnUpdatePrimFlags; if (handlerUpdatePrimFlags != null) { // byte[] data = Pack.ToBytes(); // 46,47,48 are special positions within the packet // This may change so perhaps we need a better way // of storing this (OMV.FlagUpdatePacket.UsePhysics,etc?) /* bool UsePhysics = (data[46] != 0) ? true : false; bool IsTemporary = (data[47] != 0) ? true : false; bool IsPhantom = (data[48] != 0) ? true : false; handlerUpdatePrimFlags(flags.AgentData.ObjectLocalID, UsePhysics, IsTemporary, IsPhantom, this); */ bool UsePhysics = flags.AgentData.UsePhysics; bool IsPhantom = flags.AgentData.IsPhantom; bool IsTemporary = flags.AgentData.IsTemporary; ObjectFlagUpdatePacket.ExtraPhysicsBlock[] blocks = flags.ExtraPhysics; ExtraPhysicsData physdata = new ExtraPhysicsData(); if (blocks == null || blocks.Length == 0) { physdata.PhysShapeType = PhysShapeType.invalid; } else { ObjectFlagUpdatePacket.ExtraPhysicsBlock phsblock = blocks[0]; physdata.PhysShapeType = (PhysShapeType)phsblock.PhysicsShapeType; physdata.Bounce = phsblock.Restitution; physdata.Density = phsblock.Density; physdata.Friction = phsblock.Friction; physdata.GravitationModifier = phsblock.GravityMultiplier; } handlerUpdatePrimFlags(flags.AgentData.ObjectLocalID, UsePhysics, IsTemporary, IsPhantom, physdata, this); } return true; } Dictionary objImageSeqs = null; double lastobjImageSeqsMS = 0.0; private bool HandleObjectImage(IClientAPI sender, Packet Pack) { ObjectImagePacket imagePack = (ObjectImagePacket)Pack; UpdatePrimTexture handlerUpdatePrimTexture = OnUpdatePrimTexture; if (handlerUpdatePrimTexture == null) return true; double now = Util.GetTimeStampMS(); if(objImageSeqs == null || ( now - lastobjImageSeqsMS > 30000.0)) { objImageSeqs = null; // yeah i know superstition... objImageSeqs = new Dictionary(16); } lastobjImageSeqsMS = now; uint seq = Pack.Header.Sequence; uint id; uint lastseq; ObjectImagePacket.ObjectDataBlock o; for (int i = 0; i < imagePack.ObjectData.Length; i++) { o = imagePack.ObjectData[i]; id = o.ObjectLocalID; if(objImageSeqs.TryGetValue(id, out lastseq)) { if(seq <= lastseq) continue; } objImageSeqs[id] = seq; handlerUpdatePrimTexture(id, o.TextureEntry, this); } return true; } private bool HandleObjectGrab(IClientAPI sender, Packet Pack) { ObjectGrabPacket grab = (ObjectGrabPacket)Pack; #region Packet Session and User Check if (grab.AgentData.SessionID != SessionId || grab.AgentData.AgentID != AgentId) return true; #endregion GrabObject handlerGrabObject = OnGrabObject; if (handlerGrabObject != null) { List touchArgs = new List(); if ((grab.SurfaceInfo != null) && (grab.SurfaceInfo.Length > 0)) { foreach (ObjectGrabPacket.SurfaceInfoBlock surfaceInfo in grab.SurfaceInfo) { SurfaceTouchEventArgs arg = new SurfaceTouchEventArgs(); arg.Binormal = surfaceInfo.Binormal; arg.FaceIndex = surfaceInfo.FaceIndex; arg.Normal = surfaceInfo.Normal; arg.Position = surfaceInfo.Position; arg.STCoord = surfaceInfo.STCoord; arg.UVCoord = surfaceInfo.UVCoord; touchArgs.Add(arg); } } handlerGrabObject(grab.ObjectData.LocalID, grab.ObjectData.GrabOffset, this, touchArgs); } return true; } private bool HandleObjectGrabUpdate(IClientAPI sender, Packet Pack) { ObjectGrabUpdatePacket grabUpdate = (ObjectGrabUpdatePacket)Pack; #region Packet Session and User Check if (grabUpdate.AgentData.SessionID != SessionId || grabUpdate.AgentData.AgentID != AgentId) return true; #endregion MoveObject handlerGrabUpdate = OnGrabUpdate; if (handlerGrabUpdate != null) { List touchArgs = new List(); if ((grabUpdate.SurfaceInfo != null) && (grabUpdate.SurfaceInfo.Length > 0)) { foreach (ObjectGrabUpdatePacket.SurfaceInfoBlock surfaceInfo in grabUpdate.SurfaceInfo) { SurfaceTouchEventArgs arg = new SurfaceTouchEventArgs(); arg.Binormal = surfaceInfo.Binormal; arg.FaceIndex = surfaceInfo.FaceIndex; arg.Normal = surfaceInfo.Normal; arg.Position = surfaceInfo.Position; arg.STCoord = surfaceInfo.STCoord; arg.UVCoord = surfaceInfo.UVCoord; touchArgs.Add(arg); } } handlerGrabUpdate(grabUpdate.ObjectData.ObjectID, grabUpdate.ObjectData.GrabOffsetInitial, grabUpdate.ObjectData.GrabPosition, this, touchArgs); } return true; } private bool HandleObjectDeGrab(IClientAPI sender, Packet Pack) { ObjectDeGrabPacket deGrab = (ObjectDeGrabPacket)Pack; #region Packet Session and User Check if (deGrab.AgentData.SessionID != SessionId || deGrab.AgentData.AgentID != AgentId) return true; #endregion DeGrabObject handlerDeGrabObject = OnDeGrabObject; if (handlerDeGrabObject != null) { List touchArgs = new List(); if ((deGrab.SurfaceInfo != null) && (deGrab.SurfaceInfo.Length > 0)) { foreach (ObjectDeGrabPacket.SurfaceInfoBlock surfaceInfo in deGrab.SurfaceInfo) { SurfaceTouchEventArgs arg = new SurfaceTouchEventArgs(); arg.Binormal = surfaceInfo.Binormal; arg.FaceIndex = surfaceInfo.FaceIndex; arg.Normal = surfaceInfo.Normal; arg.Position = surfaceInfo.Position; arg.STCoord = surfaceInfo.STCoord; arg.UVCoord = surfaceInfo.UVCoord; touchArgs.Add(arg); } } handlerDeGrabObject(deGrab.ObjectData.LocalID, this, touchArgs); } return true; } private bool HandleObjectSpinStart(IClientAPI sender, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinStart packet"); ObjectSpinStartPacket spinStart = (ObjectSpinStartPacket)Pack; #region Packet Session and User Check if (spinStart.AgentData.SessionID != SessionId || spinStart.AgentData.AgentID != AgentId) return true; #endregion SpinStart handlerSpinStart = OnSpinStart; if (handlerSpinStart != null) { handlerSpinStart(spinStart.ObjectData.ObjectID, this); } return true; } private bool HandleObjectSpinUpdate(IClientAPI sender, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinUpdate packet"); ObjectSpinUpdatePacket spinUpdate = (ObjectSpinUpdatePacket)Pack; #region Packet Session and User Check if (spinUpdate.AgentData.SessionID != SessionId || spinUpdate.AgentData.AgentID != AgentId) return true; #endregion Vector3 axis; float angle; spinUpdate.ObjectData.Rotation.GetAxisAngle(out axis, out angle); //m_log.Warn("[CLIENT]: ObjectSpinUpdate packet rot axis:" + axis + " angle:" + angle); SpinObject handlerSpinUpdate = OnSpinUpdate; if (handlerSpinUpdate != null) { handlerSpinUpdate(spinUpdate.ObjectData.ObjectID, spinUpdate.ObjectData.Rotation, this); } return true; } private bool HandleObjectSpinStop(IClientAPI sender, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinStop packet"); ObjectSpinStopPacket spinStop = (ObjectSpinStopPacket)Pack; #region Packet Session and User Check if (spinStop.AgentData.SessionID != SessionId || spinStop.AgentData.AgentID != AgentId) return true; #endregion SpinStop handlerSpinStop = OnSpinStop; if (handlerSpinStop != null) { handlerSpinStop(spinStop.ObjectData.ObjectID, this); } return true; } private bool HandleObjectDescription(IClientAPI sender, Packet Pack) { ObjectDescriptionPacket objDes = (ObjectDescriptionPacket)Pack; #region Packet Session and User Check if (objDes.AgentData.SessionID != SessionId || objDes.AgentData.AgentID != AgentId) return true; #endregion GenericCall7 handlerObjectDescription = null; for (int i = 0; i < objDes.ObjectData.Length; i++) { handlerObjectDescription = OnObjectDescription; if (handlerObjectDescription != null) { handlerObjectDescription(this, objDes.ObjectData[i].LocalID, Util.FieldToString(objDes.ObjectData[i].Description)); } } return true; } private bool HandleObjectName(IClientAPI sender, Packet Pack) { ObjectNamePacket objName = (ObjectNamePacket)Pack; #region Packet Session and User Check if (objName.AgentData.SessionID != SessionId || objName.AgentData.AgentID != AgentId) return true; #endregion GenericCall7 handlerObjectName = null; for (int i = 0; i < objName.ObjectData.Length; i++) { handlerObjectName = OnObjectName; if (handlerObjectName != null) { handlerObjectName(this, objName.ObjectData[i].LocalID, Util.FieldToString(objName.ObjectData[i].Name)); } } return true; } private bool HandleObjectPermissions(IClientAPI sender, Packet Pack) { if (OnObjectPermissions != null) { ObjectPermissionsPacket newobjPerms = (ObjectPermissionsPacket)Pack; #region Packet Session and User Check if (newobjPerms.AgentData.SessionID != SessionId || newobjPerms.AgentData.AgentID != AgentId) return true; #endregion UUID AgentID = newobjPerms.AgentData.AgentID; UUID SessionID = newobjPerms.AgentData.SessionID; ObjectPermissions handlerObjectPermissions = null; for (int i = 0; i < newobjPerms.ObjectData.Length; i++) { ObjectPermissionsPacket.ObjectDataBlock permChanges = newobjPerms.ObjectData[i]; byte field = permChanges.Field; uint localID = permChanges.ObjectLocalID; uint mask = permChanges.Mask; byte set = permChanges.Set; handlerObjectPermissions = OnObjectPermissions; if (handlerObjectPermissions != null) handlerObjectPermissions(this, AgentID, SessionID, field, localID, mask, set); } } // Here's our data, // PermField contains the field the info goes into // PermField determines which mask we're changing // // chmask is the mask of the change // setTF is whether we're adding it or taking it away // // objLocalID is the localID of the object. // Unfortunately, we have to pass the event the packet because objData is an array // That means multiple object perms may be updated in a single packet. return true; } private bool HandleUndo(IClientAPI sender, Packet Pack) { UndoPacket undoitem = (UndoPacket)Pack; #region Packet Session and User Check if (undoitem.AgentData.SessionID != SessionId || undoitem.AgentData.AgentID != AgentId) return true; #endregion if (undoitem.ObjectData.Length > 0) { for (int i = 0; i < undoitem.ObjectData.Length; i++) { UUID objiD = undoitem.ObjectData[i].ObjectID; AgentSit handlerOnUndo = OnUndo; if (handlerOnUndo != null) { handlerOnUndo(this, objiD); } } } return true; } private bool HandleLandUndo(IClientAPI sender, Packet Pack) { UndoLandPacket undolanditem = (UndoLandPacket)Pack; #region Packet Session and User Check if (undolanditem.AgentData.SessionID != SessionId || undolanditem.AgentData.AgentID != AgentId) return true; #endregion LandUndo handlerOnUndo = OnLandUndo; if (handlerOnUndo != null) { handlerOnUndo(this); } return true; } private bool HandleRedo(IClientAPI sender, Packet Pack) { RedoPacket redoitem = (RedoPacket)Pack; #region Packet Session and User Check if (redoitem.AgentData.SessionID != SessionId || redoitem.AgentData.AgentID != AgentId) return true; #endregion if (redoitem.ObjectData.Length > 0) { for (int i = 0; i < redoitem.ObjectData.Length; i++) { UUID objiD = redoitem.ObjectData[i].ObjectID; AgentSit handlerOnRedo = OnRedo; if (handlerOnRedo != null) { handlerOnRedo(this, objiD); } } } return true; } private bool HandleObjectDuplicateOnRay(IClientAPI sender, Packet Pack) { ObjectDuplicateOnRayPacket dupeOnRay = (ObjectDuplicateOnRayPacket)Pack; #region Packet Session and User Check if (dupeOnRay.AgentData.SessionID != SessionId || dupeOnRay.AgentData.AgentID != AgentId) return true; #endregion ObjectDuplicateOnRay handlerObjectDuplicateOnRay = null; for (int i = 0; i < dupeOnRay.ObjectData.Length; i++) { handlerObjectDuplicateOnRay = OnObjectDuplicateOnRay; if (handlerObjectDuplicateOnRay != null) { UUID rezGroupID = dupeOnRay.AgentData.GroupID; if(!IsGroupMember(rezGroupID)) rezGroupID = UUID.Zero; handlerObjectDuplicateOnRay(dupeOnRay.ObjectData[i].ObjectLocalID, dupeOnRay.AgentData.DuplicateFlags, AgentId, rezGroupID, dupeOnRay.AgentData.RayTargetID, dupeOnRay.AgentData.RayEnd, dupeOnRay.AgentData.RayStart, dupeOnRay.AgentData.BypassRaycast, dupeOnRay.AgentData.RayEndIsIntersection, dupeOnRay.AgentData.CopyCenters, dupeOnRay.AgentData.CopyRotates); } } return true; } private bool HandleRequestObjectPropertiesFamily(IClientAPI sender, Packet Pack) { //This powers the little tooltip that appears when you move your mouse over an object RequestObjectPropertiesFamilyPacket packToolTip = (RequestObjectPropertiesFamilyPacket)Pack; #region Packet Session and User Check if (packToolTip.AgentData.SessionID != SessionId || packToolTip.AgentData.AgentID != AgentId) return true; #endregion RequestObjectPropertiesFamilyPacket.ObjectDataBlock packObjBlock = packToolTip.ObjectData; RequestObjectPropertiesFamily handlerRequestObjectPropertiesFamily = OnRequestObjectPropertiesFamily; if (handlerRequestObjectPropertiesFamily != null) { handlerRequestObjectPropertiesFamily(this, m_agentId, packObjBlock.RequestFlags, packObjBlock.ObjectID); } return true; } private bool HandleObjectIncludeInSearch(IClientAPI sender, Packet Pack) { //This lets us set objects to appear in search (stuff like DataSnapshot, etc) ObjectIncludeInSearchPacket packInSearch = (ObjectIncludeInSearchPacket)Pack; ObjectIncludeInSearch handlerObjectIncludeInSearch = null; #region Packet Session and User Check if (packInSearch.AgentData.SessionID != SessionId || packInSearch.AgentData.AgentID != AgentId) return true; #endregion foreach (ObjectIncludeInSearchPacket.ObjectDataBlock objData in packInSearch.ObjectData) { bool inSearch = objData.IncludeInSearch; uint localID = objData.ObjectLocalID; handlerObjectIncludeInSearch = OnObjectIncludeInSearch; if (handlerObjectIncludeInSearch != null) { handlerObjectIncludeInSearch(this, inSearch, localID); } } return true; } private bool HandleScriptAnswerYes(IClientAPI sender, Packet Pack) { ScriptAnswerYesPacket scriptAnswer = (ScriptAnswerYesPacket)Pack; #region Packet Session and User Check if (scriptAnswer.AgentData.SessionID != SessionId || scriptAnswer.AgentData.AgentID != AgentId) return true; #endregion ScriptAnswer handlerScriptAnswer = OnScriptAnswer; if (handlerScriptAnswer != null) { handlerScriptAnswer(this, scriptAnswer.Data.TaskID, scriptAnswer.Data.ItemID, scriptAnswer.Data.Questions); } return true; } private bool HandleObjectClickAction(IClientAPI sender, Packet Pack) { ObjectClickActionPacket ocpacket = (ObjectClickActionPacket)Pack; #region Packet Session and User Check if (ocpacket.AgentData.SessionID != SessionId || ocpacket.AgentData.AgentID != AgentId) return true; #endregion GenericCall7 handlerObjectClickAction = OnObjectClickAction; if (handlerObjectClickAction != null) { foreach (ObjectClickActionPacket.ObjectDataBlock odata in ocpacket.ObjectData) { byte action = odata.ClickAction; uint localID = odata.ObjectLocalID; handlerObjectClickAction(this, localID, action.ToString()); } } return true; } private bool HandleObjectMaterial(IClientAPI sender, Packet Pack) { ObjectMaterialPacket ompacket = (ObjectMaterialPacket)Pack; #region Packet Session and User Check if (ompacket.AgentData.SessionID != SessionId || ompacket.AgentData.AgentID != AgentId) return true; #endregion GenericCall7 handlerObjectMaterial = OnObjectMaterial; if (handlerObjectMaterial != null) { foreach (ObjectMaterialPacket.ObjectDataBlock odata in ompacket.ObjectData) { byte material = odata.Material; uint localID = odata.ObjectLocalID; handlerObjectMaterial(this, localID, material.ToString()); } } return true; } #endregion Objects/m_sceneObjects #region Inventory/Asset/Other related packets private bool HandleRequestImage(IClientAPI sender, Packet Pack) { RequestImagePacket imageRequest = (RequestImagePacket)Pack; //m_log.Debug("image request: " + Pack.ToString()); #region Packet Session and User Check if (imageRequest.AgentData.SessionID != SessionId || imageRequest.AgentData.AgentID != AgentId) return true; #endregion //handlerTextureRequest = null; for (int i = 0; i < imageRequest.RequestImage.Length; i++) { TextureRequestArgs args = new TextureRequestArgs(); RequestImagePacket.RequestImageBlock block = imageRequest.RequestImage[i]; args.RequestedAssetID = block.Image; args.DiscardLevel = block.DiscardLevel; args.PacketNumber = block.Packet; args.Priority = block.DownloadPriority; args.requestSequence = imageRequest.Header.Sequence; // NOTE: This is not a built in part of the LLUDP protocol, but we double the // priority of avatar textures to get avatars rezzing in faster than the // surrounding scene if ((ImageType)block.Type == ImageType.Baked) args.Priority *= 2.0f; ImageManager.EnqueueReq(args); } return true; } /// /// This is the entry point for the UDP route by which the client can retrieve asset data. If the request /// is successful then a TransferInfo packet will be sent back, followed by one or more TransferPackets /// /// /// /// This parameter may be ignored since we appear to return true whatever happens private bool HandleTransferRequest(IClientAPI sender, Packet Pack) { //m_log.Debug("ClientView.ProcessPackets.cs:ProcessInPacket() - Got transfer request"); TransferRequestPacket transfer = (TransferRequestPacket)Pack; UUID taskID = UUID.Zero; if (transfer.TransferInfo.SourceType == (int)SourceType.SimInventoryItem) { if (!(((Scene)m_scene).Permissions.BypassPermissions())) { // We're spawning a thread because the permissions check can block this thread Util.FireAndForget(delegate { // This requests the asset if needed HandleSimInventoryTransferRequestWithPermsCheck(sender, transfer); }, null, "LLClientView.HandleTransferRequest"); return true; } } else if (transfer.TransferInfo.SourceType == (int)SourceType.SimEstate) { //TransferRequestPacket does not include covenant uuid? //get scene covenant uuid taskID = m_scene.RegionInfo.RegionSettings.Covenant; } // This is non-blocking MakeAssetRequest(transfer, taskID); return true; } private void HandleSimInventoryTransferRequestWithPermsCheck(IClientAPI sender, TransferRequestPacket transfer) { UUID taskID = new UUID(transfer.TransferInfo.Params, 48); UUID itemID = new UUID(transfer.TransferInfo.Params, 64); UUID requestID = new UUID(transfer.TransferInfo.Params, 80); //m_log.DebugFormat( // "[CLIENT]: Got request for asset {0} from item {1} in prim {2} by {3}", // requestID, itemID, taskID, Name); //m_log.Debug("Transfer Request: " + transfer.ToString()); // Validate inventory transfers // Has to be done here, because AssetCache can't do it // if (taskID != UUID.Zero) // Prim { SceneObjectPart part = ((Scene)m_scene).GetSceneObjectPart(taskID); if (part == null) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but prim does not exist", Name, requestID, itemID, taskID); return; } TaskInventoryItem tii = part.Inventory.GetInventoryItem(itemID); if (tii == null) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but item does not exist", Name, requestID, itemID, taskID); return; } if (tii.Type == (int)AssetType.LSLText) { if (!((Scene)m_scene).Permissions.CanEditScript(itemID, taskID, AgentId)) return; } else if (tii.Type == (int)AssetType.Notecard) { if (!((Scene)m_scene).Permissions.CanEditNotecard(itemID, taskID, AgentId)) return; } else { // TODO: Change this code to allow items other than notecards and scripts to be successfully // shared with group. In fact, this whole block of permissions checking should move to an IPermissionsModule if (part.OwnerID != AgentId) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but the prim is owned by {4}", Name, requestID, itemID, taskID, part.OwnerID); return; } if ((part.OwnerMask & (uint)PermissionMask.Modify) == 0) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but modify permissions are not set", Name, requestID, itemID, taskID); return; } if (tii.OwnerID != AgentId) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but the item is owned by {4}", Name, requestID, itemID, taskID, tii.OwnerID); return; } if (( tii.CurrentPermissions & ((uint)PermissionMask.Modify | (uint)PermissionMask.Copy | (uint)PermissionMask.Transfer)) != ((uint)PermissionMask.Modify | (uint)PermissionMask.Copy | (uint)PermissionMask.Transfer)) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but item permissions are not modify/copy/transfer", Name, requestID, itemID, taskID); return; } if (tii.AssetID != requestID) { m_log.WarnFormat( "[CLIENT]: {0} requested asset {1} from item {2} in prim {3} but this does not match item's asset {4}", Name, requestID, itemID, taskID, tii.AssetID); return; } } } else // Agent { IInventoryAccessModule invAccess = m_scene.RequestModuleInterface(); if (invAccess != null) { if (!invAccess.CanGetAgentInventoryItem(this, itemID, requestID)) return; } else { return; } } // Permissions out of the way, let's request the asset MakeAssetRequest(transfer, taskID); } private bool HandleAssetUploadRequest(IClientAPI sender, Packet Pack) { AssetUploadRequestPacket request = (AssetUploadRequestPacket)Pack; // m_log.Debug("upload request " + request.ToString()); // m_log.Debug("upload request was for assetid: " + request.AssetBlock.TransactionID.Combine(this.SecureSessionId).ToString()); UUID temp = UUID.Combine(request.AssetBlock.TransactionID, SecureSessionId); UDPAssetUploadRequest handlerAssetUploadRequest = OnAssetUploadRequest; if (handlerAssetUploadRequest != null) { handlerAssetUploadRequest(this, temp, request.AssetBlock.TransactionID, request.AssetBlock.Type, request.AssetBlock.AssetData, request.AssetBlock.StoreLocal, request.AssetBlock.Tempfile); } return true; } private bool HandleRequestXfer(IClientAPI sender, Packet Pack) { RequestXferPacket xferReq = (RequestXferPacket)Pack; RequestXfer handlerRequestXfer = OnRequestXfer; if (handlerRequestXfer != null) { handlerRequestXfer(this, xferReq.XferID.ID, Util.FieldToString(xferReq.XferID.Filename)); } return true; } private bool HandleSendXferPacket(IClientAPI sender, Packet Pack) { SendXferPacketPacket xferRec = (SendXferPacketPacket)Pack; XferReceive handlerXferReceive = OnXferReceive; if (handlerXferReceive != null) { handlerXferReceive(this, xferRec.XferID.ID, xferRec.XferID.Packet, xferRec.DataPacket.Data); } return true; } private bool HandleConfirmXferPacket(IClientAPI sender, Packet Pack) { ConfirmXferPacketPacket confirmXfer = (ConfirmXferPacketPacket)Pack; ConfirmXfer handlerConfirmXfer = OnConfirmXfer; if (handlerConfirmXfer != null) { handlerConfirmXfer(this, confirmXfer.XferID.ID, confirmXfer.XferID.Packet); } return true; } private bool HandleAbortXfer(IClientAPI sender, Packet Pack) { AbortXferPacket abortXfer = (AbortXferPacket)Pack; AbortXfer handlerAbortXfer = OnAbortXfer; if (handlerAbortXfer != null) { handlerAbortXfer(this, abortXfer.XferID.ID); } return true; } private bool HandleCreateInventoryFolder(IClientAPI sender, Packet Pack) { CreateInventoryFolderPacket invFolder = (CreateInventoryFolderPacket)Pack; #region Packet Session and User Check if (invFolder.AgentData.SessionID != SessionId || invFolder.AgentData.AgentID != AgentId) return true; #endregion CreateInventoryFolder handlerCreateInventoryFolder = OnCreateNewInventoryFolder; if (handlerCreateInventoryFolder != null) { handlerCreateInventoryFolder(this, invFolder.FolderData.FolderID, (ushort)invFolder.FolderData.Type, Util.FieldToString(invFolder.FolderData.Name), invFolder.FolderData.ParentID); } return true; } private bool HandleUpdateInventoryFolder(IClientAPI sender, Packet Pack) { if (OnUpdateInventoryFolder != null) { UpdateInventoryFolderPacket invFolderx = (UpdateInventoryFolderPacket)Pack; #region Packet Session and User Check if (invFolderx.AgentData.SessionID != SessionId || invFolderx.AgentData.AgentID != AgentId) return true; #endregion UpdateInventoryFolder handlerUpdateInventoryFolder = null; for (int i = 0; i < invFolderx.FolderData.Length; i++) { handlerUpdateInventoryFolder = OnUpdateInventoryFolder; if (handlerUpdateInventoryFolder != null) { OnUpdateInventoryFolder(this, invFolderx.FolderData[i].FolderID, (ushort)invFolderx.FolderData[i].Type, Util.FieldToString(invFolderx.FolderData[i].Name), invFolderx.FolderData[i].ParentID); } } } return true; } private bool HandleMoveInventoryFolder(IClientAPI sender, Packet Pack) { if (OnMoveInventoryFolder != null) { MoveInventoryFolderPacket invFoldery = (MoveInventoryFolderPacket)Pack; #region Packet Session and User Check if (invFoldery.AgentData.SessionID != SessionId || invFoldery.AgentData.AgentID != AgentId) return true; #endregion MoveInventoryFolder handlerMoveInventoryFolder = null; for (int i = 0; i < invFoldery.InventoryData.Length; i++) { handlerMoveInventoryFolder = OnMoveInventoryFolder; if (handlerMoveInventoryFolder != null) { OnMoveInventoryFolder(this, invFoldery.InventoryData[i].FolderID, invFoldery.InventoryData[i].ParentID); } } } return true; } private bool HandleCreateInventoryItem(IClientAPI sender, Packet Pack) { CreateInventoryItemPacket createItem = (CreateInventoryItemPacket)Pack; #region Packet Session and User Check if (createItem.AgentData.SessionID != SessionId || createItem.AgentData.AgentID != AgentId) return true; #endregion CreateNewInventoryItem handlerCreateNewInventoryItem = OnCreateNewInventoryItem; if (handlerCreateNewInventoryItem != null) { handlerCreateNewInventoryItem(this, createItem.InventoryBlock.TransactionID, createItem.InventoryBlock.FolderID, createItem.InventoryBlock.CallbackID, Util.FieldToString(createItem.InventoryBlock.Description), Util.FieldToString(createItem.InventoryBlock.Name), createItem.InventoryBlock.InvType, createItem.InventoryBlock.Type, createItem.InventoryBlock.WearableType, createItem.InventoryBlock.NextOwnerMask, Util.UnixTimeSinceEpoch()); } return true; } private bool HandleLinkInventoryItem(IClientAPI sender, Packet Pack) { LinkInventoryItemPacket createLink = (LinkInventoryItemPacket)Pack; #region Packet Session and User Check if (createLink.AgentData.SessionID != SessionId || createLink.AgentData.AgentID != AgentId) return true; #endregion LinkInventoryItem linkInventoryItem = OnLinkInventoryItem; if (linkInventoryItem != null) { linkInventoryItem( this, createLink.InventoryBlock.TransactionID, createLink.InventoryBlock.FolderID, createLink.InventoryBlock.CallbackID, Util.FieldToString(createLink.InventoryBlock.Description), Util.FieldToString(createLink.InventoryBlock.Name), createLink.InventoryBlock.InvType, createLink.InventoryBlock.Type, createLink.InventoryBlock.OldItemID); } return true; } private bool HandleFetchInventory(IClientAPI sender, Packet Pack) { if (OnFetchInventory != null) { FetchInventoryPacket FetchInventoryx = (FetchInventoryPacket)Pack; #region Packet Session and User Check if (FetchInventoryx.AgentData.SessionID != SessionId || FetchInventoryx.AgentData.AgentID != AgentId) return true; #endregion FetchInventory handlerFetchInventory = null; for (int i = 0; i < FetchInventoryx.InventoryData.Length; i++) { handlerFetchInventory = OnFetchInventory; if (handlerFetchInventory != null) { OnFetchInventory(this, FetchInventoryx.InventoryData[i].ItemID, FetchInventoryx.InventoryData[i].OwnerID); } } } return true; } private bool HandleFetchInventoryDescendents(IClientAPI sender, Packet Pack) { FetchInventoryDescendentsPacket Fetch = (FetchInventoryDescendentsPacket)Pack; #region Packet Session and User Check if (Fetch.AgentData.SessionID != SessionId || Fetch.AgentData.AgentID != AgentId) return true; #endregion FetchInventoryDescendents handlerFetchInventoryDescendents = OnFetchInventoryDescendents; if (handlerFetchInventoryDescendents != null) { handlerFetchInventoryDescendents(this, Fetch.InventoryData.FolderID, Fetch.InventoryData.OwnerID, Fetch.InventoryData.FetchFolders, Fetch.InventoryData.FetchItems, Fetch.InventoryData.SortOrder); } return true; } private bool HandlePurgeInventoryDescendents(IClientAPI sender, Packet Pack) { PurgeInventoryDescendentsPacket Purge = (PurgeInventoryDescendentsPacket)Pack; #region Packet Session and User Check if (Purge.AgentData.SessionID != SessionId || Purge.AgentData.AgentID != AgentId) return true; #endregion PurgeInventoryDescendents handlerPurgeInventoryDescendents = OnPurgeInventoryDescendents; if (handlerPurgeInventoryDescendents != null) { handlerPurgeInventoryDescendents(this, Purge.InventoryData.FolderID); } return true; } private bool HandleUpdateInventoryItem(IClientAPI sender, Packet Pack) { UpdateInventoryItemPacket inventoryItemUpdate = (UpdateInventoryItemPacket)Pack; #region Packet Session and User Check if (inventoryItemUpdate.AgentData.SessionID != SessionId || inventoryItemUpdate.AgentData.AgentID != AgentId) return true; #endregion if (OnUpdateInventoryItem != null) { UpdateInventoryItem handlerUpdateInventoryItem = null; for (int i = 0; i < inventoryItemUpdate.InventoryData.Length; i++) { handlerUpdateInventoryItem = OnUpdateInventoryItem; if (handlerUpdateInventoryItem != null) { InventoryItemBase itemUpd = new InventoryItemBase(); itemUpd.ID = inventoryItemUpdate.InventoryData[i].ItemID; itemUpd.Name = Util.FieldToString(inventoryItemUpdate.InventoryData[i].Name); itemUpd.Description = Util.FieldToString(inventoryItemUpdate.InventoryData[i].Description); itemUpd.GroupID = inventoryItemUpdate.InventoryData[i].GroupID; itemUpd.GroupOwned = inventoryItemUpdate.InventoryData[i].GroupOwned; itemUpd.GroupPermissions = inventoryItemUpdate.InventoryData[i].GroupMask; itemUpd.NextPermissions = inventoryItemUpdate.InventoryData[i].NextOwnerMask; itemUpd.EveryOnePermissions = inventoryItemUpdate.InventoryData[i].EveryoneMask; itemUpd.CreationDate = inventoryItemUpdate.InventoryData[i].CreationDate; itemUpd.Folder = inventoryItemUpdate.InventoryData[i].FolderID; itemUpd.InvType = inventoryItemUpdate.InventoryData[i].InvType; itemUpd.SalePrice = inventoryItemUpdate.InventoryData[i].SalePrice; itemUpd.SaleType = inventoryItemUpdate.InventoryData[i].SaleType; itemUpd.Flags = inventoryItemUpdate.InventoryData[i].Flags; OnUpdateInventoryItem(this, inventoryItemUpdate.InventoryData[i].TransactionID, inventoryItemUpdate.InventoryData[i].ItemID, itemUpd); } } } return true; } private bool HandleCopyInventoryItem(IClientAPI sender, Packet Pack) { CopyInventoryItemPacket copyitem = (CopyInventoryItemPacket)Pack; #region Packet Session and User Check if (copyitem.AgentData.SessionID != SessionId || copyitem.AgentData.AgentID != AgentId) return true; #endregion CopyInventoryItem handlerCopyInventoryItem = null; if (OnCopyInventoryItem != null) { foreach (CopyInventoryItemPacket.InventoryDataBlock datablock in copyitem.InventoryData) { handlerCopyInventoryItem = OnCopyInventoryItem; if (handlerCopyInventoryItem != null) { handlerCopyInventoryItem(this, datablock.CallbackID, datablock.OldAgentID, datablock.OldItemID, datablock.NewFolderID, Util.FieldToString(datablock.NewName)); } } } return true; } private bool HandleMoveInventoryItem(IClientAPI sender, Packet Pack) { MoveInventoryItemPacket moveitem = (MoveInventoryItemPacket)Pack; #region Packet Session and User Check if (moveitem.AgentData.SessionID != SessionId || moveitem.AgentData.AgentID != AgentId) return true; #endregion if (OnMoveInventoryItem != null) { MoveInventoryItem handlerMoveInventoryItem = null; InventoryItemBase itm = null; List items = new List(); foreach (MoveInventoryItemPacket.InventoryDataBlock datablock in moveitem.InventoryData) { itm = new InventoryItemBase(datablock.ItemID, AgentId); itm.Folder = datablock.FolderID; itm.Name = Util.FieldToString(datablock.NewName); // weird, comes out as empty string //m_log.DebugFormat("[XXX] new name: {0}", itm.Name); items.Add(itm); } handlerMoveInventoryItem = OnMoveInventoryItem; if (handlerMoveInventoryItem != null) { handlerMoveInventoryItem(this, items); } } return true; } private bool HandleRemoveInventoryItem(IClientAPI sender, Packet Pack) { RemoveInventoryItemPacket removeItem = (RemoveInventoryItemPacket)Pack; #region Packet Session and User Check if (removeItem.AgentData.SessionID != SessionId || removeItem.AgentData.AgentID != AgentId) return true; #endregion if (OnRemoveInventoryItem != null) { RemoveInventoryItem handlerRemoveInventoryItem = null; List uuids = new List(); foreach (RemoveInventoryItemPacket.InventoryDataBlock datablock in removeItem.InventoryData) { uuids.Add(datablock.ItemID); } handlerRemoveInventoryItem = OnRemoveInventoryItem; if (handlerRemoveInventoryItem != null) { handlerRemoveInventoryItem(this, uuids); } } return true; } private bool HandleRemoveInventoryFolder(IClientAPI sender, Packet Pack) { RemoveInventoryFolderPacket removeFolder = (RemoveInventoryFolderPacket)Pack; #region Packet Session and User Check if (removeFolder.AgentData.SessionID != SessionId || removeFolder.AgentData.AgentID != AgentId) return true; #endregion if (OnRemoveInventoryFolder != null) { RemoveInventoryFolder handlerRemoveInventoryFolder = null; List uuids = new List(); foreach (RemoveInventoryFolderPacket.FolderDataBlock datablock in removeFolder.FolderData) { uuids.Add(datablock.FolderID); } handlerRemoveInventoryFolder = OnRemoveInventoryFolder; if (handlerRemoveInventoryFolder != null) { handlerRemoveInventoryFolder(this, uuids); } } return true; } private bool HandleRemoveInventoryObjects(IClientAPI sender, Packet Pack) { RemoveInventoryObjectsPacket removeObject = (RemoveInventoryObjectsPacket)Pack; #region Packet Session and User Check if (removeObject.AgentData.SessionID != SessionId || removeObject.AgentData.AgentID != AgentId) return true; #endregion if (OnRemoveInventoryFolder != null) { RemoveInventoryFolder handlerRemoveInventoryFolder = null; List uuids = new List(); foreach (RemoveInventoryObjectsPacket.FolderDataBlock datablock in removeObject.FolderData) { uuids.Add(datablock.FolderID); } handlerRemoveInventoryFolder = OnRemoveInventoryFolder; if (handlerRemoveInventoryFolder != null) { handlerRemoveInventoryFolder(this, uuids); } } if (OnRemoveInventoryItem != null) { RemoveInventoryItem handlerRemoveInventoryItem = null; List uuids = new List(); foreach (RemoveInventoryObjectsPacket.ItemDataBlock datablock in removeObject.ItemData) { uuids.Add(datablock.ItemID); } handlerRemoveInventoryItem = OnRemoveInventoryItem; if (handlerRemoveInventoryItem != null) { handlerRemoveInventoryItem(this, uuids); } } return true; } private bool HandleRequestTaskInventory(IClientAPI sender, Packet Pack) { RequestTaskInventoryPacket requesttask = (RequestTaskInventoryPacket)Pack; #region Packet Session and User Check if (requesttask.AgentData.SessionID != SessionId || requesttask.AgentData.AgentID != AgentId) return true; #endregion RequestTaskInventory handlerRequestTaskInventory = OnRequestTaskInventory; if (handlerRequestTaskInventory != null) { handlerRequestTaskInventory(this, requesttask.InventoryData.LocalID); } return true; } private bool HandleUpdateTaskInventory(IClientAPI sender, Packet Pack) { UpdateTaskInventoryPacket updatetask = (UpdateTaskInventoryPacket)Pack; #region Packet Session and User Check if (updatetask.AgentData.SessionID != SessionId || updatetask.AgentData.AgentID != AgentId) return true; #endregion if (OnUpdateTaskInventory != null) { if (updatetask.UpdateData.Key == 0) { UpdateTaskInventory handlerUpdateTaskInventory = OnUpdateTaskInventory; if (handlerUpdateTaskInventory != null) { TaskInventoryItem newTaskItem = new TaskInventoryItem(); newTaskItem.ItemID = updatetask.InventoryData.ItemID; newTaskItem.ParentID = updatetask.InventoryData.FolderID; newTaskItem.CreatorID = updatetask.InventoryData.CreatorID; newTaskItem.OwnerID = updatetask.InventoryData.OwnerID; newTaskItem.GroupID = updatetask.InventoryData.GroupID; newTaskItem.BasePermissions = updatetask.InventoryData.BaseMask; newTaskItem.CurrentPermissions = updatetask.InventoryData.OwnerMask; newTaskItem.GroupPermissions = updatetask.InventoryData.GroupMask; newTaskItem.EveryonePermissions = updatetask.InventoryData.EveryoneMask; newTaskItem.NextPermissions = updatetask.InventoryData.NextOwnerMask; // Unused? Clicking share with group sets GroupPermissions instead, so perhaps this is something // different //newTaskItem.GroupOwned=updatetask.InventoryData.GroupOwned; newTaskItem.Type = updatetask.InventoryData.Type; newTaskItem.InvType = updatetask.InventoryData.InvType; newTaskItem.Flags = updatetask.InventoryData.Flags; //newTaskItem.SaleType=updatetask.InventoryData.SaleType; //newTaskItem.SalePrice=updatetask.InventoryData.SalePrice; newTaskItem.Name = Util.FieldToString(updatetask.InventoryData.Name); newTaskItem.Description = Util.FieldToString(updatetask.InventoryData.Description); newTaskItem.CreationDate = (uint)updatetask.InventoryData.CreationDate; handlerUpdateTaskInventory(this, updatetask.InventoryData.TransactionID, newTaskItem, updatetask.UpdateData.LocalID); } } } return true; } private bool HandleRemoveTaskInventory(IClientAPI sender, Packet Pack) { RemoveTaskInventoryPacket removeTask = (RemoveTaskInventoryPacket)Pack; #region Packet Session and User Check if (removeTask.AgentData.SessionID != SessionId || removeTask.AgentData.AgentID != AgentId) return true; #endregion RemoveTaskInventory handlerRemoveTaskItem = OnRemoveTaskItem; if (handlerRemoveTaskItem != null) { handlerRemoveTaskItem(this, removeTask.InventoryData.ItemID, removeTask.InventoryData.LocalID); } return true; } private bool HandleMoveTaskInventory(IClientAPI sender, Packet Pack) { MoveTaskInventoryPacket moveTaskInventoryPacket = (MoveTaskInventoryPacket)Pack; #region Packet Session and User Check if (moveTaskInventoryPacket.AgentData.SessionID != SessionId || moveTaskInventoryPacket.AgentData.AgentID != AgentId) return true; #endregion MoveTaskInventory handlerMoveTaskItem = OnMoveTaskItem; if (handlerMoveTaskItem != null) { handlerMoveTaskItem( this, moveTaskInventoryPacket.AgentData.FolderID, moveTaskInventoryPacket.InventoryData.LocalID, moveTaskInventoryPacket.InventoryData.ItemID); } return true; } private bool HandleRezScript(IClientAPI sender, Packet Pack) { //m_log.Debug(Pack.ToString()); RezScriptPacket rezScriptx = (RezScriptPacket)Pack; #region Packet Session and User Check if (rezScriptx.AgentData.SessionID != SessionId || rezScriptx.AgentData.AgentID != AgentId) return true; #endregion RezScript handlerRezScript = OnRezScript; InventoryItemBase item = new InventoryItemBase(); item.ID = rezScriptx.InventoryBlock.ItemID; item.Folder = rezScriptx.InventoryBlock.FolderID; item.CreatorId = rezScriptx.InventoryBlock.CreatorID.ToString(); item.Owner = rezScriptx.InventoryBlock.OwnerID; item.BasePermissions = rezScriptx.InventoryBlock.BaseMask; item.CurrentPermissions = rezScriptx.InventoryBlock.OwnerMask; item.EveryOnePermissions = rezScriptx.InventoryBlock.EveryoneMask; item.NextPermissions = rezScriptx.InventoryBlock.NextOwnerMask; item.GroupPermissions = rezScriptx.InventoryBlock.GroupMask; item.GroupOwned = rezScriptx.InventoryBlock.GroupOwned; item.GroupID = rezScriptx.InventoryBlock.GroupID; item.AssetType = rezScriptx.InventoryBlock.Type; item.InvType = rezScriptx.InventoryBlock.InvType; item.Flags = rezScriptx.InventoryBlock.Flags; item.SaleType = rezScriptx.InventoryBlock.SaleType; item.SalePrice = rezScriptx.InventoryBlock.SalePrice; item.Name = Util.FieldToString(rezScriptx.InventoryBlock.Name); item.Description = Util.FieldToString(rezScriptx.InventoryBlock.Description); item.CreationDate = rezScriptx.InventoryBlock.CreationDate; if (handlerRezScript != null) { handlerRezScript(this, item, rezScriptx.InventoryBlock.TransactionID, rezScriptx.UpdateBlock.ObjectLocalID); } return true; } private bool HandleMapLayerRequest(IClientAPI sender, Packet Pack) { RequestMapLayer(); return true; } private bool HandleMapBlockRequest(IClientAPI sender, Packet Pack) { MapBlockRequestPacket MapRequest = (MapBlockRequestPacket)Pack; #region Packet Session and User Check if (MapRequest.AgentData.SessionID != SessionId || MapRequest.AgentData.AgentID != AgentId) return true; #endregion RequestMapBlocks handlerRequestMapBlocks = OnRequestMapBlocks; if (handlerRequestMapBlocks != null) { handlerRequestMapBlocks(this, MapRequest.PositionData.MinX, MapRequest.PositionData.MinY, MapRequest.PositionData.MaxX, MapRequest.PositionData.MaxY, MapRequest.AgentData.Flags); } return true; } private bool HandleMapNameRequest(IClientAPI sender, Packet Pack) { MapNameRequestPacket map = (MapNameRequestPacket)Pack; #region Packet Session and User Check if (map.AgentData.SessionID != SessionId || map.AgentData.AgentID != AgentId) return true; #endregion string mapName = (map.NameData.Name.Length == 0) ? m_scene.RegionInfo.RegionName : Util.UTF8.GetString(map.NameData.Name, 0, map.NameData.Name.Length - 1); RequestMapName handlerMapNameRequest = OnMapNameRequest; if (handlerMapNameRequest != null) { handlerMapNameRequest(this, mapName, map.AgentData.Flags); } return true; } private bool HandleTeleportLandmarkRequest(IClientAPI sender, Packet Pack) { TeleportLandmarkRequestPacket tpReq = (TeleportLandmarkRequestPacket)Pack; #region Packet Session and User Check if (tpReq.Info.SessionID != SessionId || tpReq.Info.AgentID != AgentId) return true; #endregion UUID lmid = tpReq.Info.LandmarkID; AssetLandmark lm; if (lmid != UUID.Zero) { //AssetBase lma = m_assetCache.GetAsset(lmid, false); AssetBase lma = m_assetService.Get(lmid.ToString()); if (lma == null) { // Failed to find landmark // Let's try to search in the user's home asset server lma = FindAssetInUserAssetServer(lmid.ToString()); if (lma == null) { // Really doesn't exist TeleportCancelPacket tpCancel = (TeleportCancelPacket)PacketPool.Instance.GetPacket(PacketType.TeleportCancel); tpCancel.Info.SessionID = tpReq.Info.SessionID; tpCancel.Info.AgentID = tpReq.Info.AgentID; OutPacket(tpCancel, ThrottleOutPacketType.Task); } } try { lm = new AssetLandmark(lma); } catch (NullReferenceException) { // asset not found generates null ref inside the assetlandmark constructor. TeleportCancelPacket tpCancel = (TeleportCancelPacket)PacketPool.Instance.GetPacket(PacketType.TeleportCancel); tpCancel.Info.SessionID = tpReq.Info.SessionID; tpCancel.Info.AgentID = tpReq.Info.AgentID; OutPacket(tpCancel, ThrottleOutPacketType.Task); return true; } } else { // Teleport home request UUIDNameRequest handlerTeleportHomeRequest = OnTeleportHomeRequest; if (handlerTeleportHomeRequest != null) { handlerTeleportHomeRequest(AgentId, this); } return true; } TeleportLandmarkRequest handlerTeleportLandmarkRequest = OnTeleportLandmarkRequest; if (handlerTeleportLandmarkRequest != null) { handlerTeleportLandmarkRequest(this, lm); } else { //no event handler so cancel request TeleportCancelPacket tpCancel = (TeleportCancelPacket)PacketPool.Instance.GetPacket(PacketType.TeleportCancel); tpCancel.Info.AgentID = tpReq.Info.AgentID; tpCancel.Info.SessionID = tpReq.Info.SessionID; OutPacket(tpCancel, ThrottleOutPacketType.Task); } return true; } private bool HandleTeleportCancel(IClientAPI sender, Packet Pack) { TeleportCancel handlerTeleportCancel = OnTeleportCancel; if (handlerTeleportCancel != null) { handlerTeleportCancel(this); } return true; } private AssetBase FindAssetInUserAssetServer(string id) { AgentCircuitData aCircuit = ((Scene)Scene).AuthenticateHandler.GetAgentCircuitData(CircuitCode); if (aCircuit != null && aCircuit.ServiceURLs != null && aCircuit.ServiceURLs.ContainsKey("AssetServerURI")) { string assetServer = aCircuit.ServiceURLs["AssetServerURI"].ToString(); if (!string.IsNullOrEmpty(assetServer)) return ((Scene)Scene).AssetService.Get(assetServer + "/" + id); } return null; } private bool HandleTeleportLocationRequest(IClientAPI sender, Packet Pack) { TeleportLocationRequestPacket tpLocReq = (TeleportLocationRequestPacket)Pack; // m_log.Debug(tpLocReq.ToString()); #region Packet Session and User Check if (tpLocReq.AgentData.SessionID != SessionId || tpLocReq.AgentData.AgentID != AgentId) return true; #endregion TeleportLocationRequest handlerTeleportLocationRequest = OnTeleportLocationRequest; if (handlerTeleportLocationRequest != null) { // Adjust teleport location to base of a larger region if requested to teleport to a sub-region uint locX, locY; Util.RegionHandleToWorldLoc(tpLocReq.Info.RegionHandle, out locX, out locY); if ((locX >= m_scene.RegionInfo.WorldLocX) && (locX < (m_scene.RegionInfo.WorldLocX + m_scene.RegionInfo.RegionSizeX)) && (locY >= m_scene.RegionInfo.WorldLocY) && (locY < (m_scene.RegionInfo.WorldLocY + m_scene.RegionInfo.RegionSizeY))) { tpLocReq.Info.RegionHandle = m_scene.RegionInfo.RegionHandle; tpLocReq.Info.Position.X += locX - m_scene.RegionInfo.WorldLocX; tpLocReq.Info.Position.Y += locY - m_scene.RegionInfo.WorldLocY; } handlerTeleportLocationRequest(this, tpLocReq.Info.RegionHandle, tpLocReq.Info.Position, tpLocReq.Info.LookAt, 16); } else { //no event handler so cancel request TeleportCancelPacket tpCancel = (TeleportCancelPacket)PacketPool.Instance.GetPacket(PacketType.TeleportCancel); tpCancel.Info.SessionID = tpLocReq.AgentData.SessionID; tpCancel.Info.AgentID = tpLocReq.AgentData.AgentID; OutPacket(tpCancel, ThrottleOutPacketType.Task); } return true; } #endregion Inventory/Asset/Other related packets private bool HandleUUIDNameRequest(IClientAPI sender, Packet Pack) { ScenePresence sp = (ScenePresence)SceneAgent; if(sp == null || sp.IsDeleted || (sp.IsInTransit && !sp.IsInLocalTransit)) return true; UUIDNameRequestPacket incoming = (UUIDNameRequestPacket)Pack; foreach (UUIDNameRequestPacket.UUIDNameBlockBlock UUIDBlock in incoming.UUIDNameBlock) { UUIDNameRequest handlerNameRequest = OnNameFromUUIDRequest; if (handlerNameRequest != null) { handlerNameRequest(UUIDBlock.ID, this); } } return true; } #region Parcel related packets private bool HandleRegionHandleRequest(IClientAPI sender, Packet Pack) { RegionHandleRequest handlerRegionHandleRequest = OnRegionHandleRequest; if (handlerRegionHandleRequest != null) { RegionHandleRequestPacket rhrPack = (RegionHandleRequestPacket)Pack; handlerRegionHandleRequest(this, rhrPack.RequestBlock.RegionID); } return true; } private bool HandleParcelInfoRequest(IClientAPI sender, Packet Pack) { ParcelInfoRequestPacket pirPack = (ParcelInfoRequestPacket)Pack; #region Packet Session and User Check if (pirPack.AgentData.SessionID != SessionId || pirPack.AgentData.AgentID != AgentId) return true; #endregion ParcelInfoRequest handlerParcelInfoRequest = OnParcelInfoRequest; if (handlerParcelInfoRequest != null) { handlerParcelInfoRequest(this, pirPack.Data.ParcelID); } return true; } private bool HandleParcelAccessListRequest(IClientAPI sender, Packet Pack) { ParcelAccessListRequestPacket requestPacket = (ParcelAccessListRequestPacket)Pack; #region Packet Session and User Check if (requestPacket.AgentData.SessionID != SessionId || requestPacket.AgentData.AgentID != AgentId) return true; #endregion ParcelAccessListRequest handlerParcelAccessListRequest = OnParcelAccessListRequest; if (handlerParcelAccessListRequest != null) { handlerParcelAccessListRequest(requestPacket.AgentData.AgentID, requestPacket.AgentData.SessionID, requestPacket.Data.Flags, requestPacket.Data.SequenceID, requestPacket.Data.LocalID, this); } return true; } private bool HandleParcelAccessListUpdate(IClientAPI sender, Packet Pack) { if(OnParcelAccessListUpdateRequest == null) return true; ParcelAccessListUpdatePacket updatePacket = (ParcelAccessListUpdatePacket)Pack; #region Packet Session and User Check if (updatePacket.AgentData.SessionID != SessionId || updatePacket.AgentData.AgentID != AgentId) return true; #endregion // viewers do send estimated number of packets and sequenceID, but don't seem reliable. List entries = new List(); foreach (ParcelAccessListUpdatePacket.ListBlock block in updatePacket.List) { LandAccessEntry entry = new LandAccessEntry(); entry.AgentID = block.ID; entry.Flags = (AccessList)block.Flags; entry.Expires = block.Time; entries.Add(entry); } ParcelAccessListUpdateRequest handlerParcelAccessListUpdateRequest = OnParcelAccessListUpdateRequest; if (handlerParcelAccessListUpdateRequest != null) { handlerParcelAccessListUpdateRequest(updatePacket.AgentData.AgentID, updatePacket.Data.Flags, updatePacket.Data.TransactionID, updatePacket.Data.LocalID, entries, this); } return true; } private bool HandleParcelPropertiesRequest(IClientAPI sender, Packet Pack) { ParcelPropertiesRequestPacket propertiesRequest = (ParcelPropertiesRequestPacket)Pack; #region Packet Session and User Check if (propertiesRequest.AgentData.SessionID != SessionId || propertiesRequest.AgentData.AgentID != AgentId) return true; #endregion ParcelPropertiesRequest handlerParcelPropertiesRequest = OnParcelPropertiesRequest; if (handlerParcelPropertiesRequest != null) { handlerParcelPropertiesRequest((int)Math.Round(propertiesRequest.ParcelData.West), (int)Math.Round(propertiesRequest.ParcelData.South), (int)Math.Round(propertiesRequest.ParcelData.East), (int)Math.Round(propertiesRequest.ParcelData.North), propertiesRequest.ParcelData.SequenceID, propertiesRequest.ParcelData.SnapSelection, this); } return true; } private bool HandleParcelDivide(IClientAPI sender, Packet Pack) { ParcelDividePacket landDivide = (ParcelDividePacket)Pack; #region Packet Session and User Check if (landDivide.AgentData.SessionID != SessionId || landDivide.AgentData.AgentID != AgentId) return true; #endregion ParcelDivideRequest handlerParcelDivideRequest = OnParcelDivideRequest; if (handlerParcelDivideRequest != null) { handlerParcelDivideRequest((int)Math.Round(landDivide.ParcelData.West), (int)Math.Round(landDivide.ParcelData.South), (int)Math.Round(landDivide.ParcelData.East), (int)Math.Round(landDivide.ParcelData.North), this); } return true; } private bool HandleParcelJoin(IClientAPI sender, Packet Pack) { ParcelJoinPacket landJoin = (ParcelJoinPacket)Pack; #region Packet Session and User Check if (landJoin.AgentData.SessionID != SessionId || landJoin.AgentData.AgentID != AgentId) return true; #endregion ParcelJoinRequest handlerParcelJoinRequest = OnParcelJoinRequest; if (handlerParcelJoinRequest != null) { handlerParcelJoinRequest((int)Math.Round(landJoin.ParcelData.West), (int)Math.Round(landJoin.ParcelData.South), (int)Math.Round(landJoin.ParcelData.East), (int)Math.Round(landJoin.ParcelData.North), this); } return true; } private bool HandleParcelPropertiesUpdate(IClientAPI sender, Packet Pack) { ParcelPropertiesUpdatePacket parcelPropertiesPacket = (ParcelPropertiesUpdatePacket)Pack; #region Packet Session and User Check if (parcelPropertiesPacket.AgentData.SessionID != SessionId || parcelPropertiesPacket.AgentData.AgentID != AgentId) return true; #endregion ParcelPropertiesUpdateRequest handlerParcelPropertiesUpdateRequest = OnParcelPropertiesUpdateRequest; if (handlerParcelPropertiesUpdateRequest != null) { LandUpdateArgs args = new LandUpdateArgs(); args.AuthBuyerID = parcelPropertiesPacket.ParcelData.AuthBuyerID; args.Category = (ParcelCategory)parcelPropertiesPacket.ParcelData.Category; args.Desc = Utils.BytesToString(parcelPropertiesPacket.ParcelData.Desc); args.GroupID = parcelPropertiesPacket.ParcelData.GroupID; args.LandingType = parcelPropertiesPacket.ParcelData.LandingType; args.MediaAutoScale = parcelPropertiesPacket.ParcelData.MediaAutoScale; args.MediaID = parcelPropertiesPacket.ParcelData.MediaID; args.MediaURL = Utils.BytesToString(parcelPropertiesPacket.ParcelData.MediaURL); args.MusicURL = Utils.BytesToString(parcelPropertiesPacket.ParcelData.MusicURL); args.Name = Utils.BytesToString(parcelPropertiesPacket.ParcelData.Name); args.ParcelFlags = parcelPropertiesPacket.ParcelData.ParcelFlags; args.PassHours = parcelPropertiesPacket.ParcelData.PassHours; args.PassPrice = parcelPropertiesPacket.ParcelData.PassPrice; args.SalePrice = parcelPropertiesPacket.ParcelData.SalePrice; args.SnapshotID = parcelPropertiesPacket.ParcelData.SnapshotID; args.UserLocation = parcelPropertiesPacket.ParcelData.UserLocation; args.UserLookAt = parcelPropertiesPacket.ParcelData.UserLookAt; handlerParcelPropertiesUpdateRequest(args, parcelPropertiesPacket.ParcelData.LocalID, this); } return true; } private bool HandleParcelSelectObjects(IClientAPI sender, Packet Pack) { ParcelSelectObjectsPacket selectPacket = (ParcelSelectObjectsPacket)Pack; #region Packet Session and User Check if (selectPacket.AgentData.SessionID != SessionId || selectPacket.AgentData.AgentID != AgentId) return true; #endregion List returnIDs = new List(); foreach (ParcelSelectObjectsPacket.ReturnIDsBlock rb in selectPacket.ReturnIDs) { returnIDs.Add(rb.ReturnID); } ParcelSelectObjects handlerParcelSelectObjects = OnParcelSelectObjects; if (handlerParcelSelectObjects != null) { handlerParcelSelectObjects(selectPacket.ParcelData.LocalID, Convert.ToInt32(selectPacket.ParcelData.ReturnType), returnIDs, this); } return true; } private bool HandleParcelObjectOwnersRequest(IClientAPI sender, Packet Pack) { ParcelObjectOwnersRequestPacket reqPacket = (ParcelObjectOwnersRequestPacket)Pack; #region Packet Session and User Check if (reqPacket.AgentData.SessionID != SessionId || reqPacket.AgentData.AgentID != AgentId) return true; #endregion ParcelObjectOwnerRequest handlerParcelObjectOwnerRequest = OnParcelObjectOwnerRequest; if (handlerParcelObjectOwnerRequest != null) { handlerParcelObjectOwnerRequest(reqPacket.ParcelData.LocalID, this); } return true; } private bool HandleParcelGodForceOwner(IClientAPI sender, Packet Pack) { ParcelGodForceOwnerPacket godForceOwnerPacket = (ParcelGodForceOwnerPacket)Pack; #region Packet Session and User Check if (godForceOwnerPacket.AgentData.SessionID != SessionId || godForceOwnerPacket.AgentData.AgentID != AgentId) return true; #endregion ParcelGodForceOwner handlerParcelGodForceOwner = OnParcelGodForceOwner; if (handlerParcelGodForceOwner != null) { handlerParcelGodForceOwner(godForceOwnerPacket.Data.LocalID, godForceOwnerPacket.Data.OwnerID, this); } return true; } private bool HandleParcelRelease(IClientAPI sender, Packet Pack) { ParcelReleasePacket releasePacket = (ParcelReleasePacket)Pack; #region Packet Session and User Check if (releasePacket.AgentData.SessionID != SessionId || releasePacket.AgentData.AgentID != AgentId) return true; #endregion ParcelAbandonRequest handlerParcelAbandonRequest = OnParcelAbandonRequest; if (handlerParcelAbandonRequest != null) { handlerParcelAbandonRequest(releasePacket.Data.LocalID, this); } return true; } private bool HandleParcelReclaim(IClientAPI sender, Packet Pack) { ParcelReclaimPacket reclaimPacket = (ParcelReclaimPacket)Pack; #region Packet Session and User Check if (reclaimPacket.AgentData.SessionID != SessionId || reclaimPacket.AgentData.AgentID != AgentId) return true; #endregion ParcelReclaim handlerParcelReclaim = OnParcelReclaim; if (handlerParcelReclaim != null) { handlerParcelReclaim(reclaimPacket.Data.LocalID, this); } return true; } private bool HandleParcelReturnObjects(IClientAPI sender, Packet Pack) { ParcelReturnObjectsPacket parcelReturnObjects = (ParcelReturnObjectsPacket)Pack; #region Packet Session and User Check if (parcelReturnObjects.AgentData.SessionID != SessionId || parcelReturnObjects.AgentData.AgentID != AgentId) return true; #endregion UUID[] puserselectedOwnerIDs = new UUID[parcelReturnObjects.OwnerIDs.Length]; for (int parceliterator = 0; parceliterator < parcelReturnObjects.OwnerIDs.Length; parceliterator++) puserselectedOwnerIDs[parceliterator] = parcelReturnObjects.OwnerIDs[parceliterator].OwnerID; UUID[] puserselectedTaskIDs = new UUID[parcelReturnObjects.TaskIDs.Length]; for (int parceliterator = 0; parceliterator < parcelReturnObjects.TaskIDs.Length; parceliterator++) puserselectedTaskIDs[parceliterator] = parcelReturnObjects.TaskIDs[parceliterator].TaskID; ParcelReturnObjectsRequest handlerParcelReturnObjectsRequest = OnParcelReturnObjectsRequest; if (handlerParcelReturnObjectsRequest != null) { handlerParcelReturnObjectsRequest(parcelReturnObjects.ParcelData.LocalID, parcelReturnObjects.ParcelData.ReturnType, puserselectedOwnerIDs, puserselectedTaskIDs, this); } return true; } private bool HandleParcelSetOtherCleanTime(IClientAPI sender, Packet Pack) { ParcelSetOtherCleanTimePacket parcelSetOtherCleanTimePacket = (ParcelSetOtherCleanTimePacket)Pack; #region Packet Session and User Check if (parcelSetOtherCleanTimePacket.AgentData.SessionID != SessionId || parcelSetOtherCleanTimePacket.AgentData.AgentID != AgentId) return true; #endregion ParcelSetOtherCleanTime handlerParcelSetOtherCleanTime = OnParcelSetOtherCleanTime; if (handlerParcelSetOtherCleanTime != null) { handlerParcelSetOtherCleanTime(this, parcelSetOtherCleanTimePacket.ParcelData.LocalID, parcelSetOtherCleanTimePacket.ParcelData.OtherCleanTime); } return true; } private bool HandleLandStatRequest(IClientAPI sender, Packet Pack) { LandStatRequestPacket lsrp = (LandStatRequestPacket)Pack; #region Packet Session and User Check if (lsrp.AgentData.SessionID != SessionId || lsrp.AgentData.AgentID != AgentId) return true; #endregion GodLandStatRequest handlerLandStatRequest = OnLandStatRequest; if (handlerLandStatRequest != null) { handlerLandStatRequest(lsrp.RequestData.ParcelLocalID, lsrp.RequestData.ReportType, lsrp.RequestData.RequestFlags, Utils.BytesToString(lsrp.RequestData.Filter), this); } return true; } private bool HandleParcelDwellRequest(IClientAPI sender, Packet Pack) { ParcelDwellRequestPacket dwellrq = (ParcelDwellRequestPacket)Pack; #region Packet Session and User Check if (dwellrq.AgentData.SessionID != SessionId || dwellrq.AgentData.AgentID != AgentId) return true; #endregion ParcelDwellRequest handlerParcelDwellRequest = OnParcelDwellRequest; if (handlerParcelDwellRequest != null) { handlerParcelDwellRequest(dwellrq.Data.LocalID, this); } return true; } #endregion Parcel related packets #region Estate Packets private static double m_lastMapRegenTime = Double.MinValue; private bool HandleEstateOwnerMessage(IClientAPI sender, Packet Pack) { EstateOwnerMessagePacket messagePacket = (EstateOwnerMessagePacket)Pack; // m_log.InfoFormat("[LLCLIENTVIEW]: Packet: {0}", Utils.BytesToString(messagePacket.MethodData.Method)); GodLandStatRequest handlerLandStatRequest; #region Packet Session and User Check if (messagePacket.AgentData.SessionID != SessionId || messagePacket.AgentData.AgentID != AgentId) return true; #endregion string method = Utils.BytesToString(messagePacket.MethodData.Method); switch (method) { case "getinfo": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { OnDetailedEstateDataRequest(this, messagePacket.MethodData.Invoice); } return true; case "setregioninfo": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { OnSetEstateFlagsRequest(convertParamStringToBool(messagePacket.ParamList[0].Parameter), convertParamStringToBool(messagePacket.ParamList[1].Parameter), convertParamStringToBool(messagePacket.ParamList[2].Parameter), !convertParamStringToBool(messagePacket.ParamList[3].Parameter), Convert.ToInt16(Convert.ToDecimal(Utils.BytesToString(messagePacket.ParamList[4].Parameter), Culture.NumberFormatInfo)), (float)Convert.ToDecimal(Utils.BytesToString(messagePacket.ParamList[5].Parameter), Culture.NumberFormatInfo), Convert.ToInt16(Utils.BytesToString(messagePacket.ParamList[6].Parameter)), convertParamStringToBool(messagePacket.ParamList[7].Parameter), convertParamStringToBool(messagePacket.ParamList[8].Parameter)); } return true; // case "texturebase": // if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) // { // foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) // { // string s = Utils.BytesToString(block.Parameter); // string[] splitField = s.Split(' '); // if (splitField.Length == 2) // { // UUID tempUUID = new UUID(splitField[1]); // OnSetEstateTerrainBaseTexture(this, Convert.ToInt16(splitField[0]), tempUUID); // } // } // } // break; case "texturedetail": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) { string s = Utils.BytesToString(block.Parameter); string[] splitField = s.Split(' '); if (splitField.Length == 2) { Int16 corner = Convert.ToInt16(splitField[0]); UUID textureUUID = new UUID(splitField[1]); OnSetEstateTerrainDetailTexture(this, corner, textureUUID); } } } return true; case "textureheights": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) { string s = Utils.BytesToString(block.Parameter); string[] splitField = s.Split(' '); if (splitField.Length == 3) { Int16 corner = Convert.ToInt16(splitField[0]); float lowValue = (float)Convert.ToDecimal(splitField[1], Culture.NumberFormatInfo); float highValue = (float)Convert.ToDecimal(splitField[2], Culture.NumberFormatInfo); OnSetEstateTerrainTextureHeights(this, corner, lowValue, highValue); } } } return true; case "texturecommit": OnCommitEstateTerrainTextureRequest(this); return true; case "setregionterrain": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { if (messagePacket.ParamList.Length != 9) { m_log.Error("EstateOwnerMessage: SetRegionTerrain method has a ParamList of invalid length"); } else { try { string tmp = Utils.BytesToString(messagePacket.ParamList[0].Parameter); if (!tmp.Contains(".")) tmp += ".00"; float WaterHeight = (float)Convert.ToDecimal(tmp, Culture.NumberFormatInfo); tmp = Utils.BytesToString(messagePacket.ParamList[1].Parameter); if (!tmp.Contains(".")) tmp += ".00"; float TerrainRaiseLimit = (float)Convert.ToDecimal(tmp, Culture.NumberFormatInfo); tmp = Utils.BytesToString(messagePacket.ParamList[2].Parameter); if (!tmp.Contains(".")) tmp += ".00"; float TerrainLowerLimit = (float)Convert.ToDecimal(tmp, Culture.NumberFormatInfo); bool UseEstateSun = convertParamStringToBool(messagePacket.ParamList[3].Parameter); bool UseFixedSun = convertParamStringToBool(messagePacket.ParamList[4].Parameter); float SunHour = (float)Convert.ToDecimal(Utils.BytesToString(messagePacket.ParamList[5].Parameter), Culture.NumberFormatInfo); bool UseGlobal = convertParamStringToBool(messagePacket.ParamList[6].Parameter); bool EstateFixedSun = convertParamStringToBool(messagePacket.ParamList[7].Parameter); float EstateSunHour = (float)Convert.ToDecimal(Utils.BytesToString(messagePacket.ParamList[8].Parameter), Culture.NumberFormatInfo); OnSetRegionTerrainSettings(WaterHeight, TerrainRaiseLimit, TerrainLowerLimit, UseEstateSun, UseFixedSun, SunHour, UseGlobal, EstateFixedSun, EstateSunHour); } catch (Exception ex) { m_log.Error("EstateOwnerMessage: Exception while setting terrain settings: \n" + messagePacket + "\n" + ex); } } } return true; case "restart": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { // There's only 1 block in the estateResetSim.. and that's the number of seconds till restart. foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) { float timeSeconds; Utils.TryParseSingle(Utils.BytesToString(block.Parameter), out timeSeconds); timeSeconds = (int)timeSeconds; OnEstateRestartSimRequest(this, (int)timeSeconds); } } return true; case "estatechangecovenantid": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) { UUID newCovenantID = new UUID(Utils.BytesToString(block.Parameter)); OnEstateChangeCovenantRequest(this, newCovenantID); } } return true; case "estateaccessdelta": // Estate access delta manages the banlist and allow list too. if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { int estateAccessType = Convert.ToInt16(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); OnUpdateEstateAccessDeltaRequest(this, messagePacket.MethodData.Invoice, estateAccessType, new UUID(Utils.BytesToString(messagePacket.ParamList[2].Parameter))); } return true; case "simulatormessage": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = new UUID(Utils.BytesToString(messagePacket.ParamList[2].Parameter)); string SenderName = Utils.BytesToString(messagePacket.ParamList[3].Parameter); string Message = Utils.BytesToString(messagePacket.ParamList[4].Parameter); UUID sessionID = messagePacket.AgentData.SessionID; OnSimulatorBlueBoxMessageRequest(this, invoice, SenderID, sessionID, SenderName, Message); } return true; case "instantmessage": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { if (messagePacket.ParamList.Length < 2) return true; UUID invoice = messagePacket.MethodData.Invoice; UUID sessionID = messagePacket.AgentData.SessionID; UUID SenderID; string SenderName; string Message; if (messagePacket.ParamList.Length < 5) { SenderID = AgentId; SenderName = Utils.BytesToString(messagePacket.ParamList[0].Parameter); Message = Utils.BytesToString(messagePacket.ParamList[1].Parameter); } else { SenderID = new UUID(Utils.BytesToString(messagePacket.ParamList[2].Parameter)); SenderName = Utils.BytesToString(messagePacket.ParamList[3].Parameter); Message = Utils.BytesToString(messagePacket.ParamList[4].Parameter); } OnEstateBlueBoxMessageRequest(this, invoice, SenderID, sessionID, SenderName, Message); } return true; case "setregiondebug": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; bool scripted = convertParamStringToBool(messagePacket.ParamList[0].Parameter); bool collisionEvents = convertParamStringToBool(messagePacket.ParamList[1].Parameter); bool physics = convertParamStringToBool(messagePacket.ParamList[2].Parameter); OnEstateDebugRegionRequest(this, invoice, SenderID, scripted, collisionEvents, physics); } return true; case "teleporthomeuser": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; UUID Prey; UUID.TryParse(Utils.BytesToString(messagePacket.ParamList[1].Parameter), out Prey); OnEstateTeleportOneUserHomeRequest(this, invoice, SenderID, Prey, false); } return true; case "teleporthomeallusers": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; OnEstateTeleportAllUsersHomeRequest(this, invoice, SenderID); } return true; case "colliders": handlerLandStatRequest = OnLandStatRequest; if (handlerLandStatRequest != null) { handlerLandStatRequest(0, 1, 0, "", this); } return true; case "scripts": handlerLandStatRequest = OnLandStatRequest; if (handlerLandStatRequest != null) { handlerLandStatRequest(0, 0, 0, "", this); } return true; case "terrain": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { if (messagePacket.ParamList.Length > 0) { if (Utils.BytesToString(messagePacket.ParamList[0].Parameter) == "bake") { BakeTerrain handlerBakeTerrain = OnBakeTerrain; if (handlerBakeTerrain != null) { handlerBakeTerrain(this); } } if (Utils.BytesToString(messagePacket.ParamList[0].Parameter) == "download filename") { if (messagePacket.ParamList.Length > 1) { RequestTerrain handlerRequestTerrain = OnRequestTerrain; if (handlerRequestTerrain != null) { handlerRequestTerrain(this, Utils.BytesToString(messagePacket.ParamList[1].Parameter)); } } } if (Utils.BytesToString(messagePacket.ParamList[0].Parameter) == "upload filename") { if (messagePacket.ParamList.Length > 1) { RequestTerrain handlerUploadTerrain = OnUploadTerrain; if (handlerUploadTerrain != null) { handlerUploadTerrain(this, Utils.BytesToString(messagePacket.ParamList[1].Parameter)); } } } } } return true; case "estatechangeinfo": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; UInt32 param1 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); UInt32 param2 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[2].Parameter)); EstateChangeInfo handlerEstateChangeInfo = OnEstateChangeInfo; if (handlerEstateChangeInfo != null) { handlerEstateChangeInfo(this, invoice, SenderID, param1, param2); } } return true; case "telehub": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; UInt32 param1 = 0u; string command = (string)Utils.BytesToString(messagePacket.ParamList[0].Parameter); if (command != "info ui") { try { param1 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); } catch { } } EstateManageTelehub handlerEstateManageTelehub = OnEstateManageTelehub; if (handlerEstateManageTelehub != null) { handlerEstateManageTelehub(this, invoice, SenderID, command, param1); } } return true; case "refreshmapvisibility": if (((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { IWorldMapModule mapModule = Scene.RequestModuleInterface(); if (mapModule == null) { SendAlertMessage("Terrain map generator not avaiable"); return true; } if (m_lastMapRegenTime == Double.MaxValue) { SendAlertMessage("Terrain map generation still in progress"); return true; } double now = Util.GetTimeStamp(); if (now - m_lastMapRegenTime < 120) // 2 minutes global cool down { SendAlertMessage("Please wait at least 2 minutes between map generation commands"); return true; } m_lastMapRegenTime = Double.MaxValue; mapModule.GenerateMaptile(); SendAlertMessage("Terrain map generated"); m_lastMapRegenTime = now; } return true; case "kickestate": if(((Scene)m_scene).Permissions.CanIssueEstateCommand(AgentId, false)) { UUID invoice = messagePacket.MethodData.Invoice; UUID SenderID = messagePacket.AgentData.AgentID; UUID Prey; UUID.TryParse(Utils.BytesToString(messagePacket.ParamList[0].Parameter), out Prey); OnEstateTeleportOneUserHomeRequest(this, invoice, SenderID, Prey, true); } return true; default: m_log.WarnFormat( "[LLCLIENTVIEW]: EstateOwnerMessage: Unknown method {0} requested for {1} in {2}", method, Name, Scene.Name); for (int i = 0; i < messagePacket.ParamList.Length; i++) { EstateOwnerMessagePacket.ParamListBlock block = messagePacket.ParamList[i]; string data = (string)Utils.BytesToString(block.Parameter); m_log.DebugFormat("[LLCLIENTVIEW]: Param {0}={1}", i, data); } return true; } //int parcelID, uint reportType, uint requestflags, string filter //lsrp.RequestData.ParcelLocalID; //lsrp.RequestData.ReportType; // 1 = colliders, 0 = scripts //lsrp.RequestData.RequestFlags; //lsrp.RequestData.Filter; } private bool HandleRequestRegionInfo(IClientAPI sender, Packet Pack) { RequestRegionInfoPacket.AgentDataBlock mPacket = ((RequestRegionInfoPacket)Pack).AgentData; #region Packet Session and User Check if (mPacket.SessionID != SessionId || mPacket.AgentID != AgentId) return true; #endregion RegionInfoRequest handlerRegionInfoRequest = OnRegionInfoRequest; if (handlerRegionInfoRequest != null) { handlerRegionInfoRequest(this); } return true; } private bool HandleEstateCovenantRequest(IClientAPI sender, Packet Pack) { //EstateCovenantRequestPacket.AgentDataBlock epack = // ((EstateCovenantRequestPacket)Pack).AgentData; EstateCovenantRequest handlerEstateCovenantRequest = OnEstateCovenantRequest; if (handlerEstateCovenantRequest != null) { handlerEstateCovenantRequest(this); } return true; } #endregion Estate Packets #region GodPackets private bool HandleRequestGodlikePowers(IClientAPI sender, Packet Pack) { RequestGodlikePowersPacket rglpPack = (RequestGodlikePowersPacket)Pack; if (rglpPack.AgentData.SessionID != SessionId || rglpPack.AgentData.AgentID != AgentId) return true; RequestGodlikePowersPacket.RequestBlockBlock rblock = rglpPack.RequestBlock; UUID token = rblock.Token; RequestGodlikePowersPacket.AgentDataBlock ablock = rglpPack.AgentData; RequestGodlikePowers handlerReqGodlikePowers = OnRequestGodlikePowers; if (handlerReqGodlikePowers != null) { handlerReqGodlikePowers(ablock.AgentID, ablock.SessionID, token, rblock.Godlike); } return true; } private bool HandleGodUpdateRegionInfoUpdate(IClientAPI client, Packet Packet) { GodUpdateRegionInfoPacket GodUpdateRegionInfo = (GodUpdateRegionInfoPacket)Packet; if (GodUpdateRegionInfo.AgentData.SessionID != SessionId || GodUpdateRegionInfo.AgentData.AgentID != AgentId) return true; GodUpdateRegionInfoUpdate handlerGodUpdateRegionInfo = OnGodUpdateRegionInfoUpdate; if (handlerGodUpdateRegionInfo != null) { handlerGodUpdateRegionInfo(this, GodUpdateRegionInfo.RegionInfo.BillableFactor, GodUpdateRegionInfo.RegionInfo.EstateID, GodUpdateRegionInfo.RegionInfo.RegionFlags, GodUpdateRegionInfo.RegionInfo.SimName, GodUpdateRegionInfo.RegionInfo.RedirectGridX, GodUpdateRegionInfo.RegionInfo.RedirectGridY); return true; } return false; } private bool HandleSimWideDeletes(IClientAPI client, Packet Packet) { SimWideDeletesPacket SimWideDeletesRequest = (SimWideDeletesPacket)Packet; SimWideDeletesDelegate handlerSimWideDeletesRequest = OnSimWideDeletes; if (handlerSimWideDeletesRequest != null) { handlerSimWideDeletesRequest(this, SimWideDeletesRequest.AgentData.AgentID,(int)SimWideDeletesRequest.DataBlock.Flags,SimWideDeletesRequest.DataBlock.TargetID); return true; } return false; } private bool HandleGodlikeMessage(IClientAPI client, Packet Packet) { GodlikeMessagePacket GodlikeMessage = (GodlikeMessagePacket)Packet; if (GodlikeMessage.AgentData.SessionID != SessionId || GodlikeMessage.AgentData.AgentID != AgentId) return true; GodlikeMessage handlerGodlikeMessage = onGodlikeMessage; if (handlerGodlikeMessage != null) { handlerGodlikeMessage(this, GodlikeMessage.MethodData.Invoice, GodlikeMessage.MethodData.Method, GodlikeMessage.ParamList[0].Parameter); return true; } return false; } private bool HandleSaveStatePacket(IClientAPI client, Packet Packet) { StateSavePacket SaveStateMessage = (StateSavePacket)Packet; if (SaveStateMessage.AgentData.SessionID != SessionId || SaveStateMessage.AgentData.AgentID != AgentId) return true; SaveStateHandler handlerSaveStatePacket = OnSaveState; if (handlerSaveStatePacket != null) { handlerSaveStatePacket(this,SaveStateMessage.AgentData.AgentID); return true; } return false; } private bool HandleGodKickUser(IClientAPI sender, Packet Pack) { GodKickUserPacket gkupack = (GodKickUserPacket)Pack; if (gkupack.UserInfo.GodSessionID != SessionId || gkupack.UserInfo.GodID != AgentId) return true; GodKickUser handlerGodKickUser = OnGodKickUser; if (handlerGodKickUser != null) { handlerGodKickUser(gkupack.UserInfo.GodID, gkupack.UserInfo.AgentID, gkupack.UserInfo.KickFlags, gkupack.UserInfo.Reason); } return true; } #endregion GodPackets #region Economy/Transaction Packets private bool HandleMoneyBalanceRequest(IClientAPI sender, Packet Pack) { MoneyBalanceRequestPacket moneybalancerequestpacket = (MoneyBalanceRequestPacket)Pack; #region Packet Session and User Check if (moneybalancerequestpacket.AgentData.SessionID != SessionId || moneybalancerequestpacket.AgentData.AgentID != AgentId) return true; #endregion MoneyBalanceRequest handlerMoneyBalanceRequest = OnMoneyBalanceRequest; if (handlerMoneyBalanceRequest != null) { handlerMoneyBalanceRequest(this, moneybalancerequestpacket.AgentData.AgentID, moneybalancerequestpacket.AgentData.SessionID, moneybalancerequestpacket.MoneyData.TransactionID); } return true; } private bool HandleEconomyDataRequest(IClientAPI sender, Packet Pack) { EconomyDataRequest handlerEconomoyDataRequest = OnEconomyDataRequest; if (handlerEconomoyDataRequest != null) { handlerEconomoyDataRequest(this); } return true; } private bool HandleRequestPayPrice(IClientAPI sender, Packet Pack) { RequestPayPricePacket requestPayPricePacket = (RequestPayPricePacket)Pack; RequestPayPrice handlerRequestPayPrice = OnRequestPayPrice; if (handlerRequestPayPrice != null) { handlerRequestPayPrice(this, requestPayPricePacket.ObjectData.ObjectID); } return true; } private bool HandleObjectSaleInfo(IClientAPI sender, Packet Pack) { ObjectSaleInfoPacket objectSaleInfoPacket = (ObjectSaleInfoPacket)Pack; #region Packet Session and User Check if (objectSaleInfoPacket.AgentData.SessionID != SessionId || objectSaleInfoPacket.AgentData.AgentID != AgentId) return true; #endregion ObjectSaleInfo handlerObjectSaleInfo = OnObjectSaleInfo; if (handlerObjectSaleInfo != null) { foreach (ObjectSaleInfoPacket.ObjectDataBlock d in objectSaleInfoPacket.ObjectData) { handlerObjectSaleInfo(this, objectSaleInfoPacket.AgentData.AgentID, objectSaleInfoPacket.AgentData.SessionID, d.LocalID, d.SaleType, d.SalePrice); } } return true; } private bool HandleObjectBuy(IClientAPI sender, Packet Pack) { ObjectBuyPacket objectBuyPacket = (ObjectBuyPacket)Pack; #region Packet Session and User Check if (objectBuyPacket.AgentData.SessionID != SessionId || objectBuyPacket.AgentData.AgentID != AgentId) return true; #endregion ObjectBuy handlerObjectBuy = OnObjectBuy; if (handlerObjectBuy != null) { foreach (ObjectBuyPacket.ObjectDataBlock d in objectBuyPacket.ObjectData) { handlerObjectBuy(this, objectBuyPacket.AgentData.AgentID, objectBuyPacket.AgentData.SessionID, objectBuyPacket.AgentData.GroupID, objectBuyPacket.AgentData.CategoryID, d.ObjectLocalID, d.SaleType, d.SalePrice); } } return true; } #endregion Economy/Transaction Packets #region Script Packets private bool HandleGetScriptRunning(IClientAPI sender, Packet Pack) { GetScriptRunningPacket scriptRunning = (GetScriptRunningPacket)Pack; GetScriptRunning handlerGetScriptRunning = OnGetScriptRunning; if (handlerGetScriptRunning != null) { handlerGetScriptRunning(this, scriptRunning.Script.ObjectID, scriptRunning.Script.ItemID); } return true; } private bool HandleSetScriptRunning(IClientAPI sender, Packet Pack) { SetScriptRunningPacket setScriptRunning = (SetScriptRunningPacket)Pack; #region Packet Session and User Check if (setScriptRunning.AgentData.SessionID != SessionId || setScriptRunning.AgentData.AgentID != AgentId) return true; #endregion SetScriptRunning handlerSetScriptRunning = OnSetScriptRunning; if (handlerSetScriptRunning != null) { handlerSetScriptRunning(this, setScriptRunning.Script.ObjectID, setScriptRunning.Script.ItemID, setScriptRunning.Script.Running); } return true; } private bool HandleScriptReset(IClientAPI sender, Packet Pack) { ScriptResetPacket scriptResetPacket = (ScriptResetPacket)Pack; #region Packet Session and User Check if (scriptResetPacket.AgentData.SessionID != SessionId || scriptResetPacket.AgentData.AgentID != AgentId) return true; #endregion ScriptReset handlerScriptReset = OnScriptReset; if (handlerScriptReset != null) { handlerScriptReset(this, scriptResetPacket.Script.ObjectID, scriptResetPacket.Script.ItemID); } return true; } #endregion Script Packets #region Gesture Managment private bool HandleActivateGestures(IClientAPI sender, Packet Pack) { ActivateGesturesPacket activateGesturePacket = (ActivateGesturesPacket)Pack; #region Packet Session and User Check if (activateGesturePacket.AgentData.SessionID != SessionId || activateGesturePacket.AgentData.AgentID != AgentId) return true; #endregion ActivateGesture handlerActivateGesture = OnActivateGesture; if (handlerActivateGesture != null) { handlerActivateGesture(this, activateGesturePacket.Data[0].AssetID, activateGesturePacket.Data[0].ItemID); } else m_log.Error("Null pointer for activateGesture"); return true; } private bool HandleDeactivateGestures(IClientAPI sender, Packet Pack) { DeactivateGesturesPacket deactivateGesturePacket = (DeactivateGesturesPacket)Pack; #region Packet Session and User Check if (deactivateGesturePacket.AgentData.SessionID != SessionId || deactivateGesturePacket.AgentData.AgentID != AgentId) return true; #endregion DeactivateGesture handlerDeactivateGesture = OnDeactivateGesture; if (handlerDeactivateGesture != null) { handlerDeactivateGesture(this, deactivateGesturePacket.Data[0].ItemID); } return true; } private bool HandleObjectOwner(IClientAPI sender, Packet Pack) { ObjectOwnerPacket objectOwnerPacket = (ObjectOwnerPacket)Pack; #region Packet Session and User Check if (objectOwnerPacket.AgentData.SessionID != SessionId || objectOwnerPacket.AgentData.AgentID != AgentId) return true; #endregion List localIDs = new List(); foreach (ObjectOwnerPacket.ObjectDataBlock d in objectOwnerPacket.ObjectData) localIDs.Add(d.ObjectLocalID); ObjectOwner handlerObjectOwner = OnObjectOwner; if (handlerObjectOwner != null) { handlerObjectOwner(this, objectOwnerPacket.HeaderData.OwnerID, objectOwnerPacket.HeaderData.GroupID, localIDs); } return true; } #endregion Gesture Managment private bool HandleAgentFOV(IClientAPI sender, Packet Pack) { AgentFOVPacket fovPacket = (AgentFOVPacket)Pack; if (fovPacket.FOVBlock.GenCounter > m_agentFOVCounter) { m_agentFOVCounter = fovPacket.FOVBlock.GenCounter; AgentFOV handlerAgentFOV = OnAgentFOV; if (handlerAgentFOV != null) { handlerAgentFOV(this, fovPacket.FOVBlock.VerticalAngle); } } return true; } #region unimplemented handlers private bool HandleViewerStats(IClientAPI sender, Packet Pack) { // TODO: handle this packet //m_log.Warn("[CLIENT]: unhandled ViewerStats packet"); return true; } private bool HandleMapItemRequest(IClientAPI sender, Packet Pack) { MapItemRequestPacket mirpk = (MapItemRequestPacket)Pack; #region Packet Session and User Check if (mirpk.AgentData.SessionID != SessionId || mirpk.AgentData.AgentID != AgentId) return true; #endregion //m_log.Debug(mirpk.ToString()); MapItemRequest handlerMapItemRequest = OnMapItemRequest; if (handlerMapItemRequest != null) { try { handlerMapItemRequest(this, mirpk.AgentData.Flags, mirpk.AgentData.EstateID, mirpk.AgentData.Godlike, mirpk.RequestData.ItemType, mirpk.RequestData.RegionHandle); } catch( Exception e) { m_log.ErrorFormat("{0} HandleMapItemRequest exception: {1}", LogHeader, e.Message); } } return true; } private bool HandleTransferAbort(IClientAPI sender, Packet Pack) { return true; } private bool HandleMuteListRequest(IClientAPI sender, Packet Pack) { MuteListRequestPacket muteListRequest = (MuteListRequestPacket)Pack; #region Packet Session and User Check if (muteListRequest.AgentData.SessionID != SessionId || muteListRequest.AgentData.AgentID != AgentId) return true; #endregion MuteListRequest handlerMuteListRequest = OnMuteListRequest; if (handlerMuteListRequest != null) { handlerMuteListRequest(this, muteListRequest.MuteData.MuteCRC); } else { if(muteListRequest.MuteData.MuteCRC == 0) SendEmpytMuteList(); else SendUseCachedMuteList(); } return true; } private bool HandleUpdateMuteListEntry(IClientAPI client, Packet Packet) { UpdateMuteListEntryPacket UpdateMuteListEntry = (UpdateMuteListEntryPacket)Packet; MuteListEntryUpdate handlerUpdateMuteListEntry = OnUpdateMuteListEntry; if (handlerUpdateMuteListEntry != null) { handlerUpdateMuteListEntry(this, UpdateMuteListEntry.MuteData.MuteID, Utils.BytesToString(UpdateMuteListEntry.MuteData.MuteName), UpdateMuteListEntry.MuteData.MuteType, UpdateMuteListEntry.MuteData.MuteFlags); return true; } return false; } private bool HandleRemoveMuteListEntry(IClientAPI client, Packet Packet) { RemoveMuteListEntryPacket RemoveMuteListEntry = (RemoveMuteListEntryPacket)Packet; MuteListEntryRemove handlerRemoveMuteListEntry = OnRemoveMuteListEntry; if (handlerRemoveMuteListEntry != null) { handlerRemoveMuteListEntry(this, RemoveMuteListEntry.MuteData.MuteID, Utils.BytesToString(RemoveMuteListEntry.MuteData.MuteName)); return true; } return false; } private bool HandleUserReport(IClientAPI client, Packet Packet) { UserReportPacket UserReport = (UserReportPacket)Packet; NewUserReport handlerUserReport = OnUserReport; if (handlerUserReport != null) { handlerUserReport(this, Utils.BytesToString(UserReport.ReportData.AbuseRegionName), UserReport.ReportData.AbuserID, UserReport.ReportData.Category, UserReport.ReportData.CheckFlags, Utils.BytesToString(UserReport.ReportData.Details), UserReport.ReportData.ObjectID, UserReport.ReportData.Position, UserReport.ReportData.ReportType, UserReport.ReportData.ScreenshotID, Utils.BytesToString(UserReport.ReportData.Summary), UserReport.AgentData.AgentID); return true; } return false; } private bool HandleSendPostcard(IClientAPI client, Packet packet) { // SendPostcardPacket SendPostcard = // (SendPostcardPacket)packet; SendPostcard handlerSendPostcard = OnSendPostcard; if (handlerSendPostcard != null) { handlerSendPostcard(this); return true; } return false; } private bool HandleChangeInventoryItemFlags(IClientAPI client, Packet packet) { ChangeInventoryItemFlagsPacket ChangeInventoryItemFlags = (ChangeInventoryItemFlagsPacket)packet; ChangeInventoryItemFlags handlerChangeInventoryItemFlags = OnChangeInventoryItemFlags; if (handlerChangeInventoryItemFlags != null) { foreach(ChangeInventoryItemFlagsPacket.InventoryDataBlock b in ChangeInventoryItemFlags.InventoryData) handlerChangeInventoryItemFlags(this, b.ItemID, b.Flags); return true; } return false; } private bool HandleUseCircuitCode(IClientAPI sender, Packet Pack) { return true; } private bool HandleCreateNewOutfitAttachments(IClientAPI sender, Packet Pack) { CreateNewOutfitAttachmentsPacket packet = (CreateNewOutfitAttachmentsPacket)Pack; #region Packet Session and User Check if (packet.AgentData.SessionID != SessionId || packet.AgentData.AgentID != AgentId) return true; #endregion MoveItemsAndLeaveCopy handlerMoveItemsAndLeaveCopy = null; List items = new List(); foreach (CreateNewOutfitAttachmentsPacket.ObjectDataBlock n in packet.ObjectData) { InventoryItemBase b = new InventoryItemBase(); b.ID = n.OldItemID; b.Folder = n.OldFolderID; items.Add(b); } handlerMoveItemsAndLeaveCopy = OnMoveItemsAndLeaveCopy; if (handlerMoveItemsAndLeaveCopy != null) { handlerMoveItemsAndLeaveCopy(this, items, packet.HeaderData.NewFolderID); } return true; } private bool HandleAgentHeightWidth(IClientAPI sender, Packet Pack) { return true; } private bool HandleInventoryDescendents(IClientAPI sender, Packet Pack) { return true; } #endregion unimplemented handlers #region Dir handlers private bool HandleDirPlacesQuery(IClientAPI sender, Packet Pack) { DirPlacesQueryPacket dirPlacesQueryPacket = (DirPlacesQueryPacket)Pack; //m_log.Debug(dirPlacesQueryPacket.ToString()); #region Packet Session and User Check if (dirPlacesQueryPacket.AgentData.SessionID != SessionId || dirPlacesQueryPacket.AgentData.AgentID != AgentId) return true; #endregion DirPlacesQuery handlerDirPlacesQuery = OnDirPlacesQuery; if (handlerDirPlacesQuery != null) { handlerDirPlacesQuery(this, dirPlacesQueryPacket.QueryData.QueryID, Utils.BytesToString( dirPlacesQueryPacket.QueryData.QueryText), (int)dirPlacesQueryPacket.QueryData.QueryFlags, (int)dirPlacesQueryPacket.QueryData.Category, Utils.BytesToString( dirPlacesQueryPacket.QueryData.SimName), dirPlacesQueryPacket.QueryData.QueryStart); } return true; } private bool HandleDirFindQuery(IClientAPI sender, Packet Pack) { DirFindQueryPacket dirFindQueryPacket = (DirFindQueryPacket)Pack; #region Packet Session and User Check if (dirFindQueryPacket.AgentData.SessionID != SessionId || dirFindQueryPacket.AgentData.AgentID != AgentId) return true; #endregion DirFindQuery handlerDirFindQuery = OnDirFindQuery; if (handlerDirFindQuery != null) { handlerDirFindQuery(this, dirFindQueryPacket.QueryData.QueryID, Utils.BytesToString( dirFindQueryPacket.QueryData.QueryText).Trim(), dirFindQueryPacket.QueryData.QueryFlags, dirFindQueryPacket.QueryData.QueryStart); } return true; } private bool HandleDirLandQuery(IClientAPI sender, Packet Pack) { DirLandQueryPacket dirLandQueryPacket = (DirLandQueryPacket)Pack; #region Packet Session and User Check if (dirLandQueryPacket.AgentData.SessionID != SessionId || dirLandQueryPacket.AgentData.AgentID != AgentId) return true; #endregion DirLandQuery handlerDirLandQuery = OnDirLandQuery; if (handlerDirLandQuery != null) { handlerDirLandQuery(this, dirLandQueryPacket.QueryData.QueryID, dirLandQueryPacket.QueryData.QueryFlags, dirLandQueryPacket.QueryData.SearchType, dirLandQueryPacket.QueryData.Price, dirLandQueryPacket.QueryData.Area, dirLandQueryPacket.QueryData.QueryStart); } return true; } private bool HandleDirPopularQuery(IClientAPI sender, Packet Pack) { DirPopularQueryPacket dirPopularQueryPacket = (DirPopularQueryPacket)Pack; #region Packet Session and User Check if (dirPopularQueryPacket.AgentData.SessionID != SessionId || dirPopularQueryPacket.AgentData.AgentID != AgentId) return true; #endregion DirPopularQuery handlerDirPopularQuery = OnDirPopularQuery; if (handlerDirPopularQuery != null) { handlerDirPopularQuery(this, dirPopularQueryPacket.QueryData.QueryID, dirPopularQueryPacket.QueryData.QueryFlags); } return true; } private bool HandleDirClassifiedQuery(IClientAPI sender, Packet Pack) { DirClassifiedQueryPacket dirClassifiedQueryPacket = (DirClassifiedQueryPacket)Pack; #region Packet Session and User Check if (dirClassifiedQueryPacket.AgentData.SessionID != SessionId || dirClassifiedQueryPacket.AgentData.AgentID != AgentId) return true; #endregion DirClassifiedQuery handlerDirClassifiedQuery = OnDirClassifiedQuery; if (handlerDirClassifiedQuery != null) { handlerDirClassifiedQuery(this, dirClassifiedQueryPacket.QueryData.QueryID, Utils.BytesToString( dirClassifiedQueryPacket.QueryData.QueryText), dirClassifiedQueryPacket.QueryData.QueryFlags, dirClassifiedQueryPacket.QueryData.Category, dirClassifiedQueryPacket.QueryData.QueryStart); } return true; } private bool HandleEventInfoRequest(IClientAPI sender, Packet Pack) { EventInfoRequestPacket eventInfoRequestPacket = (EventInfoRequestPacket)Pack; #region Packet Session and User Check if (eventInfoRequestPacket.AgentData.SessionID != SessionId || eventInfoRequestPacket.AgentData.AgentID != AgentId) return true; #endregion if (OnEventInfoRequest != null) { OnEventInfoRequest(this, eventInfoRequestPacket.EventData.EventID); } return true; } #endregion #region Calling Card private bool HandleOfferCallingCard(IClientAPI sender, Packet Pack) { OfferCallingCardPacket offerCallingCardPacket = (OfferCallingCardPacket)Pack; #region Packet Session and User Check if (offerCallingCardPacket.AgentData.SessionID != SessionId || offerCallingCardPacket.AgentData.AgentID != AgentId) return true; #endregion if (OnOfferCallingCard != null) { OnOfferCallingCard(this, offerCallingCardPacket.AgentBlock.DestID, offerCallingCardPacket.AgentBlock.TransactionID); } return true; } private bool HandleAcceptCallingCard(IClientAPI sender, Packet Pack) { AcceptCallingCardPacket acceptCallingCardPacket = (AcceptCallingCardPacket)Pack; #region Packet Session and User Check if (acceptCallingCardPacket.AgentData.SessionID != SessionId || acceptCallingCardPacket.AgentData.AgentID != AgentId) return true; #endregion // according to http://wiki.secondlife.com/wiki/AcceptCallingCard FolderData should // contain exactly one entry if (OnAcceptCallingCard != null && acceptCallingCardPacket.FolderData.Length > 0) { OnAcceptCallingCard(this, acceptCallingCardPacket.TransactionBlock.TransactionID, acceptCallingCardPacket.FolderData[0].FolderID); } return true; } private bool HandleDeclineCallingCard(IClientAPI sender, Packet Pack) { DeclineCallingCardPacket declineCallingCardPacket = (DeclineCallingCardPacket)Pack; #region Packet Session and User Check if (declineCallingCardPacket.AgentData.SessionID != SessionId || declineCallingCardPacket.AgentData.AgentID != AgentId) return true; #endregion if (OnDeclineCallingCard != null) { OnDeclineCallingCard(this, declineCallingCardPacket.TransactionBlock.TransactionID); } return true; } #endregion Calling Card #region Groups private bool HandleActivateGroup(IClientAPI sender, Packet Pack) { ActivateGroupPacket activateGroupPacket = (ActivateGroupPacket)Pack; #region Packet Session and User Check if (activateGroupPacket.AgentData.SessionID != SessionId || activateGroupPacket.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.ActivateGroup(this, activateGroupPacket.AgentData.GroupID); } return true; } private bool HandleGroupVoteHistoryRequest(IClientAPI client, Packet Packet) { GroupVoteHistoryRequestPacket GroupVoteHistoryRequest = (GroupVoteHistoryRequestPacket)Packet; GroupVoteHistoryRequest handlerGroupVoteHistoryRequest = OnGroupVoteHistoryRequest; if (handlerGroupVoteHistoryRequest != null) { handlerGroupVoteHistoryRequest(this, GroupVoteHistoryRequest.AgentData.AgentID,GroupVoteHistoryRequest.AgentData.SessionID,GroupVoteHistoryRequest.GroupData.GroupID,GroupVoteHistoryRequest.TransactionData.TransactionID); return true; } return false; } private bool HandleGroupActiveProposalsRequest(IClientAPI client, Packet Packet) { GroupActiveProposalsRequestPacket GroupActiveProposalsRequest = (GroupActiveProposalsRequestPacket)Packet; GroupActiveProposalsRequest handlerGroupActiveProposalsRequest = OnGroupActiveProposalsRequest; if (handlerGroupActiveProposalsRequest != null) { handlerGroupActiveProposalsRequest(this, GroupActiveProposalsRequest.AgentData.AgentID,GroupActiveProposalsRequest.AgentData.SessionID,GroupActiveProposalsRequest.GroupData.GroupID,GroupActiveProposalsRequest.TransactionData.TransactionID); return true; } return false; } private bool HandleGroupAccountDetailsRequest(IClientAPI client, Packet Packet) { GroupAccountDetailsRequestPacket GroupAccountDetailsRequest = (GroupAccountDetailsRequestPacket)Packet; GroupAccountDetailsRequest handlerGroupAccountDetailsRequest = OnGroupAccountDetailsRequest; if (handlerGroupAccountDetailsRequest != null) { handlerGroupAccountDetailsRequest(this, GroupAccountDetailsRequest.AgentData.AgentID,GroupAccountDetailsRequest.AgentData.GroupID,GroupAccountDetailsRequest.MoneyData.RequestID,GroupAccountDetailsRequest.AgentData.SessionID); return true; } return false; } private bool HandleGroupAccountSummaryRequest(IClientAPI client, Packet Packet) { GroupAccountSummaryRequestPacket GroupAccountSummaryRequest = (GroupAccountSummaryRequestPacket)Packet; GroupAccountSummaryRequest handlerGroupAccountSummaryRequest = OnGroupAccountSummaryRequest; if (handlerGroupAccountSummaryRequest != null) { handlerGroupAccountSummaryRequest(this, GroupAccountSummaryRequest.AgentData.AgentID,GroupAccountSummaryRequest.AgentData.GroupID); return true; } return false; } private bool HandleGroupTransactionsDetailsRequest(IClientAPI client, Packet Packet) { GroupAccountTransactionsRequestPacket GroupAccountTransactionsRequest = (GroupAccountTransactionsRequestPacket)Packet; GroupAccountTransactionsRequest handlerGroupAccountTransactionsRequest = OnGroupAccountTransactionsRequest; if (handlerGroupAccountTransactionsRequest != null) { handlerGroupAccountTransactionsRequest(this, GroupAccountTransactionsRequest.AgentData.AgentID,GroupAccountTransactionsRequest.AgentData.GroupID,GroupAccountTransactionsRequest.MoneyData.RequestID,GroupAccountTransactionsRequest.AgentData.SessionID); return true; } return false; } private bool HandleGroupTitlesRequest(IClientAPI sender, Packet Pack) { GroupTitlesRequestPacket groupTitlesRequest = (GroupTitlesRequestPacket)Pack; #region Packet Session and User Check if (groupTitlesRequest.AgentData.SessionID != SessionId || groupTitlesRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { GroupTitlesReplyPacket groupTitlesReply = (GroupTitlesReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupTitlesReply); groupTitlesReply.AgentData = new GroupTitlesReplyPacket.AgentDataBlock(); groupTitlesReply.AgentData.AgentID = AgentId; groupTitlesReply.AgentData.GroupID = groupTitlesRequest.AgentData.GroupID; groupTitlesReply.AgentData.RequestID = groupTitlesRequest.AgentData.RequestID; List titles = m_GroupsModule.GroupTitlesRequest(this, groupTitlesRequest.AgentData.GroupID); groupTitlesReply.GroupData = new GroupTitlesReplyPacket.GroupDataBlock[titles.Count]; int i = 0; foreach (GroupTitlesData d in titles) { groupTitlesReply.GroupData[i] = new GroupTitlesReplyPacket.GroupDataBlock(); groupTitlesReply.GroupData[i].Title = Util.StringToBytes256(d.Name); groupTitlesReply.GroupData[i].RoleID = d.UUID; groupTitlesReply.GroupData[i].Selected = d.Selected; i++; } OutPacket(groupTitlesReply, ThrottleOutPacketType.Task); } return true; } UUID lastGroupProfileRequestID = UUID.Zero; double lastGroupProfileRequestTS = Util.GetTimeStampMS(); private bool HandleGroupProfileRequest(IClientAPI sender, Packet Pack) { if(m_GroupsModule == null) return true; GroupProfileRequestPacket groupProfileRequest = (GroupProfileRequestPacket)Pack; #region Packet Session and User Check if (groupProfileRequest.AgentData.SessionID != SessionId || groupProfileRequest.AgentData.AgentID != AgentId) return true; #endregion UUID grpID = groupProfileRequest.GroupData.GroupID; double ts = Util.GetTimeStampMS(); if(grpID == lastGroupProfileRequestID && ts - lastGroupProfileRequestTS < 10000) return true; lastGroupProfileRequestID = grpID; lastGroupProfileRequestTS = ts; GroupProfileReplyPacket groupProfileReply = (GroupProfileReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupProfileReply); groupProfileReply.AgentData = new GroupProfileReplyPacket.AgentDataBlock(); groupProfileReply.GroupData = new GroupProfileReplyPacket.GroupDataBlock(); groupProfileReply.AgentData.AgentID = AgentId; GroupProfileData d = m_GroupsModule.GroupProfileRequest(this, groupProfileRequest.GroupData.GroupID); if(d.GroupID == UUID.Zero) // don't send broken data return true; groupProfileReply.GroupData.GroupID = d.GroupID; groupProfileReply.GroupData.Name = Util.StringToBytes256(d.Name); groupProfileReply.GroupData.Charter = Util.StringToBytes1024(d.Charter); groupProfileReply.GroupData.ShowInList = d.ShowInList; groupProfileReply.GroupData.MemberTitle = Util.StringToBytes256(d.MemberTitle); groupProfileReply.GroupData.PowersMask = d.PowersMask; groupProfileReply.GroupData.InsigniaID = d.InsigniaID; groupProfileReply.GroupData.FounderID = d.FounderID; groupProfileReply.GroupData.MembershipFee = d.MembershipFee; groupProfileReply.GroupData.OpenEnrollment = d.OpenEnrollment; groupProfileReply.GroupData.Money = d.Money; groupProfileReply.GroupData.GroupMembershipCount = d.GroupMembershipCount; groupProfileReply.GroupData.GroupRolesCount = d.GroupRolesCount; groupProfileReply.GroupData.AllowPublish = d.AllowPublish; groupProfileReply.GroupData.MaturePublish = d.MaturePublish; groupProfileReply.GroupData.OwnerRole = d.OwnerRole; Scene scene = (Scene)m_scene; if (scene.Permissions.IsGod(sender.AgentId) && (!sender.IsGroupMember(groupProfileRequest.GroupData.GroupID))) { ScenePresence p; if (scene.TryGetScenePresence(sender.AgentId, out p)) { if (p.IsViewerUIGod) { groupProfileReply.GroupData.OpenEnrollment = true; groupProfileReply.GroupData.MembershipFee = 0; } } } OutPacket(groupProfileReply, ThrottleOutPacketType.Task); if(grpID == lastGroupProfileRequestID) lastGroupProfileRequestTS = Util.GetTimeStampMS() - 7000; return true; } private bool HandleGroupMembersRequest(IClientAPI sender, Packet Pack) { GroupMembersRequestPacket groupMembersRequestPacket = (GroupMembersRequestPacket)Pack; #region Packet Session and User Check if (groupMembersRequestPacket.AgentData.SessionID != SessionId || groupMembersRequestPacket.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { List members = m_GroupsModule.GroupMembersRequest(this, groupMembersRequestPacket.GroupData.GroupID); int memberCount = members.Count; int indx = 0; while (indx < memberCount) { int blockCount = memberCount - indx; if (blockCount > 25) blockCount = 25; GroupMembersReplyPacket groupMembersReply = (GroupMembersReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupMembersReply); groupMembersReply.AgentData = new GroupMembersReplyPacket.AgentDataBlock(); groupMembersReply.GroupData = new GroupMembersReplyPacket.GroupDataBlock(); groupMembersReply.MemberData = new GroupMembersReplyPacket.MemberDataBlock[ blockCount]; groupMembersReply.AgentData.AgentID = AgentId; groupMembersReply.GroupData.GroupID = groupMembersRequestPacket.GroupData.GroupID; groupMembersReply.GroupData.RequestID = groupMembersRequestPacket.GroupData.RequestID; groupMembersReply.GroupData.MemberCount = memberCount; for (int i = 0; i < blockCount; i++) { GroupMembersData m = members[indx++]; groupMembersReply.MemberData[i] = new GroupMembersReplyPacket.MemberDataBlock(); groupMembersReply.MemberData[i].AgentID = m.AgentID; groupMembersReply.MemberData[i].Contribution = m.Contribution; groupMembersReply.MemberData[i].OnlineStatus = Util.StringToBytes256(m.OnlineStatus); groupMembersReply.MemberData[i].AgentPowers = m.AgentPowers; groupMembersReply.MemberData[i].Title = Util.StringToBytes256(m.Title); groupMembersReply.MemberData[i].IsOwner = m.IsOwner; } OutPacket(groupMembersReply, ThrottleOutPacketType.Task); } } return true; } private bool HandleGroupRoleDataRequest(IClientAPI sender, Packet Pack) { GroupRoleDataRequestPacket groupRolesRequest = (GroupRoleDataRequestPacket)Pack; #region Packet Session and User Check if (groupRolesRequest.AgentData.SessionID != SessionId || groupRolesRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { GroupRoleDataReplyPacket groupRolesReply = (GroupRoleDataReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupRoleDataReply); groupRolesReply.AgentData = new GroupRoleDataReplyPacket.AgentDataBlock(); groupRolesReply.AgentData.AgentID = AgentId; groupRolesReply.GroupData = new GroupRoleDataReplyPacket.GroupDataBlock(); groupRolesReply.GroupData.GroupID = groupRolesRequest.GroupData.GroupID; groupRolesReply.GroupData.RequestID = groupRolesRequest.GroupData.RequestID; List titles = m_GroupsModule.GroupRoleDataRequest(this, groupRolesRequest.GroupData.GroupID); groupRolesReply.GroupData.RoleCount = titles.Count; groupRolesReply.RoleData = new GroupRoleDataReplyPacket.RoleDataBlock[titles.Count]; int i = 0; foreach (GroupRolesData d in titles) { groupRolesReply.RoleData[i] = new GroupRoleDataReplyPacket.RoleDataBlock(); groupRolesReply.RoleData[i].RoleID = d.RoleID; groupRolesReply.RoleData[i].Name = Util.StringToBytes256(d.Name); groupRolesReply.RoleData[i].Title = Util.StringToBytes256(d.Title); groupRolesReply.RoleData[i].Description = Util.StringToBytes1024(d.Description); groupRolesReply.RoleData[i].Powers = d.Powers; groupRolesReply.RoleData[i].Members = (uint)d.Members; i++; } OutPacket(groupRolesReply, ThrottleOutPacketType.Task); } return true; } private bool HandleGroupRoleMembersRequest(IClientAPI sender, Packet Pack) { GroupRoleMembersRequestPacket groupRoleMembersRequest = (GroupRoleMembersRequestPacket)Pack; #region Packet Session and User Check if (groupRoleMembersRequest.AgentData.SessionID != SessionId || groupRoleMembersRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { List mappings = m_GroupsModule.GroupRoleMembersRequest(this, groupRoleMembersRequest.GroupData.GroupID); int mappingsCount = mappings.Count; while (mappings.Count > 0) { int pairs = mappings.Count; if (pairs > 32) pairs = 32; GroupRoleMembersReplyPacket groupRoleMembersReply = (GroupRoleMembersReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupRoleMembersReply); groupRoleMembersReply.AgentData = new GroupRoleMembersReplyPacket.AgentDataBlock(); groupRoleMembersReply.AgentData.AgentID = AgentId; groupRoleMembersReply.AgentData.GroupID = groupRoleMembersRequest.GroupData.GroupID; groupRoleMembersReply.AgentData.RequestID = groupRoleMembersRequest.GroupData.RequestID; groupRoleMembersReply.AgentData.TotalPairs = (uint)mappingsCount; groupRoleMembersReply.MemberData = new GroupRoleMembersReplyPacket.MemberDataBlock[pairs]; for (int i = 0; i < pairs; i++) { GroupRoleMembersData d = mappings[0]; mappings.RemoveAt(0); groupRoleMembersReply.MemberData[i] = new GroupRoleMembersReplyPacket.MemberDataBlock(); groupRoleMembersReply.MemberData[i].RoleID = d.RoleID; groupRoleMembersReply.MemberData[i].MemberID = d.MemberID; } OutPacket(groupRoleMembersReply, ThrottleOutPacketType.Task); } } return true; } private bool HandleCreateGroupRequest(IClientAPI sender, Packet Pack) { CreateGroupRequestPacket createGroupRequest = (CreateGroupRequestPacket)Pack; #region Packet Session and User Check if (createGroupRequest.AgentData.SessionID != SessionId || createGroupRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.CreateGroup(this, Utils.BytesToString(createGroupRequest.GroupData.Name), Utils.BytesToString(createGroupRequest.GroupData.Charter), createGroupRequest.GroupData.ShowInList, createGroupRequest.GroupData.InsigniaID, createGroupRequest.GroupData.MembershipFee, createGroupRequest.GroupData.OpenEnrollment, createGroupRequest.GroupData.AllowPublish, createGroupRequest.GroupData.MaturePublish); } return true; } private bool HandleUpdateGroupInfo(IClientAPI sender, Packet Pack) { UpdateGroupInfoPacket updateGroupInfo = (UpdateGroupInfoPacket)Pack; #region Packet Session and User Check if (updateGroupInfo.AgentData.SessionID != SessionId || updateGroupInfo.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.UpdateGroupInfo(this, updateGroupInfo.GroupData.GroupID, Utils.BytesToString(updateGroupInfo.GroupData.Charter), updateGroupInfo.GroupData.ShowInList, updateGroupInfo.GroupData.InsigniaID, updateGroupInfo.GroupData.MembershipFee, updateGroupInfo.GroupData.OpenEnrollment, updateGroupInfo.GroupData.AllowPublish, updateGroupInfo.GroupData.MaturePublish); } return true; } private bool HandleSetGroupAcceptNotices(IClientAPI sender, Packet Pack) { SetGroupAcceptNoticesPacket setGroupAcceptNotices = (SetGroupAcceptNoticesPacket)Pack; #region Packet Session and User Check if (setGroupAcceptNotices.AgentData.SessionID != SessionId || setGroupAcceptNotices.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.SetGroupAcceptNotices(this, setGroupAcceptNotices.Data.GroupID, setGroupAcceptNotices.Data.AcceptNotices, setGroupAcceptNotices.NewData.ListInProfile); } return true; } private bool HandleGroupTitleUpdate(IClientAPI sender, Packet Pack) { GroupTitleUpdatePacket groupTitleUpdate = (GroupTitleUpdatePacket)Pack; #region Packet Session and User Check if (groupTitleUpdate.AgentData.SessionID != SessionId || groupTitleUpdate.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.GroupTitleUpdate(this, groupTitleUpdate.AgentData.GroupID, groupTitleUpdate.AgentData.TitleRoleID); } return true; } private bool HandleParcelDeedToGroup(IClientAPI sender, Packet Pack) { ParcelDeedToGroupPacket parcelDeedToGroup = (ParcelDeedToGroupPacket)Pack; if (m_GroupsModule != null) { ParcelDeedToGroup handlerParcelDeedToGroup = OnParcelDeedToGroup; if (handlerParcelDeedToGroup != null) { handlerParcelDeedToGroup(parcelDeedToGroup.Data.LocalID, parcelDeedToGroup.Data.GroupID, this); } } return true; } private bool HandleGroupNoticesListRequest(IClientAPI sender, Packet Pack) { GroupNoticesListRequestPacket groupNoticesListRequest = (GroupNoticesListRequestPacket)Pack; #region Packet Session and User Check if (groupNoticesListRequest.AgentData.SessionID != SessionId || groupNoticesListRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { GroupNoticeData[] gn = m_GroupsModule.GroupNoticesListRequest(this, groupNoticesListRequest.Data.GroupID); GroupNoticesListReplyPacket groupNoticesListReply = (GroupNoticesListReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupNoticesListReply); groupNoticesListReply.AgentData = new GroupNoticesListReplyPacket.AgentDataBlock(); groupNoticesListReply.AgentData.AgentID = AgentId; groupNoticesListReply.AgentData.GroupID = groupNoticesListRequest.Data.GroupID; groupNoticesListReply.Data = new GroupNoticesListReplyPacket.DataBlock[gn.Length]; int i = 0; foreach (GroupNoticeData g in gn) { groupNoticesListReply.Data[i] = new GroupNoticesListReplyPacket.DataBlock(); groupNoticesListReply.Data[i].NoticeID = g.NoticeID; groupNoticesListReply.Data[i].Timestamp = g.Timestamp; groupNoticesListReply.Data[i].FromName = Util.StringToBytes256(g.FromName); groupNoticesListReply.Data[i].Subject = Util.StringToBytes256(g.Subject); groupNoticesListReply.Data[i].HasAttachment = g.HasAttachment; groupNoticesListReply.Data[i].AssetType = g.AssetType; i++; } OutPacket(groupNoticesListReply, ThrottleOutPacketType.Task); } return true; } private bool HandleGroupNoticeRequest(IClientAPI sender, Packet Pack) { GroupNoticeRequestPacket groupNoticeRequest = (GroupNoticeRequestPacket)Pack; #region Packet Session and User Check if (groupNoticeRequest.AgentData.SessionID != SessionId || groupNoticeRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.GroupNoticeRequest(this, groupNoticeRequest.Data.GroupNoticeID); } return true; } private bool HandleGroupRoleUpdate(IClientAPI sender, Packet Pack) { GroupRoleUpdatePacket groupRoleUpdate = (GroupRoleUpdatePacket)Pack; #region Packet Session and User Check if (groupRoleUpdate.AgentData.SessionID != SessionId || groupRoleUpdate.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { foreach (GroupRoleUpdatePacket.RoleDataBlock d in groupRoleUpdate.RoleData) { m_GroupsModule.GroupRoleUpdate(this, groupRoleUpdate.AgentData.GroupID, d.RoleID, Utils.BytesToString(d.Name), Utils.BytesToString(d.Description), Utils.BytesToString(d.Title), d.Powers, d.UpdateType); } m_GroupsModule.NotifyChange(groupRoleUpdate.AgentData.GroupID); } return true; } private bool HandleGroupRoleChanges(IClientAPI sender, Packet Pack) { GroupRoleChangesPacket groupRoleChanges = (GroupRoleChangesPacket)Pack; #region Packet Session and User Check if (groupRoleChanges.AgentData.SessionID != SessionId || groupRoleChanges.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { foreach (GroupRoleChangesPacket.RoleChangeBlock d in groupRoleChanges.RoleChange) { m_GroupsModule.GroupRoleChanges(this, groupRoleChanges.AgentData.GroupID, d.RoleID, d.MemberID, d.Change); } m_GroupsModule.NotifyChange(groupRoleChanges.AgentData.GroupID); } return true; } private bool HandleJoinGroupRequest(IClientAPI sender, Packet Pack) { JoinGroupRequestPacket joinGroupRequest = (JoinGroupRequestPacket)Pack; #region Packet Session and User Check if (joinGroupRequest.AgentData.SessionID != SessionId || joinGroupRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.JoinGroupRequest(this, joinGroupRequest.GroupData.GroupID); } return true; } private bool HandleLeaveGroupRequest(IClientAPI sender, Packet Pack) { LeaveGroupRequestPacket leaveGroupRequest = (LeaveGroupRequestPacket)Pack; #region Packet Session and User Check if (leaveGroupRequest.AgentData.SessionID != SessionId || leaveGroupRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { m_GroupsModule.LeaveGroupRequest(this, leaveGroupRequest.GroupData.GroupID); } return true; } private bool HandleEjectGroupMemberRequest(IClientAPI sender, Packet Pack) { EjectGroupMemberRequestPacket ejectGroupMemberRequest = (EjectGroupMemberRequestPacket)Pack; #region Packet Session and User Check if (ejectGroupMemberRequest.AgentData.SessionID != SessionId || ejectGroupMemberRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { foreach (EjectGroupMemberRequestPacket.EjectDataBlock e in ejectGroupMemberRequest.EjectData) { m_GroupsModule.EjectGroupMemberRequest(this, ejectGroupMemberRequest.GroupData.GroupID, e.EjecteeID); } } return true; } private bool HandleInviteGroupRequest(IClientAPI sender, Packet Pack) { InviteGroupRequestPacket inviteGroupRequest = (InviteGroupRequestPacket)Pack; #region Packet Session and User Check if (inviteGroupRequest.AgentData.SessionID != SessionId || inviteGroupRequest.AgentData.AgentID != AgentId) return true; #endregion if (m_GroupsModule != null) { foreach (InviteGroupRequestPacket.InviteDataBlock b in inviteGroupRequest.InviteData) { m_GroupsModule.InviteGroupRequest(this, inviteGroupRequest.GroupData.GroupID, b.InviteeID, b.RoleID); } } return true; } #endregion Groups private bool HandleStartLure(IClientAPI sender, Packet Pack) { StartLurePacket startLureRequest = (StartLurePacket)Pack; #region Packet Session and User Check if (startLureRequest.AgentData.SessionID != SessionId || startLureRequest.AgentData.AgentID != AgentId) return true; #endregion StartLure handlerStartLure = OnStartLure; if (handlerStartLure != null) { for (int i = 0 ; i < startLureRequest.TargetData.Length ; i++) { handlerStartLure(startLureRequest.Info.LureType, Utils.BytesToString( startLureRequest.Info.Message), startLureRequest.TargetData[i].TargetID, this); } } return true; } private bool HandleTeleportLureRequest(IClientAPI sender, Packet Pack) { TeleportLureRequestPacket teleportLureRequest = (TeleportLureRequestPacket)Pack; #region Packet Session and User Check if (teleportLureRequest.Info.SessionID != SessionId || teleportLureRequest.Info.AgentID != AgentId) return true; #endregion TeleportLureRequest handlerTeleportLureRequest = OnTeleportLureRequest; if (handlerTeleportLureRequest != null) handlerTeleportLureRequest( teleportLureRequest.Info.LureID, teleportLureRequest.Info.TeleportFlags, this); return true; } private bool HandleClassifiedInfoRequest(IClientAPI sender, Packet Pack) { ClassifiedInfoRequestPacket classifiedInfoRequest = (ClassifiedInfoRequestPacket)Pack; #region Packet Session and User Check if (classifiedInfoRequest.AgentData.SessionID != SessionId || classifiedInfoRequest.AgentData.AgentID != AgentId) return true; #endregion ClassifiedInfoRequest handlerClassifiedInfoRequest = OnClassifiedInfoRequest; if (handlerClassifiedInfoRequest != null) handlerClassifiedInfoRequest( classifiedInfoRequest.Data.ClassifiedID, this); return true; } private bool HandleClassifiedInfoUpdate(IClientAPI sender, Packet Pack) { ClassifiedInfoUpdatePacket classifiedInfoUpdate = (ClassifiedInfoUpdatePacket)Pack; #region Packet Session and User Check if (classifiedInfoUpdate.AgentData.SessionID != SessionId || classifiedInfoUpdate.AgentData.AgentID != AgentId) return true; #endregion ClassifiedInfoUpdate handlerClassifiedInfoUpdate = OnClassifiedInfoUpdate; if (handlerClassifiedInfoUpdate != null) handlerClassifiedInfoUpdate( classifiedInfoUpdate.Data.ClassifiedID, classifiedInfoUpdate.Data.Category, Utils.BytesToString( classifiedInfoUpdate.Data.Name), Utils.BytesToString( classifiedInfoUpdate.Data.Desc), classifiedInfoUpdate.Data.ParcelID, classifiedInfoUpdate.Data.ParentEstate, classifiedInfoUpdate.Data.SnapshotID, new Vector3( classifiedInfoUpdate.Data.PosGlobal), classifiedInfoUpdate.Data.ClassifiedFlags, classifiedInfoUpdate.Data.PriceForListing, this); return true; } private bool HandleClassifiedDelete(IClientAPI sender, Packet Pack) { ClassifiedDeletePacket classifiedDelete = (ClassifiedDeletePacket)Pack; #region Packet Session and User Check if (classifiedDelete.AgentData.SessionID != SessionId || classifiedDelete.AgentData.AgentID != AgentId) return true; #endregion ClassifiedDelete handlerClassifiedDelete = OnClassifiedDelete; if (handlerClassifiedDelete != null) handlerClassifiedDelete( classifiedDelete.Data.ClassifiedID, this); return true; } private bool HandleClassifiedGodDelete(IClientAPI sender, Packet Pack) { ClassifiedGodDeletePacket classifiedGodDelete = (ClassifiedGodDeletePacket)Pack; #region Packet Session and User Check if (classifiedGodDelete.AgentData.SessionID != SessionId || classifiedGodDelete.AgentData.AgentID != AgentId) return true; #endregion ClassifiedGodDelete handlerClassifiedGodDelete = OnClassifiedGodDelete; if (handlerClassifiedGodDelete != null) handlerClassifiedGodDelete( classifiedGodDelete.Data.ClassifiedID, classifiedGodDelete.Data.QueryID, this); return true; } private bool HandleEventGodDelete(IClientAPI sender, Packet Pack) { EventGodDeletePacket eventGodDelete = (EventGodDeletePacket)Pack; #region Packet Session and User Check if (eventGodDelete.AgentData.SessionID != SessionId || eventGodDelete.AgentData.AgentID != AgentId) return true; #endregion EventGodDelete handlerEventGodDelete = OnEventGodDelete; if (handlerEventGodDelete != null) handlerEventGodDelete( eventGodDelete.EventData.EventID, eventGodDelete.QueryData.QueryID, Utils.BytesToString( eventGodDelete.QueryData.QueryText), eventGodDelete.QueryData.QueryFlags, eventGodDelete.QueryData.QueryStart, this); return true; } private bool HandleEventNotificationAddRequest(IClientAPI sender, Packet Pack) { EventNotificationAddRequestPacket eventNotificationAdd = (EventNotificationAddRequestPacket)Pack; #region Packet Session and User Check if (eventNotificationAdd.AgentData.SessionID != SessionId || eventNotificationAdd.AgentData.AgentID != AgentId) return true; #endregion EventNotificationAddRequest handlerEventNotificationAddRequest = OnEventNotificationAddRequest; if (handlerEventNotificationAddRequest != null) handlerEventNotificationAddRequest( eventNotificationAdd.EventData.EventID, this); return true; } private bool HandleEventNotificationRemoveRequest(IClientAPI sender, Packet Pack) { EventNotificationRemoveRequestPacket eventNotificationRemove = (EventNotificationRemoveRequestPacket)Pack; #region Packet Session and User Check if (eventNotificationRemove.AgentData.SessionID != SessionId || eventNotificationRemove.AgentData.AgentID != AgentId) return true; #endregion EventNotificationRemoveRequest handlerEventNotificationRemoveRequest = OnEventNotificationRemoveRequest; if (handlerEventNotificationRemoveRequest != null) handlerEventNotificationRemoveRequest( eventNotificationRemove.EventData.EventID, this); return true; } private bool HandleRetrieveInstantMessages(IClientAPI sender, Packet Pack) { RetrieveInstantMessagesPacket rimpInstantMessagePack = (RetrieveInstantMessagesPacket)Pack; #region Packet Session and User Check if (rimpInstantMessagePack.AgentData.SessionID != SessionId || rimpInstantMessagePack.AgentData.AgentID != AgentId) return true; #endregion RetrieveInstantMessages handlerRetrieveInstantMessages = OnRetrieveInstantMessages; if (handlerRetrieveInstantMessages != null) handlerRetrieveInstantMessages(this); return true; } private bool HandlePickDelete(IClientAPI sender, Packet Pack) { PickDeletePacket pickDelete = (PickDeletePacket)Pack; #region Packet Session and User Check if (pickDelete.AgentData.SessionID != SessionId || pickDelete.AgentData.AgentID != AgentId) return true; #endregion PickDelete handlerPickDelete = OnPickDelete; if (handlerPickDelete != null) handlerPickDelete(this, pickDelete.Data.PickID); return true; } private bool HandlePickGodDelete(IClientAPI sender, Packet Pack) { PickGodDeletePacket pickGodDelete = (PickGodDeletePacket)Pack; #region Packet Session and User Check if (pickGodDelete.AgentData.SessionID != SessionId || pickGodDelete.AgentData.AgentID != AgentId) return true; #endregion PickGodDelete handlerPickGodDelete = OnPickGodDelete; if (handlerPickGodDelete != null) handlerPickGodDelete(this, pickGodDelete.AgentData.AgentID, pickGodDelete.Data.PickID, pickGodDelete.Data.QueryID); return true; } private bool HandlePickInfoUpdate(IClientAPI sender, Packet Pack) { PickInfoUpdatePacket pickInfoUpdate = (PickInfoUpdatePacket)Pack; #region Packet Session and User Check if (pickInfoUpdate.AgentData.SessionID != SessionId || pickInfoUpdate.AgentData.AgentID != AgentId) return true; #endregion PickInfoUpdate handlerPickInfoUpdate = OnPickInfoUpdate; if (handlerPickInfoUpdate != null) handlerPickInfoUpdate(this, pickInfoUpdate.Data.PickID, pickInfoUpdate.Data.CreatorID, pickInfoUpdate.Data.TopPick, Utils.BytesToString(pickInfoUpdate.Data.Name), Utils.BytesToString(pickInfoUpdate.Data.Desc), pickInfoUpdate.Data.SnapshotID, pickInfoUpdate.Data.SortOrder, pickInfoUpdate.Data.Enabled); return true; } private bool HandleAvatarNotesUpdate(IClientAPI sender, Packet Pack) { AvatarNotesUpdatePacket avatarNotesUpdate = (AvatarNotesUpdatePacket)Pack; #region Packet Session and User Check if (avatarNotesUpdate.AgentData.SessionID != SessionId || avatarNotesUpdate.AgentData.AgentID != AgentId) return true; #endregion AvatarNotesUpdate handlerAvatarNotesUpdate = OnAvatarNotesUpdate; if (handlerAvatarNotesUpdate != null) handlerAvatarNotesUpdate(this, avatarNotesUpdate.Data.TargetID, Utils.BytesToString(avatarNotesUpdate.Data.Notes)); return true; } private bool HandleAvatarInterestsUpdate(IClientAPI sender, Packet Pack) { AvatarInterestsUpdatePacket avatarInterestUpdate = (AvatarInterestsUpdatePacket)Pack; #region Packet Session and User Check if (avatarInterestUpdate.AgentData.SessionID != SessionId || avatarInterestUpdate.AgentData.AgentID != AgentId) return true; #endregion AvatarInterestUpdate handlerAvatarInterestUpdate = OnAvatarInterestUpdate; if (handlerAvatarInterestUpdate != null) handlerAvatarInterestUpdate(this, avatarInterestUpdate.PropertiesData.WantToMask, Utils.BytesToString(avatarInterestUpdate.PropertiesData.WantToText), avatarInterestUpdate.PropertiesData.SkillsMask, Utils.BytesToString(avatarInterestUpdate.PropertiesData.SkillsText), Utils.BytesToString(avatarInterestUpdate.PropertiesData.LanguagesText)); return true; } private bool HandleGrantUserRights(IClientAPI sender, Packet Pack) { GrantUserRightsPacket GrantUserRights = (GrantUserRightsPacket)Pack; #region Packet Session and User Check if (GrantUserRights.AgentData.SessionID != SessionId || GrantUserRights.AgentData.AgentID != AgentId) return true; #endregion GrantUserFriendRights GrantUserRightsHandler = OnGrantUserRights; if (GrantUserRightsHandler != null) GrantUserRightsHandler(this, GrantUserRights.Rights[0].AgentRelated, GrantUserRights.Rights[0].RelatedRights); return true; } private bool HandleRevokePermissions(IClientAPI sender, Packet Pack) { RevokePermissionsPacket pkt = (RevokePermissionsPacket)Pack; if (pkt.AgentData.SessionID != SessionId || pkt .AgentData.AgentID != AgentId) return true; // don't use multidelegate "event" ScenePresence sp = (ScenePresence)SceneAgent; if(sp != null && !sp.IsDeleted && !sp.IsInTransit) { UUID objectID = pkt.Data.ObjectID; uint permissions = pkt.Data.ObjectPermissions; sp.HandleRevokePermissions(objectID , permissions); } return true; } private bool HandlePlacesQuery(IClientAPI sender, Packet Pack) { PlacesQueryPacket placesQueryPacket = (PlacesQueryPacket)Pack; PlacesQuery handlerPlacesQuery = OnPlacesQuery; if (handlerPlacesQuery != null) handlerPlacesQuery(placesQueryPacket.AgentData.QueryID, placesQueryPacket.TransactionData.TransactionID, Utils.BytesToString( placesQueryPacket.QueryData.QueryText), placesQueryPacket.QueryData.QueryFlags, (byte)placesQueryPacket.QueryData.Category, Utils.BytesToString( placesQueryPacket.QueryData.SimName), this); return true; } #endregion Packet Handlers public void SendScriptQuestion(UUID taskID, string taskName, string ownerName, UUID itemID, int question) { ScriptQuestionPacket scriptQuestion = (ScriptQuestionPacket)PacketPool.Instance.GetPacket(PacketType.ScriptQuestion); scriptQuestion.Data = new ScriptQuestionPacket.DataBlock(); // TODO: don't create new blocks if recycling an old packet scriptQuestion.Data.TaskID = taskID; scriptQuestion.Data.ItemID = itemID; scriptQuestion.Data.Questions = question; scriptQuestion.Data.ObjectName = Util.StringToBytes256(taskName); scriptQuestion.Data.ObjectOwner = Util.StringToBytes256(ownerName); OutPacket(scriptQuestion, ThrottleOutPacketType.Task); } /// /// Handler called when we receive a logout packet. /// /// /// /// protected virtual bool HandleLogout(IClientAPI client, Packet packet) { if (packet.Type == PacketType.LogoutRequest) { if (((LogoutRequestPacket)packet).AgentData.SessionID != SessionId) return false; } return Logout(client); } /// /// /// /// /// protected virtual bool Logout(IClientAPI client) { m_log.InfoFormat("[CLIENT]: Got a logout request for {0} in {1}", Name, Scene.RegionInfo.RegionName); Action handlerLogout = OnLogout; if (handlerLogout != null) { handlerLogout(client); } return true; } /// /// /// /// At the moment, we always reply that there is no cached texture. /// /// /// /// protected bool HandleAgentTextureCached(IClientAPI simclient, Packet packet) { //m_log.Debug("texture cached: " + packet.ToString()); AgentCachedTexturePacket cachedtex = (AgentCachedTexturePacket)packet; AgentCachedTextureResponsePacket cachedresp = (AgentCachedTextureResponsePacket)PacketPool.Instance.GetPacket(PacketType.AgentCachedTextureResponse); if (cachedtex.AgentData.SessionID != SessionId) return false; // TODO: don't create new blocks if recycling an old packet cachedresp.AgentData.AgentID = AgentId; cachedresp.AgentData.SessionID = m_sessionId; cachedresp.AgentData.SerialNum = cachedtex.AgentData.SerialNum; cachedresp.WearableData = new AgentCachedTextureResponsePacket.WearableDataBlock[cachedtex.WearableData.Length]; int cacheHits = 0; // We need to make sure the asset stored in the bake is available on this server also by it's assetid before we map it to a Cacheid WearableCacheItem[] cacheItems = null; ScenePresence p = m_scene.GetScenePresence(AgentId); if (p != null && p.Appearance != null) { cacheItems = p.Appearance.WearableCacheItems; } int maxWearablesLoop = cachedtex.WearableData.Length; if (cacheItems != null) { if (maxWearablesLoop > cacheItems.Length) maxWearablesLoop = cacheItems.Length; for (int i = 0; i < maxWearablesLoop; i++) { int idx = cachedtex.WearableData[i].TextureIndex; cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock(); cachedresp.WearableData[i].TextureIndex = cachedtex.WearableData[i].TextureIndex; cachedresp.WearableData[i].HostName = new byte[0]; if (cachedtex.WearableData[i].ID == cacheItems[idx].CacheId) { cachedresp.WearableData[i].TextureID = cacheItems[idx].TextureID; cacheHits++; } else { cachedresp.WearableData[i].TextureID = UUID.Zero; } } } else { for (int i = 0; i < maxWearablesLoop; i++) { cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock(); cachedresp.WearableData[i].TextureIndex = cachedtex.WearableData[i].TextureIndex; cachedresp.WearableData[i].TextureID = UUID.Zero; cachedresp.WearableData[i].HostName = new byte[0]; } } //m_log.DebugFormat("texture cached: hits {0}", cacheHits); cachedresp.Header.Zerocoded = true; OutPacket(cachedresp, ThrottleOutPacketType.Task); return true; } /// /// Send a response back to a client when it asks the asset server (via the region server) if it has /// its appearance texture cached. /// /// /// /// /// public void SendCachedTextureResponse(ISceneEntity avatar, int serial, List cachedTextures) { ScenePresence presence = avatar as ScenePresence; if (presence == null) return; AgentCachedTextureResponsePacket cachedresp = (AgentCachedTextureResponsePacket)PacketPool.Instance.GetPacket(PacketType.AgentCachedTextureResponse); // TODO: don't create new blocks if recycling an old packet cachedresp.AgentData.AgentID = m_agentId; cachedresp.AgentData.SessionID = m_sessionId; cachedresp.AgentData.SerialNum = serial; cachedresp.WearableData = new AgentCachedTextureResponsePacket.WearableDataBlock[cachedTextures.Count]; for (int i = 0; i < cachedTextures.Count; i++) { cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock(); cachedresp.WearableData[i].TextureIndex = (byte)cachedTextures[i].BakedTextureIndex; cachedresp.WearableData[i].TextureID = cachedTextures[i].BakedTextureID; cachedresp.WearableData[i].HostName = new byte[0]; } cachedresp.Header.Zerocoded = true; OutPacket(cachedresp, ThrottleOutPacketType.Task); } protected bool HandleMultipleObjUpdate(IClientAPI simClient, Packet packet) { MultipleObjectUpdatePacket multipleupdate = (MultipleObjectUpdatePacket)packet; if (multipleupdate.AgentData.SessionID != SessionId) return false; // m_log.DebugFormat( // "[CLIENT]: Incoming MultipleObjectUpdatePacket contained {0} blocks", multipleupdate.ObjectData.Length); Scene tScene = (Scene)m_scene; for (int i = 0; i < multipleupdate.ObjectData.Length; i++) { MultipleObjectUpdatePacket.ObjectDataBlock block = multipleupdate.ObjectData[i]; // Can't act on Null Data if (block.Data != null) { uint localId = block.ObjectLocalID; SceneObjectPart part = tScene.GetSceneObjectPart(localId); if (part == null) { // It's a ghost! tell the client to delete it from view. simClient.SendKillObject(new List { localId }); } else { ClientChangeObject updatehandler = onClientChangeObject; if (updatehandler != null) { ObjectChangeData udata = new ObjectChangeData(); /*ubit from ll JIRA: * 0x01 position * 0x02 rotation * 0x04 scale * 0x08 LINK_SET * 0x10 UNIFORM for scale */ // translate to internal changes // not all cases .. just the ones older code did switch (block.Type) { case 1: //change position sp udata.position = new Vector3(block.Data, 0); udata.change = ObjectChangeType.primP; updatehandler(localId, udata, this); break; case 2: // rotation sp udata.rotation = new Quaternion(block.Data, 0, true); udata.change = ObjectChangeType.primR; updatehandler(localId, udata, this); break; case 3: // position plus rotation udata.position = new Vector3(block.Data, 0); udata.rotation = new Quaternion(block.Data, 12, true); udata.change = ObjectChangeType.primPR; updatehandler(localId, udata, this); break; case 4: // scale sp udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.primS; updatehandler(localId, udata, this); break; case 0x14: // uniform scale sp udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.primUS; updatehandler(localId, udata, this); break; case 5: // scale and position sp udata.position = new Vector3(block.Data, 0); udata.scale = new Vector3(block.Data, 12); udata.change = ObjectChangeType.primPS; updatehandler(localId, udata, this); break; case 0x15: //uniform scale and position udata.position = new Vector3(block.Data, 0); udata.scale = new Vector3(block.Data, 12); udata.change = ObjectChangeType.primPUS; updatehandler(localId, udata, this); break; // now group related (bit 4) case 9: //( 8 + 1 )group position udata.position = new Vector3(block.Data, 0); udata.change = ObjectChangeType.groupP; updatehandler(localId, udata, this); break; case 0x0A: // (8 + 2) group rotation udata.rotation = new Quaternion(block.Data, 0, true); udata.change = ObjectChangeType.groupR; updatehandler(localId, udata, this); break; case 0x0B: //( 8 + 2 + 1) group rotation and position udata.position = new Vector3(block.Data, 0); udata.rotation = new Quaternion(block.Data, 12, true); udata.change = ObjectChangeType.groupPR; updatehandler(localId, udata, this); break; case 0x0C: // (8 + 4) group scale // only afects root prim and only sent by viewer editor object tab scaling // mouse edition only allows uniform scaling // SL MAY CHANGE THIS in viewers udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.groupS; updatehandler(localId, udata, this); break; case 0x0D: //(8 + 4 + 1) group scale and position // exception as above udata.position = new Vector3(block.Data, 0); udata.scale = new Vector3(block.Data, 12); udata.change = ObjectChangeType.groupPS; updatehandler(localId, udata, this); break; case 0x1C: // (0x10 + 8 + 4 ) group scale UNIFORM udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.groupUS; updatehandler(localId, udata, this); break; case 0x1D: // (UNIFORM + GROUP + SCALE + POS) udata.position = new Vector3(block.Data, 0); udata.scale = new Vector3(block.Data, 12); udata.change = ObjectChangeType.groupPUS; updatehandler(localId, udata, this); break; default: m_log.Debug("[CLIENT]: MultipleObjUpdate recieved an unknown packet type: " + (block.Type)); break; } } } } } return true; } public void RequestMapLayer() { //should be getting the map layer from the grid server //send a layer covering the 800,800 - 1200,1200 area (should be covering the requested area) MapLayerReplyPacket mapReply = (MapLayerReplyPacket)PacketPool.Instance.GetPacket(PacketType.MapLayerReply); // TODO: don't create new blocks if recycling an old packet mapReply.AgentData.AgentID = AgentId; mapReply.AgentData.Flags = 0; mapReply.LayerData = new MapLayerReplyPacket.LayerDataBlock[1]; mapReply.LayerData[0] = new MapLayerReplyPacket.LayerDataBlock(); mapReply.LayerData[0].Bottom = 0; mapReply.LayerData[0].Left = 0; mapReply.LayerData[0].Top = 30000; mapReply.LayerData[0].Right = 30000; mapReply.LayerData[0].ImageID = new UUID("00000000-0000-1111-9999-000000000006"); mapReply.Header.Zerocoded = true; OutPacket(mapReply, ThrottleOutPacketType.Land); } public void RequestMapBlocksX(int minX, int minY, int maxX, int maxY) { /* IList simMapProfiles = m_gridServer.RequestMapBlocks(minX, minY, maxX, maxY); MapBlockReplyPacket mbReply = new MapBlockReplyPacket(); mbReply.AgentData.AgentId = AgentId; int len; if (simMapProfiles == null) len = 0; else len = simMapProfiles.Count; mbReply.Data = new MapBlockReplyPacket.DataBlock[len]; int iii; for (iii = 0; iii < len; iii++) { Hashtable mp = (Hashtable)simMapProfiles[iii]; mbReply.Data[iii] = new MapBlockReplyPacket.DataBlock(); mbReply.Data[iii].Name = Util.UTF8.GetBytes((string)mp["name"]); mbReply.Data[iii].Access = System.Convert.ToByte(mp["access"]); mbReply.Data[iii].Agents = System.Convert.ToByte(mp["agents"]); mbReply.Data[iii].MapImageID = new UUID((string)mp["map-image-id"]); mbReply.Data[iii].RegionFlags = System.Convert.ToUInt32(mp["region-flags"]); mbReply.Data[iii].WaterHeight = System.Convert.ToByte(mp["water-height"]); mbReply.Data[iii].X = System.Convert.ToUInt16(mp["x"]); mbReply.Data[iii].Y = System.Convert.ToUInt16(mp["y"]); } this.OutPacket(mbReply, ThrottleOutPacketType.Land); */ } /// /// Sets the throttles from values supplied by the client /// /// public void SetChildAgentThrottle(byte[] throttles) { SetChildAgentThrottle(throttles, 1.0f); } public void SetChildAgentThrottle(byte[] throttles,float factor) { m_udpClient.SetThrottles(throttles, factor); GenericCall2 handler = OnUpdateThrottles; if (handler != null) { handler(); } } /// /// Sets the throttles from values supplied caller /// /// public void SetAgentThrottleSilent(int throttle, int setting) { m_udpClient.ForceThrottleSetting(throttle,setting); } public int GetAgentThrottleSilent(int throttle) { return m_udpClient.GetThrottleSetting(throttle); } /// /// Get the current throttles for this client as a packed byte array /// /// Unused /// public byte[] GetThrottlesPacked(float multiplier) { return m_udpClient.GetThrottlesPacked(multiplier); } /// /// Cruft? /// public virtual void InPacket(object NewPack) { throw new NotImplementedException(); } /// /// This is the starting point for sending a simulator packet out to the client /// /// Packet to send /// Throttling category for the packet protected void OutPacket(Packet packet, ThrottleOutPacketType throttlePacketType) { #region BinaryStats LLUDPServer.LogPacketHeader(false, m_circuitCode, 0, packet.Type, (ushort)packet.Length); #endregion BinaryStats OutPacket(packet, throttlePacketType, true); } /// /// 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 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_outPacketsToDrop != null) { if (m_outPacketsToDrop.Contains(packet.Type.ToString())) { PacketPool.Instance.ReturnPacket(packet); return; } } if (DebugPacketLevel > 0) { bool logPacket = true; if (DebugPacketLevel <= 255 && (packet.Type == PacketType.SimStats || packet.Type == PacketType.SimulatorViewerTimeMessage)) logPacket = false; if (DebugPacketLevel <= 200 && (packet.Type == PacketType.ImagePacket || packet.Type == PacketType.ImageData || packet.Type == PacketType.LayerData || packet.Type == PacketType.CoarseLocationUpdate)) logPacket = false; if (DebugPacketLevel <= 100 && (packet.Type == PacketType.AvatarAnimation || packet.Type == PacketType.ViewerEffect)) logPacket = false; if (DebugPacketLevel <= 50 && (packet.Type == PacketType.ImprovedTerseObjectUpdate || packet.Type == PacketType.ObjectUpdate)) logPacket = false; if (DebugPacketLevel <= 25 && packet.Type == PacketType.ObjectPropertiesFamily) logPacket = false; if (logPacket) m_log.DebugFormat( "[CLIENT]: PACKET OUT to {0} ({1}) in {2} - {3}", Name, SceneAgent.IsChildAgent ? "child" : "root ", m_scene.RegionInfo.RegionName, packet.Type); } m_udpServer.SendPacket(m_udpClient, packet, throttlePacketType, doAutomaticSplitting, method); } protected void HandleAutopilot(Object sender, string method, List args) { float locx = 0; float locy = 0; float locz = 0; uint regionX = 0; uint regionY = 0; Utils.LongToUInts(m_scene.RegionInfo.RegionHandle, out regionX, out regionY); locx = (float)(Convert.ToDouble(args[0]) - (double)regionX); locy = (float)(Convert.ToDouble(args[1]) - (double)regionY); locz = Convert.ToSingle(args[2]); Action handlerAutoPilotGo = OnAutoPilotGo; if (handlerAutoPilotGo != null) handlerAutoPilotGo(new Vector3(locx, locy, locz), false, false); } /// /// Entryway from the client to the simulator. All UDP packets from the client will end up here /// /// OpenMetaverse.packet public void ProcessInPacket(Packet packet) { if (m_inPacketsToDrop != null) if (m_inPacketsToDrop.Contains(packet.Type.ToString())) return; if (DebugPacketLevel > 0) { bool logPacket = true; if (DebugPacketLevel <= 255 && packet.Type == PacketType.AgentUpdate) logPacket = false; if (DebugPacketLevel <= 200 && packet.Type == PacketType.RequestImage) logPacket = false; if (DebugPacketLevel <= 100 && (packet.Type == PacketType.ViewerEffect || packet.Type == PacketType.AgentAnimation)) logPacket = false; if (DebugPacketLevel <= 25 && packet.Type == PacketType.RequestObjectPropertiesFamily) logPacket = false; if (logPacket) m_log.DebugFormat( "[CLIENT]: PACKET IN from {0} ({1}) in {2} - {3}", Name, SceneAgent.IsChildAgent ? "child" : "root ", Scene.Name, packet.Type); } if (!ProcessPacketMethod(packet)) m_log.WarnFormat( "[CLIENT]: Unhandled packet {0} from {1} ({2}) in {3}. Ignoring.", packet.Type, Name, SceneAgent.IsChildAgent ? "child" : "root ", Scene.Name); } private static PrimitiveBaseShape GetShapeFromAddPacket(ObjectAddPacket addPacket) { PrimitiveBaseShape shape = new PrimitiveBaseShape(); shape.PCode = addPacket.ObjectData.PCode; shape.State = addPacket.ObjectData.State; shape.LastAttachPoint = addPacket.ObjectData.State; shape.PathBegin = addPacket.ObjectData.PathBegin; shape.PathEnd = addPacket.ObjectData.PathEnd; shape.PathScaleX = addPacket.ObjectData.PathScaleX; shape.PathScaleY = addPacket.ObjectData.PathScaleY; shape.PathShearX = addPacket.ObjectData.PathShearX; shape.PathShearY = addPacket.ObjectData.PathShearY; shape.PathSkew = addPacket.ObjectData.PathSkew; shape.ProfileBegin = addPacket.ObjectData.ProfileBegin; shape.ProfileEnd = addPacket.ObjectData.ProfileEnd; shape.Scale = addPacket.ObjectData.Scale; shape.PathCurve = addPacket.ObjectData.PathCurve; shape.ProfileCurve = addPacket.ObjectData.ProfileCurve; shape.ProfileHollow = addPacket.ObjectData.ProfileHollow; shape.PathRadiusOffset = addPacket.ObjectData.PathRadiusOffset; shape.PathRevolutions = addPacket.ObjectData.PathRevolutions; shape.PathTaperX = addPacket.ObjectData.PathTaperX; shape.PathTaperY = addPacket.ObjectData.PathTaperY; shape.PathTwist = addPacket.ObjectData.PathTwist; shape.PathTwistBegin = addPacket.ObjectData.PathTwistBegin; Primitive.TextureEntry ntex = new Primitive.TextureEntry(new UUID("89556747-24cb-43ed-920b-47caed15465f")); shape.TextureEntry = ntex.GetBytes(); //shape.Textures = ntex; return shape; } public ClientInfo GetClientInfo() { ClientInfo info = m_udpClient.GetClientInfo(); info.proxyEP = null; if (info.agentcircuit == null) info.agentcircuit = RequestClientInfo(); return info; } public void SetClientInfo(ClientInfo info) { m_udpClient.SetClientInfo(info); } #region Media Parcel Members public void SendParcelMediaCommand(uint flags, ParcelMediaCommandEnum command, float time) { ParcelMediaCommandMessagePacket commandMessagePacket = new ParcelMediaCommandMessagePacket(); commandMessagePacket.CommandBlock.Flags = flags; commandMessagePacket.CommandBlock.Command = (uint)command; commandMessagePacket.CommandBlock.Time = time; OutPacket(commandMessagePacket, ThrottleOutPacketType.Task); } public void SendParcelMediaUpdate(string mediaUrl, UUID mediaTextureID, byte autoScale, string mediaType, string mediaDesc, int mediaWidth, int mediaHeight, byte mediaLoop) { ParcelMediaUpdatePacket updatePacket = new ParcelMediaUpdatePacket(); updatePacket.DataBlock.MediaURL = Util.StringToBytes256(mediaUrl); updatePacket.DataBlock.MediaID = mediaTextureID; updatePacket.DataBlock.MediaAutoScale = autoScale; updatePacket.DataBlockExtended.MediaType = Util.StringToBytes256(mediaType); updatePacket.DataBlockExtended.MediaDesc = Util.StringToBytes256(mediaDesc); updatePacket.DataBlockExtended.MediaWidth = mediaWidth; updatePacket.DataBlockExtended.MediaHeight = mediaHeight; updatePacket.DataBlockExtended.MediaLoop = mediaLoop; OutPacket(updatePacket, ThrottleOutPacketType.Task); } #endregion #region Camera public void SendSetFollowCamProperties(UUID objectID, SortedDictionary parameters) { SetFollowCamPropertiesPacket packet = (SetFollowCamPropertiesPacket)PacketPool.Instance.GetPacket(PacketType.SetFollowCamProperties); packet.ObjectData.ObjectID = objectID; SetFollowCamPropertiesPacket.CameraPropertyBlock[] camPropBlock = new SetFollowCamPropertiesPacket.CameraPropertyBlock[parameters.Count]; uint idx = 0; foreach (KeyValuePair pair in parameters) { SetFollowCamPropertiesPacket.CameraPropertyBlock block = new SetFollowCamPropertiesPacket.CameraPropertyBlock(); block.Type = pair.Key; block.Value = pair.Value; camPropBlock[idx++] = block; } packet.CameraProperty = camPropBlock; OutPacket(packet, ThrottleOutPacketType.Task); } public void SendClearFollowCamProperties(UUID objectID) { ClearFollowCamPropertiesPacket packet = (ClearFollowCamPropertiesPacket)PacketPool.Instance.GetPacket(PacketType.ClearFollowCamProperties); packet.ObjectData.ObjectID = objectID; OutPacket(packet, ThrottleOutPacketType.Task); } #endregion public void SetClientOption(string option, string value) { switch (option) { default: break; } } public string GetClientOption(string option) { switch (option) { default: break; } return string.Empty; } #region IClientCore private readonly Dictionary m_clientInterfaces = new Dictionary(); /// /// Register an interface on this client, should only be called in the constructor. /// /// /// protected void RegisterInterface(T iface) { lock (m_clientInterfaces) { if (!m_clientInterfaces.ContainsKey(typeof(T))) { m_clientInterfaces.Add(typeof(T), iface); } } } public bool TryGet(out T iface) { if (m_clientInterfaces.ContainsKey(typeof(T))) { iface = (T)m_clientInterfaces[typeof(T)]; return true; } iface = default(T); return false; } public T Get() { return (T)m_clientInterfaces[typeof(T)]; } public void Disconnect(string reason) { Kick(reason); Thread.Sleep(1000); Disconnect(); } public void Disconnect() { Close(); } #endregion public void RefreshGroupMembership() { lock(m_groupPowers) { GroupMembershipData activeMembership = null; if (m_GroupsModule != null) { GroupMembershipData[] GroupMembership = m_GroupsModule.GetMembershipData(AgentId); m_groupPowers.Clear(); if (GroupMembership != null) { for (int i = 0; i < GroupMembership.Length; i++) { m_groupPowers[GroupMembership[i].GroupID] = GroupMembership[i].GroupPowers; } } activeMembership = m_GroupsModule.GetActiveMembershipData(AgentId); if(activeMembership != null) { if(!m_groupPowers.ContainsKey(activeMembership.GroupID)) activeMembership = null; else { m_activeGroupID = activeMembership.GroupID; m_activeGroupName = activeMembership.GroupName; m_activeGroupPowers = ActiveGroupPowers; } } } if(activeMembership == null) { m_activeGroupID = UUID.Zero; m_activeGroupName = ""; m_activeGroupPowers = 0; } } } public void UpdateGroupMembership(GroupMembershipData[] data) { lock(m_groupPowers) { m_groupPowers.Clear(); if (data != null) { for (int i = 0; i < data.Length; i++) m_groupPowers[data[i].GroupID] = data[i].GroupPowers; } } } public void GroupMembershipRemove(UUID GroupID) { lock(m_groupPowers) { if(m_groupPowers.ContainsKey(GroupID)) m_groupPowers.Remove(GroupID); } } public void GroupMembershipAddReplace(UUID GroupID,ulong GroupPowers) { lock(m_groupPowers) { m_groupPowers[GroupID] = GroupPowers; } } public string Report() { return m_udpClient.GetStats(); } public string XReport(string uptime, string version) { return String.Empty; } public OSDMap OReport(string uptime, string version) { return new OSDMap(); } /// /// Make an asset request to the asset service in response to a client request. /// /// /// protected void MakeAssetRequest(TransferRequestPacket transferRequest, UUID taskID) { UUID requestID = UUID.Zero; int sourceType = transferRequest.TransferInfo.SourceType; if (sourceType == (int)SourceType.Asset) { requestID = new UUID(transferRequest.TransferInfo.Params, 0); } else if (sourceType == (int)SourceType.SimInventoryItem) { requestID = new UUID(transferRequest.TransferInfo.Params, 80); } else if (sourceType == (int)SourceType.SimEstate) { requestID = taskID; } // m_log.DebugFormat( // "[LLCLIENTVIEW]: Received transfer request for {0} in {1} type {2} by {3}", // requestID, taskID, (SourceType)sourceType, Name); //Note, the bool returned from the below function is useless since it is always false. m_assetService.Get(requestID.ToString(), transferRequest, AssetReceived); } /// /// When we get a reply back from the asset service in response to a client request, send back the data. /// /// /// /// protected void AssetReceived(string id, Object sender, AssetBase asset) { TransferRequestPacket transferRequest = (TransferRequestPacket)sender; UUID requestID = UUID.Zero; byte source = (byte)SourceType.Asset; AssetRequestToClient req = new AssetRequestToClient(); if (asset == null) { // Try the user's asset server IInventoryAccessModule inventoryAccessModule = Scene.RequestModuleInterface(); string assetServerURL = string.Empty; if (inventoryAccessModule.IsForeignUser(AgentId, out assetServerURL) && !string.IsNullOrEmpty(assetServerURL)) { if (!assetServerURL.EndsWith("/") && !assetServerURL.EndsWith("=")) assetServerURL = assetServerURL + "/"; //m_log.DebugFormat("[LLCLIENTVIEW]: asset {0} not found in local storage. Trying user's storage.", assetServerURL + id); asset = m_scene.AssetService.Get(assetServerURL + id); } if (asset == null) { req.AssetInf = null; req.AssetRequestSource = source; req.IsTextureRequest = false; req.NumPackets = 0; req.Params = transferRequest.TransferInfo.Params; req.RequestAssetID = requestID; req.TransferRequestID = transferRequest.TransferInfo.TransferID; SendAssetNotFound(req); return; } } if (transferRequest.TransferInfo.SourceType == (int)SourceType.Asset) { requestID = new UUID(transferRequest.TransferInfo.Params, 0); } else if (transferRequest.TransferInfo.SourceType == (int)SourceType.SimInventoryItem) { requestID = new UUID(transferRequest.TransferInfo.Params, 80); source = (byte)SourceType.SimInventoryItem; //m_log.Debug("asset request " + requestID); } // Scripts cannot be retrieved by direct request if (transferRequest.TransferInfo.SourceType == (int)SourceType.Asset && asset.Type == 10) return; // The asset is known to exist and is in our cache, so add it to the AssetRequests list req.AssetInf = asset; req.AssetRequestSource = source; req.IsTextureRequest = false; req.NumPackets = CalculateNumPackets(asset.Data); req.Params = transferRequest.TransferInfo.Params; req.RequestAssetID = requestID; req.TransferRequestID = transferRequest.TransferInfo.TransferID; SendAsset(req); } /// /// Calculate the number of packets required to send the asset to the client. /// /// /// private static int CalculateNumPackets(byte[] data) { // const uint m_maxPacketSize = 600; uint m_maxPacketSize = MaxTransferBytesPerPacket; int numPackets = 1; if (data == null) return 0; if (data.LongLength > m_maxPacketSize) { // over max number of bytes so split up file long restData = data.LongLength - m_maxPacketSize; int restPackets = (int)((restData + m_maxPacketSize - 1) / m_maxPacketSize); numPackets += restPackets; } return numPackets; } public void SendRebakeAvatarTextures(UUID textureID) { RebakeAvatarTexturesPacket pack = (RebakeAvatarTexturesPacket)PacketPool.Instance.GetPacket(PacketType.RebakeAvatarTextures); pack.TextureData = new RebakeAvatarTexturesPacket.TextureDataBlock(); pack.TextureData.TextureID = textureID; OutPacket(pack, ThrottleOutPacketType.Task); } public struct PacketProcessor { /// /// Packet handling method. /// public PacketMethod method { get; set; } /// /// Should this packet be handled asynchronously? /// public bool Async { get; set; } } public class AsyncPacketProcess { public bool result = false; public readonly LLClientView ClientView = null; public readonly Packet Pack = null; public readonly PacketMethod Method = null; public AsyncPacketProcess(LLClientView pClientview, PacketMethod pMethod, Packet pPack) { ClientView = pClientview; Method = pMethod; Pack = pPack; } } public void SendAvatarInterestsReply(UUID avatarID, uint wantMask, string wantText, uint skillsMask, string skillsText, string languages) { AvatarInterestsReplyPacket packet = (AvatarInterestsReplyPacket)PacketPool.Instance.GetPacket(PacketType.AvatarInterestsReply); packet.AgentData = new AvatarInterestsReplyPacket.AgentDataBlock(); packet.AgentData.AgentID = AgentId; packet.AgentData.AvatarID = avatarID; packet.PropertiesData = new AvatarInterestsReplyPacket.PropertiesDataBlock(); packet.PropertiesData.WantToMask = wantMask; packet.PropertiesData.WantToText = Utils.StringToBytes(wantText); packet.PropertiesData.SkillsMask = skillsMask; packet.PropertiesData.SkillsText = Utils.StringToBytes(skillsText); packet.PropertiesData.LanguagesText = Utils.StringToBytes(languages); OutPacket(packet, ThrottleOutPacketType.Task); } public void SendChangeUserRights(UUID agentID, UUID friendID, int rights) { ChangeUserRightsPacket packet = (ChangeUserRightsPacket)PacketPool.Instance.GetPacket(PacketType.ChangeUserRights); packet.AgentData = new ChangeUserRightsPacket.AgentDataBlock(); packet.AgentData.AgentID = agentID; packet.Rights = new ChangeUserRightsPacket.RightsBlock[1]; packet.Rights[0] = new ChangeUserRightsPacket.RightsBlock(); packet.Rights[0].AgentRelated = friendID; packet.Rights[0].RelatedRights = rights; OutPacket(packet, ThrottleOutPacketType.Task); } public void SendTextBoxRequest(string message, int chatChannel, string objectname, UUID ownerID, string ownerFirstName, string ownerLastName, UUID objectId) { ScriptDialogPacket dialog = (ScriptDialogPacket)PacketPool.Instance.GetPacket(PacketType.ScriptDialog); dialog.Data.ObjectID = objectId; dialog.Data.ChatChannel = chatChannel; dialog.Data.ImageID = UUID.Zero; dialog.Data.ObjectName = Util.StringToBytes256(objectname); // this is the username of the *owner* dialog.Data.FirstName = Util.StringToBytes256(ownerFirstName); dialog.Data.LastName = Util.StringToBytes256(ownerLastName); dialog.Data.Message = Util.StringToBytes(message,512); ScriptDialogPacket.ButtonsBlock[] buttons = new ScriptDialogPacket.ButtonsBlock[1]; buttons[0] = new ScriptDialogPacket.ButtonsBlock(); buttons[0].ButtonLabel = Util.StringToBytes256("!!llTextBox!!"); dialog.Buttons = buttons; dialog.OwnerData = new ScriptDialogPacket.OwnerDataBlock[1]; dialog.OwnerData[0] = new ScriptDialogPacket.OwnerDataBlock(); dialog.OwnerData[0].OwnerID = ownerID; OutPacket(dialog, ThrottleOutPacketType.Task); } public void SendAgentTerseUpdate(ISceneEntity p) { if (p is ScenePresence) { UDPPacketBuffer buf = m_udpServer.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); //setup header and regioninfo block Buffer.BlockCopy(terseUpdateHeader, 0, buf.Data, 0, 7); Utils.UInt64ToBytesSafepos(m_scene.RegionInfo.RegionHandle, buf.Data, 7); Utils.UInt16ToBytes(Utils.FloatToUInt16(m_scene.TimeDilation, 0.0f, 1.0f), buf.Data, 15); buf.Data[17] = 1; int pos = 18; CreateImprovedTerseBlock(p, buf.Data, ref pos, false); buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); } } public void SendPlacesReply(UUID queryID, UUID transactionID, PlacesReplyData[] data) { PlacesReplyPacket reply = null; PlacesReplyPacket.QueryDataBlock[] dataBlocks = new PlacesReplyPacket.QueryDataBlock[0]; for (int i = 0 ; i < data.Length ; i++) { PlacesReplyPacket.QueryDataBlock block = new PlacesReplyPacket.QueryDataBlock(); block.OwnerID = data[i].OwnerID; block.Name = Util.StringToBytes256(data[i].Name); block.Desc = Util.StringToBytes1024(data[i].Desc); block.ActualArea = data[i].ActualArea; block.BillableArea = data[i].BillableArea; block.Flags = data[i].Flags; block.GlobalX = data[i].GlobalX; block.GlobalY = data[i].GlobalY; block.GlobalZ = data[i].GlobalZ; block.SimName = Util.StringToBytes256(data[i].SimName); block.SnapshotID = data[i].SnapshotID; block.Dwell = data[i].Dwell; block.Price = data[i].Price; if (reply != null && reply.Length + block.Length > 1400) { OutPacket(reply, ThrottleOutPacketType.Task); reply = null; dataBlocks = new PlacesReplyPacket.QueryDataBlock[0]; } if (reply == null) { reply = (PlacesReplyPacket)PacketPool.Instance.GetPacket(PacketType.PlacesReply); reply.AgentData = new PlacesReplyPacket.AgentDataBlock(); reply.AgentData.AgentID = AgentId; reply.AgentData.QueryID = queryID; reply.TransactionData = new PlacesReplyPacket.TransactionDataBlock(); reply.TransactionData.TransactionID = transactionID; reply.QueryData = dataBlocks; } Array.Resize(ref dataBlocks, dataBlocks.Length + 1); dataBlocks[dataBlocks.Length - 1] = block; reply.QueryData = dataBlocks; } if (reply != null) OutPacket(reply, ThrottleOutPacketType.Task); } public void SendRemoveInventoryItems(UUID[] items) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } StringBuilder sb = eq.StartEvent("RemoveInventoryItem"); LLSDxmlEncode.AddArrayAndMap("AgentData", sb); LLSDxmlEncode.AddElem("AgentID", AgentId, sb); LLSDxmlEncode.AddElem("SessionID", SessionId, sb); LLSDxmlEncode.AddEndMapAndArray(sb); LLSDxmlEncode.AddArray("InventoryData", sb); foreach (UUID item in items) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("ItemID",item, sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); OSD ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } public void SendRemoveInventoryFolders(UUID[] folders) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } StringBuilder sb = eq.StartEvent("RemoveInventoryFolder"); LLSDxmlEncode.AddArrayAndMap("AgentData", sb); LLSDxmlEncode.AddElem("AgentID", AgentId, sb); LLSDxmlEncode.AddElem("SessionID", SessionId, sb); LLSDxmlEncode.AddEndMapAndArray(sb); LLSDxmlEncode.AddArray("FolderData", sb); foreach (UUID folder in folders) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("FolderID", folder, sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); OSD ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } public void SendBulkUpdateInventory(InventoryFolderBase[] folders, InventoryItemBase[] items) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq == null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } StringBuilder sb = eq.StartEvent("BulkUpdateInventory"); LLSDxmlEncode.AddArrayAndMap("AgentData", sb); LLSDxmlEncode.AddElem("AgentID", AgentId, sb); LLSDxmlEncode.AddElem("TransactionID", UUID.Random(), sb); LLSDxmlEncode.AddEndMapAndArray(sb); if(folders.Length == 0) { LLSDxmlEncode.AddEmptyArray("FolderData", sb); } else { LLSDxmlEncode.AddArray("FolderData", sb); foreach (InventoryFolderBase folder in folders) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("FolderID", folder.ID, sb); LLSDxmlEncode.AddElem("ParentID", folder.ParentID, sb); LLSDxmlEncode.AddElem("Type", (int)folder.Type, sb); LLSDxmlEncode.AddElem("Name", folder.Name, sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); } if(items.Length == 0) { LLSDxmlEncode.AddEmptyArray("ItemData", sb); } else { LLSDxmlEncode.AddArray("ItemData", sb); foreach (InventoryItemBase item in items) { LLSDxmlEncode.AddMap(sb); LLSDxmlEncode.AddElem("ItemID", item.ID, sb); LLSDxmlEncode.AddElem("CallbackID", (uint)0, sb); LLSDxmlEncode.AddElem("FolderID", item.Folder, sb); LLSDxmlEncode.AddElem("CreatorID", item.CreatorIdAsUuid, sb); LLSDxmlEncode.AddElem("OwnerID", item.Owner, sb); LLSDxmlEncode.AddElem("GroupID", item.GroupID, sb); LLSDxmlEncode.AddElem("BaseMask", item.BasePermissions, sb); LLSDxmlEncode.AddElem("OwnerMask", item.CurrentPermissions, sb); LLSDxmlEncode.AddElem("GroupMask", item.GroupPermissions, sb); LLSDxmlEncode.AddElem("EveryoneMask", item.EveryOnePermissions, sb); LLSDxmlEncode.AddElem("NextOwnerMask", item.NextPermissions, sb); LLSDxmlEncode.AddElem("GroupOwned", item.GroupOwned, sb); LLSDxmlEncode.AddElem("AssetID", item.AssetID, sb); LLSDxmlEncode.AddElem("Type", item.AssetType, sb); LLSDxmlEncode.AddElem("InvType", item.InvType, sb); LLSDxmlEncode.AddElem("Flags", item.Flags, sb); LLSDxmlEncode.AddElem("SaleType", item.SaleType, sb); LLSDxmlEncode.AddElem("SalePrice", item.SalePrice, sb); LLSDxmlEncode.AddElem("Name", item.Name, sb); LLSDxmlEncode.AddElem("Description", item.Description, sb); LLSDxmlEncode.AddElem("CreationDate", item.CreationDate, sb); LLSDxmlEncode.AddElem("CRC", Helpers.InventoryCRC(1000, 0, (sbyte)item.InvType, (sbyte)item.AssetType, item.AssetID, item.GroupID, 100, item.Owner, item.CreatorIdAsUuid, item.ID, item.Folder, (uint)PermissionMask.All, 1, (uint)PermissionMask.All, (uint)PermissionMask.All, (uint)PermissionMask.All), sb); LLSDxmlEncode.AddEndMap(sb); } LLSDxmlEncode.AddEndArray(sb); } OSD ev = new OSDllsdxml(eq.EndEvent(sb)); eq.Enqueue(ev, AgentId); } private HashSet m_outPacketsToDrop; public bool AddOutPacketToDropSet(string packetName) { if (m_outPacketsToDrop == null) m_outPacketsToDrop = new HashSet(); return m_outPacketsToDrop.Add(packetName); } public bool RemoveOutPacketFromDropSet(string packetName) { if (m_outPacketsToDrop == null) return false; return m_outPacketsToDrop.Remove(packetName); } public HashSet GetOutPacketDropSet() { return new HashSet(m_outPacketsToDrop); } private HashSet m_inPacketsToDrop; public bool AddInPacketToDropSet(string packetName) { if (m_inPacketsToDrop == null) m_inPacketsToDrop = new HashSet(); return m_inPacketsToDrop.Add(packetName); } public bool RemoveInPacketFromDropSet(string packetName) { if (m_inPacketsToDrop == null) return false; return m_inPacketsToDrop.Remove(packetName); } public HashSet GetInPacketDropSet() { return new HashSet(m_inPacketsToDrop); } public void CheckViewerCaps() { m_SupportObjectAnimations = false; if (m_scene.CapsModule != null) { Caps cap = m_scene.CapsModule.GetCapsForUser(CircuitCode); if (cap != null && (cap.Flags & Caps.CapsFlags.ObjectAnim) != 0) m_SupportObjectAnimations = true; } } } }