/* * 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.Collections.Frozen; using System.Net; using System.Reflection; using System.Runtime; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Threading; using log4net; using OpenMetaverse; using OpenMetaverse.Packets; 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 void PacketMethod(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; } private delegate void StaticPacketMethod(LLClientView cli, Packet packet); #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 AgentDataUpdate 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); private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static readonly 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; internal readonly UUID m_sessionId; private readonly UUID m_secureSessionId; internal readonly UUID m_agentId; protected readonly UUID m_scopeId; private readonly uint m_circuitCode; private readonly byte[] m_regionChannelVersion = Utils.EmptyBytes; private readonly IGroupsModule m_GroupsModule; //private int m_cachedTextureSerial; private readonly PriorityQueue m_entityUpdates; private readonly PriorityQueue m_entityProps; private readonly Prioritizer m_prioritizer; private bool m_disableFacelights; private HashSet GroupsInView = []; #pragma warning disable 0414 private bool m_VelocityInterpolate; #pragma warning restore 0414 private const uint MaxTransferBytesPerPacket = 600; private bool m_SupportObjectAnimations; private bool m_SupportPBR; public bool SupportTerrainPBR { get; private set; } public ViewerFlags ViewerFlags { get; private set; } /// /// 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_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 readonly AgentUpdateArgs m_thisAgentUpdateArgs = new(); protected Dictionary m_packetHandlers = new(); protected Dictionary m_genericPacketHandlers = new(); //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(); protected int m_terrainCheckerCount; protected uint m_agentFOVCounter; protected IAssetService m_assetService; protected bool m_supportViewerCache = false; #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 float StartFar { get; set; } public float FOV { get; set; } = 1.25f; public int viewHeight { get; set; } = 480; public int viewWidth { get; set; } = 640; public UUID AgentId { get { return m_agentId; } } public UUID ScopeId { get { return m_scopeId; } } 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 readonly object CloseSyncLock = new object(); 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; 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_regionChannelVersion = Util.StringToBytes1024(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; StartFar = sessionInfo.LoginInfo.StartFar; m_udpServer = udpServer; m_udpClient = udpClient; m_udpClient.OnQueueEmpty += HandleQueueEmpty; m_udpClient.HasUpdates += HandleHasUpdates; m_udpClient.OnPacketStats += PopulateStats; m_prioritizer = new Prioritizer(m_scene); // Pick up agent scope which, for gods, can be different from the region scope IUserAccountService userAccountService = m_scene.RequestModuleInterface(); var myself = userAccountService.GetUserAccount(m_scene.RegionInfo.ScopeID, m_agentId); RegisterLocalPacketHandlers(); string name = $"AsyncInUDP-{m_agentId}"; m_asyncPacketProcess = new JobEngine(name, name, 5000); IsActive = true; m_supportViewerCache = m_udpServer.SupportViewerObjectsCache; } #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 OnConnectionClosed?.Invoke(this); m_asyncPacketProcess.Stop(); // Flush all of the packets out of the UDP server for this client m_udpServer?.Flush(m_udpClient); // Remove ourselves from the scene m_scene.RemoveClient(m_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 = m_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) { OnNetworkStatsUpdate?.Invoke(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. [MethodImpl(MethodImplOptions.AggressiveInlining)] 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. [MethodImpl(MethodImplOptions.AggressiveInlining)] public bool AddLocalPacketHandler(PacketType packetType, PacketMethod handler, bool doAsync) { lock (m_packetHandlers) return m_packetHandlers.TryAdd(packetType, new PacketProcessor(handler, doAsync)); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public bool AddGenericPacketHandler(string MethodName, GenericMessage handler) { MethodName = MethodName.ToLower().Trim(); lock (m_genericPacketHandlers) return m_genericPacketHandlers.TryAdd(MethodName, handler); } /// /// 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) { if (m_staticHandlers.TryGetValue(packet.Type, out StaticPacketProcessor spp)) { if (spp.Async) { Packet lp = packet; StaticPacketMethod m = spp.method; LLClientView c = this; _ = m_asyncPacketProcess.QueueJob(packet.Type.ToString(), () => { m(c,lp); }); } else { spp.method(this, packet); } return true; } if (m_packetHandlers.TryGetValue(packet.Type, out PacketProcessor pprocessor)) { if (pprocessor.Async) { Packet lp = packet; PacketMethod m = pprocessor.method; _ = m_asyncPacketProcess.QueueJob(packet.Type.ToString(), () => { m(lp); }); } else { pprocessor.method(packet); } return true; } return false; } #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 // temporary here ( from estatemanagermodule) private uint GetRegionFlags() { RegionFlags flags = RegionFlags.None; if (Scene.RegionInfo.RegionSettings.AllowDamage) flags |= RegionFlags.AllowDamage; if (Scene.RegionInfo.EstateSettings.AllowLandmark) flags |= RegionFlags.AllowLandmark; if (Scene.RegionInfo.EstateSettings.AllowSetHome) flags |= RegionFlags.AllowSetHome; if (Scene.RegionInfo.EstateSettings.ResetHomeOnTeleport) flags |= RegionFlags.ResetHomeOnTeleport; if (Scene.RegionInfo.RegionSettings.FixedSun) flags |= RegionFlags.SunFixed; // allow access override (was taxfree) if (!Scene.RegionInfo.EstateSettings.TaxFree) // this is now wrong means !ALLOW_ACCESS_OVERRIDE flags |= RegionFlags.AllowParcelAccessOverride; if (Scene.RegionInfo.RegionSettings.BlockTerraform) flags |= RegionFlags.BlockTerraform; if (!Scene.RegionInfo.RegionSettings.AllowLandResell) flags |= RegionFlags.BlockLandResell; if (Scene.RegionInfo.RegionSettings.Sandbox) flags |= RegionFlags.Sandbox; // nulllayer not used if (Scene.RegionInfo.RegionSettings.Casino) flags |= RegionFlags.SkipAgentAction; // redefined if (Scene.RegionInfo.RegionSettings.GodBlockSearch) flags |= RegionFlags.SkipUpdateInterestList; // redefined if (Scene.RegionInfo.RegionSettings.DisableCollisions) flags |= RegionFlags.SkipCollisions; if (Scene.RegionInfo.RegionSettings.DisableScripts) flags |= RegionFlags.SkipScripts; if (Scene.RegionInfo.RegionSettings.DisablePhysics) flags |= RegionFlags.SkipPhysics; if (Scene.RegionInfo.EstateSettings.PublicAccess) flags |= RegionFlags.ExternallyVisible; // ???? need revision //MainlandVisible -> allow return enc object //PublicAllowed -> allow return enc estate object if (Scene.RegionInfo.EstateSettings.BlockDwell) flags |= RegionFlags.BlockDwell; if (Scene.RegionInfo.RegionSettings.BlockFly) flags |= RegionFlags.NoFly; if (Scene.RegionInfo.EstateSettings.AllowDirectTeleport) flags |= RegionFlags.AllowDirectTeleport; if (Scene.RegionInfo.EstateSettings.EstateSkipScripts) flags |= RegionFlags.EstateSkipScripts; if (Scene.RegionInfo.RegionSettings.RestrictPushing) flags |= RegionFlags.RestrictPushObject; if (Scene.RegionInfo.EstateSettings.DenyAnonymous) flags |= RegionFlags.DenyAnonymous; //DenyIdentified unused //DenyTransacted unused if (Scene.RegionInfo.RegionSettings.AllowLandJoinDivide) flags |= RegionFlags.AllowParcelChanges; //AbuseEmailToEstateOwner -> block flyover if (Scene.RegionInfo.EstateSettings.AllowVoice) flags |= RegionFlags.AllowVoice; if (Scene.RegionInfo.RegionSettings.BlockShowInSearch) flags |= RegionFlags.BlockParcelSearch; if (Scene.RegionInfo.EstateSettings.DenyMinors) flags |= RegionFlags.DenyAgeUnverified; if(Scene.RegionInfo.EstateSettings.AllowEnvironmentOverride) flags |= RegionFlags.AllowEnvironmentOverride; return (uint)flags; } // Region handshake may need a more detailed look static private readonly byte[] RegionHandshakeHeader = new byte[] { Helpers.MSG_RELIABLE | Helpers.MSG_ZEROCODED, 0, 0, 0, 0, // sequence number 0, // extra //0xff, 0xff, 0, 148 // ID 148 (low frequency bigendian) 0xff, 0xff, 0, 1, 148 // ID 148 (low frequency bigendian) zero encoded }; public void SendRegionHandshake() { GetViewerCaps(); // make sure this is up to date RegionInfo regionInfo = m_scene.RegionInfo; RegionSettings regionSettings = regionInfo.RegionSettings; EstateSettings es = regionInfo.EstateSettings; bool isEstateManager = m_scene.Permissions.IsEstateManager(m_agentId); // go by oficial path uint regionFlags = GetRegionFlags(); UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(RegionHandshakeHeader, 0, buf.Data, 0, 11); // inline zeroencode LLUDPZeroEncoder zc = new(buf.Data); zc.Position = 11; //RegionInfo Block //RegionFlags U32 zc.AddUInt(regionFlags); //SimAccess U8 zc.AddByte(regionInfo.AccessLevel); //SimName zc.AddShortString(regionInfo.RegionName, 255); //SimOwner zc.AddUUID(es.EstateOwner); //IsEstateManager zc.AddByte((byte)(isEstateManager ? 1 : 0)); //WaterHeight zc.AddFloat((float)regionSettings.WaterHeight); // why is this a double ?? //BillableFactor zc.AddFloat(es.BillableFactor); //CacheID zc.AddUUID(regionSettings.CacheID); //TerrainBase0 //TerrainBase1 //TerrainBase2 //TerrainBase3 // this seem now obsolete, sending zero uuids // we should send the basic low resolution default ? zc.AddZeros(16 * 4); if(SupportTerrainPBR) { //TerrainDetail0 zc.AddUUID(regionSettings.TerrainPBR1); //TerrainDetail1 zc.AddUUID(regionSettings.TerrainPBR2); //TerrainDetail2 zc.AddUUID(regionSettings.TerrainPBR3); //TerrainDetail3 zc.AddUUID(regionSettings.TerrainPBR4); ViewerFlags |= ViewerFlags.SentTPBR; } else { //TerrainDetail0 zc.AddUUID(regionSettings.TerrainTexture1); //TerrainDetail1 zc.AddUUID(regionSettings.TerrainTexture2); //TerrainDetail2 zc.AddUUID(regionSettings.TerrainTexture3); //TerrainDetail3 zc.AddUUID(regionSettings.TerrainTexture4); } //TerrainStartHeight00 zc.AddFloat((float)regionSettings.Elevation1SW); //TerrainStartHeight01 zc.AddFloat((float)regionSettings.Elevation1NW); //TerrainStartHeight10 zc.AddFloat((float)regionSettings.Elevation1SE); //TerrainStartHeight11 zc.AddFloat((float)regionSettings.Elevation1NE); //TerrainHeightRange00 zc.AddFloat((float)regionSettings.Elevation2SW); //TerrainHeightRange01 zc.AddFloat((float)regionSettings.Elevation2NW); //TerrainHeightRange10 zc.AddFloat((float)regionSettings.Elevation2SE); //TerrainHeightRange11 zc.AddFloat((float)regionSettings.Elevation2NE); //RegionInfo2 block //region ID zc.AddUUID(regionInfo.RegionID); //RegionInfo3 block //CPUClassID zc.AddInt(9); //CPURatio zc.AddInt(1); // ColoName (string) // ProductSKU (string) // both empty strings zc.AddZeros(2); //ProductName zc.AddShortString(regionInfo.RegionType, 255); //RegionInfo4 block //RegionFlagsExtended //zc.AddZeros(1); // if we dont have this else zc.AddByte(1); zc.AddUInt64(regionFlags); // we have nothing other base flags //RegionProtocols // bit 0 signals server side texture baking // bit 63 signals more than 6 baked textures support" zc.AddUInt64(1UL << 63); buf.DataLength = zc.Finish(); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Unknown); } static private readonly byte[] AgentMovementCompleteHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 250 // ID 250 (low frequency bigendian) }; public unsafe void MoveAgentIntoRegion(RegionInfo regInfo, Vector3 pos, Vector3 look) { // reset agent update args m_thisAgentUpdateArgs.CameraAtAxis.X = float.MinValue; m_thisAgentUpdateArgs.lastUpdateTS = 0; m_thisAgentUpdateArgs.lastMoveUpdateTS = 0; m_thisAgentUpdateArgs.ControlFlags = 0; UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] bdata = buf.Data; //setup header Buffer.BlockCopy(AgentMovementCompleteHeader, 0, bdata, 0, 10); fixed(byte* data = bdata) { //AgentData block m_agentId.ToBytes(data + 10); // 26 m_sessionId.ToBytes(data + 26); // 42 //Data block if ((pos.X == 0) && (pos.Y == 0) && (pos.Z == 0)) m_startpos.ToBytes(data + 42); //54 else pos.ToBytes(data + 42); //54 look.ToBytes(data + 54); // 66 //Utils.UInt64ToBytesSafepos(regInfo.RegionHandle, data + 66); // 74 //Utils.UIntToBytesSafepos((uint)Util.UnixTimeSinceEpoch(), data + 74); //78 Utils.UInt64ToBytes(regInfo.RegionHandle, data + 66); // 74 Utils.IntToBytes((int)Util.UnixTimeSinceEpoch(), data + 74); //78 //SimData int len = m_regionChannelVersion.Length; if(len == 0) { data[78] = 0; data[79] = 0; } else { data[78] = (byte)len; data[79] = (byte)(len >> 8); Buffer.BlockCopy(m_regionChannelVersion, 0, bdata, 80, len); } buf.DataLength = 80 + len; } m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Unknown); } static private readonly byte[] ChatFromSimulatorHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 139 // ID 139 (low frequency bigendian) }; public void SendChatMessage(string message, byte chattype, Vector3 fromPos, string fromName, UUID sourceID, UUID ownerID, byte sourcetype, byte audible) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(ChatFromSimulatorHeader, 0, data, 0, 10); int pos = 11; int len = Util.osUTF8Getbytes(fromName, data, 11, 255, true); data[10] = (byte)len; if (len > 0) pos += len; sourceID.ToBytes(data, pos); pos += 16; ownerID.ToBytes(data, pos); pos += 16; data[pos++] = sourcetype; data[pos++] = chattype; data[pos++] = audible; fromPos.ToBytes(data, pos); pos += 12; byte[] msg = Util.StringToBytes1024(message); len = msg.Length; if (len == 0) { data[pos++] = 0; data[pos++] = 0; } else { data[pos++] = (byte)len; data[pos++] = (byte)(len >> 8); Buffer.BlockCopy(msg, 0, data, pos, len); pos += len; } buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Unknown); } /// /// Send an instant message to this client /// // static private readonly byte[] ImprovedInstantMessageHeader = new byte[] { Helpers.MSG_RELIABLE, //| Helpers.MSG_ZEROCODED, not doing spec zeroencode on this 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 254 // ID 139 (low frequency bigendian) }; public void SendInstantMessage(GridInstantMessage im) { UUID fromAgentID = new(im.fromAgentID); UUID toAgentID = new(im.toAgentID); if (!m_scene.Permissions.CanInstantMessage(fromAgentID, toAgentID)) return; UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(ImprovedInstantMessageHeader, 0, data, 0, 10); //agentdata block fromAgentID.ToBytes(data, 10); // 26 UUID.Zero.ToBytes(data, 26); // 42 sessionID zero?? TO check int pos = 42; //MessageBlock data[pos++] = (byte)((im.fromGroup) ? 1 : 0); toAgentID.ToBytes(data, pos); pos += 16; Utils.UIntToBytesSafepos(im.ParentEstateID, data, pos); pos += 4; (new UUID(im.RegionID)).ToBytes(data, pos); pos += 16; (im.Position).ToBytes(data, pos); pos += 12; data[pos++] = im.offline; data[pos++] = im.dialog; // this is odd if (im.imSessionID == UUID.Zero.Guid) (fromAgentID ^ toAgentID).ToBytes(data, pos); else (new UUID(im.imSessionID)).ToBytes(data, pos); pos += 16; Utils.UIntToBytesSafepos(im.timestamp, data, pos); pos += 4; int len = Util.osUTF8Getbytes(im.fromAgentName, data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; len = Util.osUTF8Getbytes(im.message, data, pos + 2, 1024, true); if (len == 0) { data[pos++] = 0; data[pos++] = 0; } else { data[pos++] = (byte)len; data[pos++] = (byte)(len >> 8); pos += len; } byte[] tmp = im.binaryBucket; if(tmp is null) { data[pos++] = 0; data[pos++] = 0; } else { len = tmp.Length; if (len == 0) { data[pos++] = 0; data[pos++] = 0; } else { data[pos++] = (byte)len; data[pos++] = (byte)(len >> 8); Buffer.BlockCopy(tmp, 0, data, pos, len); pos += len; } } //EstateBlock does not seem in use TODO //Utils.UIntToBytesSafepos(m_scene.RegionInfo.EstateSettings.EstateID, data, pos); pos += 4; data[pos++] = 0; data[pos++] = 0; data[pos++] = 0; data[pos++] = 0; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Unknown, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Unknown); } static private readonly byte[] GenericMessageHeader = new byte[] { Helpers.MSG_RELIABLE, //| Helpers.MSG_ZEROCODED, not doing spec zeroencode on this 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 1, 5 // ID 261 (low frequency bigendian) }; public void SendGenericMessage(string method, UUID invoice, List message) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(GenericMessageHeader, 0, data, 0, 10); //agentdata block m_agentId.ToBytes(data, 10); // 26 m_sessionId.ToBytes(data, 26); // 42 sessionID zero?? TO check UUID.Zero.ToBytes(data, 42); // 58 int pos = 58; //method block int len = Util.osUTF8Getbytes(method, data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; invoice.ToBytes(data, pos); pos += 16; //ParamList block if (message.Count == 0) { data[pos++] = 0; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); return; } int countpos = pos; ++pos; int count = 0; for(int indx = 0; indx < message.Count; ++indx) { len = Util.osUTF8Getbytes(message[indx], data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; if (pos > LLUDPServer.MAXPAYLOAD - 100) { data[countpos] = (byte)count; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); UDPPacketBuffer newbuf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(data, 0, newbuf.Data, 0, countpos); buf = newbuf; data = buf.Data; pos = countpos + 1; count = 1; } else ++count; } if (count > 0) { data[countpos] = (byte)count; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } } public void SendGenericMessage(string method, UUID invoice, List message) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(GenericMessageHeader, 0, data, 0, 10); //agentdata block m_agentId.ToBytes(data, 10); // 26 m_sessionId.ToBytes(data, 26); // 42 sessionID zero?? TO check UUID.Zero.ToBytes(data, 42); // 58 int pos = 58; //method block int len = Util.osUTF8Getbytes(method, data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; invoice.ToBytes(data, pos); pos += 16; //ParamList block if (message.Count == 0) { data[pos++] = 0; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); return; } int countpos = pos; ++pos; byte[] val; int count = 0; for (int indx = 0; indx < message.Count; ++indx) { val = message[indx]; len = val.Length; if(len > 255) len = 255; if (pos + len >= LLUDPServer.MAXPAYLOAD) { data[countpos] = (byte)count; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); UDPPacketBuffer newbuf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(data, 0, newbuf.Data, 0, countpos); buf = newbuf; data = buf.Data; pos = countpos + 1; count = 1; } else ++count; data[pos++] = (byte)len; if (len > 0) Buffer.BlockCopy(val, 0, data, pos, len); pos += len; } if (count > 0) { data[countpos] = (byte)count; buf.DataLength = pos; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } } public void SendGroupActiveProposals(UUID groupID, UUID transactionID, GroupActiveProposals[] Proposals) { /* not in use and broken GroupActiveProposalItemReplyPacket GAPIRP = new(); GAPIRP.AgentData.AgentID = m_agentId; GAPIRP.AgentData.GroupID = groupID; GAPIRP.TransactionData.TransactionID = transactionID; if (Proposals.Length == 0) { GAPIRP.TransactionData.TotalNumItems = 1; GAPIRP.ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock[] { new GroupActiveProposalItemReplyPacket.ProposalDataBlock() { VoteCast = Utils.StringToBytes("false"), VoteID = UUID.Zero, VoteInitiator = UUID.Zero, Majority = 0, Quorum = 0, TerseDateID = Array.Empty(), StartDateTime = Array.Empty(), EndDateTime = Array.Empty(), ProposalText = Array.Empty(), AlreadyVoted = false } }; } else { GAPIRP.TransactionData.TotalNumItems = (uint)Proposals.Length; GAPIRP.ProposalData = new GroupActiveProposalItemReplyPacket.ProposalDataBlock[Proposals.Length]; int i = 0; foreach (GroupActiveProposals Proposal in Proposals) { GAPIRP.ProposalData[i++] = new GroupActiveProposalItemReplyPacket.ProposalDataBlock() { VoteCast = Utils.StringToBytes("false"), VoteID = new UUID(Proposal.VoteID), VoteInitiator = new UUID(Proposal.VoteInitiator), Majority = Convert.ToSingle(Proposal.Majority), Quorum = Convert.ToInt32(Proposal.Quorum), TerseDateID = Utils.StringToBytes(Proposal.TerseDateID), StartDateTime = Utils.StringToBytes(Proposal.StartDateTime), EndDateTime = Utils.StringToBytes(Proposal.EndDateTime), ProposalText = Utils.StringToBytes(Proposal.ProposalText), AlreadyVoted = false }; } } OutPacket(GAPIRP, ThrottleOutPacketType.Task); */ } public void SendGroupVoteHistory(UUID groupID, UUID transactionID, GroupVoteHistory[] Votes) { /* not in use and broken GroupVoteHistoryItemReplyPacket GVHIRP = new GroupVoteHistoryItemReplyPacket(); GVHIRP.AgentData.AgentID = m_agentId; GVHIRP.AgentData.GroupID = groupID; GVHIRP.TransactionData.TransactionID = transactionID; if (Votes.Length == 0) { 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 = Array.Empty(); GVHIRP.HistoryItemData.StartDateTime = Array.Empty(); GVHIRP.HistoryItemData.EndDateTime = Array.Empty(); GVHIRP.HistoryItemData.VoteType = Array.Empty(); GVHIRP.HistoryItemData.VoteResult = Array.Empty(); GVHIRP.HistoryItemData.ProposalText = Array.Empty(); GVHIRP.VoteItem = new GroupVoteHistoryItemReplyPacket.VoteItemBlock[] { new GroupVoteHistoryItemReplyPacket.VoteItemBlock() { CandidateID = UUID.Zero, NumVotes = 0, //TODO: FIX THIS!!! VoteCast = Utils.StringToBytes("No") } }; } else { GVHIRP.TransactionData.TotalNumItems = (uint)Votes.Length; int i = 0; foreach (GroupVoteHistory Vote in Votes) { 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; i++; } } OutPacket(GVHIRP, ThrottleOutPacketType.Task); */ } public void SendGroupAccountingDetails(IClientAPI sender,UUID groupID, UUID transactionID, UUID sessionID, int amt) { GroupAccountDetailsReplyPacket GADRP = new(); GADRP.AgentData.AgentID = sender.AgentId; GADRP.AgentData.GroupID = groupID; GADRP.MoneyData.CurrentInterval = 0; GADRP.MoneyData.IntervalDays = 7; GADRP.MoneyData.RequestID = transactionID; GADRP.MoneyData.StartDate = Utils.StringToBytes(DateTime.Today.ToString()); GADRP.HistoryData = new GroupAccountDetailsReplyPacket.HistoryDataBlock[] { new GroupAccountDetailsReplyPacket.HistoryDataBlock() { Amount = amt, Description = Array.Empty() } }; 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.AgentID = sender.AgentId; GASRP.AgentData.GroupID = groupID; 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) { // bad ! GroupAccountTransactionsReplyPacket GATRP = (GroupAccountTransactionsReplyPacket)PacketPool.Instance.GetPacket( PacketType.GroupAccountTransactionsReply); GATRP.AgentData.AgentID = sender.AgentId; GATRP.AgentData.GroupID = groupID; 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[] { new GroupAccountTransactionsReplyPacket.HistoryDataBlock() { Amount = amt, Item = Array.Empty(), Time = Array.Empty(), Type = 0, User = Array.Empty(), } }; OutPacket(GATRP, ThrottleOutPacketType.Task); } public virtual bool CanSendLayerData() { int n = m_udpClient.GetPacketsQueuedCount(ThrottleOutPacketType.Land); return n <= 128; } /// /// 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 is 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 = OpenSimUDPBase.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(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 > 900 && i != numberPatchs - 1) { //finish this packet bitpack.PackBitsFromByte(END_OF_PATCHES); // fix the datablock length 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); // start another buf = OpenSimUDPBase.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); } catch (Exception e) { m_log.Error("[CLIENT]: SendLayerData() Failed with exception: " + e.Message, e); } } private static 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 readonly static Dictionary lastWindVersion = new(); private readonly static Dictionary> lastWindPackets = new(); /// /// 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); } /// /// Tell the client that the given neighbour region is ready to receive a child agent. /// public virtual void InformClientOfNeighbour(ulong neighbourHandle, IPEndPoint neighbourEndPoint) { byte[] byteIP = neighbourEndPoint.Address.GetAddressBytes(); uint neighbourIP = (uint)byteIP[3] << 24 | (uint)byteIP[2] << 16 | (uint)byteIP[1] << 8 | (uint)byteIP[0]; ushort neighbourPort = (ushort)neighbourEndPoint.Port; EnableSimulatorPacket enablesimpacket = new(); enablesimpacket.SimulatorInfo.Handle = neighbourHandle; enablesimpacket.SimulatorInfo.IP = neighbourIP; enablesimpacket.SimulatorInfo.Port = neighbourPort; enablesimpacket.Header.Reliable = true; // ESP's should be reliable. OutPacket(enablesimpacket, ThrottleOutPacketType.Task); } public AgentCircuitData RequestClientInfo() { AgentCircuitData agentData = new() { AgentID = m_agentId, SessionID = m_sessionId, SecureSessionID = SecureSessionId, circuitcode = m_circuitCode, child = false, firstname = m_firstName, lastname = m_lastName }; ICapabilitiesModule capsModule = m_scene.RequestModuleInterface(); if (capsModule is not null) // can happen when shutting down. { 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(lookAt.X * 10, lookAt.Y * 10, lookAt.Z * 10); byte[] byteIP = externalIPEndPoint.Address.GetAddressBytes(); uint externalIP = (uint)byteIP[3] << 24 | (uint)byteIP[2] << 16 | (uint)byteIP[1] << 8 | (uint)byteIP[0]; CrossedRegionPacket newSimPack = new(); newSimPack.AgentData.AgentID = m_agentId; newSimPack.AgentData.SessionID = m_sessionId; newSimPack.Info.Position = pos; newSimPack.Info.LookAt = look; newSimPack.RegionData.RegionHandle = newRegionHandle; newSimPack.RegionData.SimIP = externalIP; 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 = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header and agentinfo block Buffer.BlockCopy(MapBlockItemHeader, 0, data, 0, 10); m_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; 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; int len = Util.osUTF8Getbytes(mr.name, data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; if (pos < capacity) ++count; else { // prepare next packet UDPPacketBuffer newbuf = OpenSimUDPBase.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); 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); } } 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; MapBlockData md; for (int indx = 0; indx < mapBlocks.Count; ++indx) { md = mapBlocks[indx]; ut = md.SizeX; sizes[count++] = ut; if (ut > 256) needSizes = true; ut = md.SizeY; sizes[count++] = ut; if (ut > 256) needSizes = true; } UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header and agentinfo block Buffer.BlockCopy(MapBlockReplyHeader, 0, data, 0, 10); m_agentId.ToBytes(data, 10); // 26 Utils.UIntToBytesSafepos(flags, data, 26); // 30 int countpos = 30; int pos = 31; int lastpos; int capacity = LLUDPServer.MAXPAYLOAD - pos; count = 0; for (int indx = 0; indx < mapBlocks.Count; ++indx) { md = mapBlocks[indx]; lastpos = pos; Utils.UInt16ToBytes(md.X, data, pos); pos += 2; Utils.UInt16ToBytes(md.Y, data, pos); pos += 2; int len = Util.osUTF8Getbytes(md.Name, data, pos + 1, 255, true); data[pos++] = (byte)len; if (len > 0) pos += len; 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 = OpenSimUDPBase.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); 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); } } public void SendLocalTeleport(Vector3 position, Vector3 lookAt, uint flags) { TeleportLocalPacket tpLocal = (TeleportLocalPacket)PacketPool.Instance.GetPacket(PacketType.TeleportLocal); tpLocal.Info.AgentID = m_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 = new(); teleport.Info.AgentID = m_agentId; teleport.Info.RegionHandle = regionHandle; teleport.Info.SimAccess = simAccess; teleport.Info.SeedCapability = Util.StringToBytes256(capsURL); byte[] byteIP = newRegionEndPoint.Address.GetAddressBytes(); teleport.Info.SimIP = (uint)byteIP[3] << 24 | (uint)byteIP[2] << 16 | (uint)byteIP[1] << 8 | (uint)byteIP[0]; 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 = m_agentId; tpFailed.Info.Reason = Util.StringToBytes256(reason); tpFailed.AlertInfo = Array.Empty(); // 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 = m_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 = m_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 = new(); payPriceReply.ObjectData.ObjectID = objectID; payPriceReply.ObjectData.DefaultPayPrice = payPrice[0]; payPriceReply.ButtonData = new PayPriceReplyPacket.ButtonDataBlock[] { new PayPriceReplyPacket.ButtonDataBlock { PayButton = payPrice[1] }, new PayPriceReplyPacket.ButtonDataBlock { PayButton = payPrice[2] }, new PayPriceReplyPacket.ButtonDataBlock { PayButton = payPrice[3] }, new PayPriceReplyPacket.ButtonDataBlock { 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. m_entityProps.Remove(localIDs); 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 { 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, int descendents, 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, descendents, 0, 0); // To preserve SL compatibility, we will NOT combine folders and items in one packet // while (itemsSent < totalItems || foldersSent < totalFolders) { if (currentPacket is 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, descendents, 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 static InventoryDescendentsPacket.FolderDataBlock CreateFolderDataBlock(InventoryFolderBase folder) { InventoryDescendentsPacket.FolderDataBlock newBlock = new() { FolderID = folder.ID, Name = Util.StringToBytes256(folder.Name), ParentID = folder.ParentID, Type = (sbyte)folder.Type }; //if (newBlock.Type == InventoryItemBase.SUITCASE_FOLDER_TYPE) // newBlock.Type = InventoryItemBase.SUITCASE_FOLDER_FAKE_TYPE; return newBlock; } private static InventoryDescendentsPacket.ItemDataBlock CreateItemDataBlock(InventoryItemBase item) { InventoryDescendentsPacket.ItemDataBlock newBlock = new() { ItemID = item.ID, AssetID = item.AssetID, CreatorID = item.CreatorIdAsUuid, BaseMask = item.BasePermissions, Description = Util.StringToBytes256(item.Description), EveryoneMask = item.EveryOnePermissions, OwnerMask = item.CurrentPermissions, FolderID = item.Folder, InvType = (sbyte)item.InvType, Name = Util.StringToBytes256(item.Name), NextOwnerMask = item.NextPermissions, OwnerID = item.Owner, Type = (sbyte)item.AssetType, GroupID = item.GroupID, GroupOwned = item.GroupOwned, GroupMask = item.GroupPermissions, CreationDate = item.CreationDate, SalePrice = item.SalePrice, SaleType = item.SaleType, 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 static void AddNullFolderBlockToDecendentsPacket(ref InventoryDescendentsPacket packet) { packet.FolderData = new InventoryDescendentsPacket.FolderDataBlock[] { new InventoryDescendentsPacket.FolderDataBlock { FolderID = UUID.Zero, ParentID = UUID.Zero, Type = -1, Name = Array.Empty() } }; } private static void AddNullItemBlockToDescendentsPacket(ref InventoryDescendentsPacket packet) { packet.ItemData = new InventoryDescendentsPacket.ItemDataBlock[] { new InventoryDescendentsPacket.ItemDataBlock { ItemID = UUID.Zero, AssetID = UUID.Zero, CreatorID = UUID.Zero, BaseMask = 0, Description = Array.Empty(), EveryoneMask = 0, OwnerMask = 0, FolderID = UUID.Zero, InvType = (sbyte)0, Name = Array.Empty(), NextOwnerMask = 0, OwnerID = UUID.Zero, Type = -1, GroupID = UUID.Zero, GroupOwned = false, GroupMask = 0, CreationDate = 0, SalePrice = 0, SaleType = 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 = m_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(InventoryItemBase[] items) { // 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); inventoryReply.AgentData.AgentID = m_agentId; int total = items.Length; int count = 0; for(int i = 0; i < items.Length; ++i) { if(count == 0) { if(total < 10) { inventoryReply.InventoryData = new FetchInventoryReplyPacket.InventoryDataBlock[total]; total = 0; } else { inventoryReply.InventoryData = new FetchInventoryReplyPacket.InventoryDataBlock[10]; total -= 10; } } inventoryReply.InventoryData[count] = new FetchInventoryReplyPacket.InventoryDataBlock(); FetchInventoryReplyPacket.InventoryDataBlock data = inventoryReply.InventoryData[count]; data.ItemID = items[i].ID; data.AssetID = items[i].AssetID; data.CreatorID = items[i].CreatorIdAsUuid; data.BaseMask = items[i].BasePermissions; data.CreationDate = items[i].CreationDate; data.Description = Util.StringToBytes256(items[i].Description); data.EveryoneMask = items[i].EveryOnePermissions; data.FolderID = items[i].Folder; data.InvType = (sbyte)items[i].InvType; data.Name = Util.StringToBytes256(items[i].Name); data.NextOwnerMask = items[i].NextPermissions; data.OwnerID = items[i].Owner; data.OwnerMask = items[i].CurrentPermissions; data.Type = (sbyte)items[i].AssetType; data.GroupID = items[i].GroupID; data.GroupOwned = items[i].GroupOwned; data.GroupMask = items[i].GroupPermissions; data.Flags = items[i].Flags; data.SalePrice = items[i].SalePrice; data.SaleType = items[i].SaleType; data.CRC = Helpers.InventoryCRC( 1000, 0, data.InvType, data.Type, data.AssetID, data.GroupID, 100, data.OwnerID, data.CreatorID, data.ItemID, data.FolderID, FULL_MASK_PERMISSIONS, 1, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS, FULL_MASK_PERMISSIONS); ++count; if(count == 10 || total == 0) { inventoryReply.Header.Zerocoded = true; OutPacket(inventoryReply, ThrottleOutPacketType.Asset); if(total == 0) break; count = 0; } } } protected void SendBulkUpdateInventoryFolder(InventoryFolderBase folderBase, UUID? transationID) { // We will use the same transaction id for all the separate packets to be sent out in this update. UUID transactionId = transationID ?? UUID.Random(); List folderDataBlocks = new(); 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 = m_agentId; bulkUpdate.AgentData.TransactionID = transactionId; bulkUpdate.FolderData = folderDataBlocks.ToArray(); bulkUpdate.ItemData = Array.Empty(); //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(m_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 = m_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[^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; for (int indx = 0; indx < subFolders.Count; ++indx) { SendBulkUpdateInventoryFolderRecursive(subFolders[indx], ref folderDataBlocks, transactionId); } } /// /// Generate a bulk update inventory data block for the given folder /// /// /// private static BulkUpdateInventoryPacket.FolderDataBlock GenerateBulkUpdateFolderDataBlock(InventoryFolderBase folder) { BulkUpdateInventoryPacket.FolderDataBlock folderBlock = new() { FolderID = folder.ID, ParentID = folder.ParentID, 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; Name = Util.StringToBytes256(folder.Name) }; return folderBlock; } /// /// Generate a bulk update inventory data block for the given item /// /// /// private static BulkUpdateInventoryPacket.ItemDataBlock GenerateBulkUpdateItemDataBlock(InventoryItemBase item) { BulkUpdateInventoryPacket.ItemDataBlock itemBlock = new() { ItemID = item.ID, AssetID = item.AssetID, CreatorID = item.CreatorIdAsUuid, BaseMask = item.BasePermissions, Description = Util.StringToBytes256(item.Description), EveryoneMask = item.EveryOnePermissions, FolderID = item.Folder, InvType = (sbyte)item.InvType, Name = Util.StringToBytes256(item.Name), NextOwnerMask = item.NextPermissions, OwnerID = item.Owner, OwnerMask = item.CurrentPermissions, Type = (sbyte)item.AssetType, GroupID = item.GroupID, GroupOwned = item.GroupOwned, GroupMask = item.GroupPermissions, Flags = item.Flags & 0x2000ff, SalePrice = item.SalePrice, SaleType = item.SaleType, 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, UUID? transationID = null) { if (node is InventoryItemBase itbase) SendBulkUpdateInventoryItem(itbase, transationID); else if (node is InventoryFolderBase ftbase) SendBulkUpdateInventoryFolder(ftbase, transationID); else if (node is not null) m_log.Error($"[CLIENT]: {Name} sent unknown inventory node named {node.Name}"); else m_log.Error($"[CLIENT]: {Name} sent null inventory node"); } protected void SendBulkUpdateInventoryItem(InventoryItemBase item, UUID? transationID = null) { IEventQueue eq = Scene.RequestModuleInterface(); eq?.SendBulkUpdateInventoryItem(item, m_agentId, transationID); } 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 = m_agentId; InventoryReply.AgentData.SimApproved = true; InventoryReply.AgentData.TransactionID = transactionID; InventoryReply.InventoryData = new UpdateCreateInventoryItemPacket.InventoryDataBlock[] { new UpdateCreateInventoryItemPacket.InventoryDataBlock { ItemID = Item.ID, AssetID = Item.AssetID, CreatorID = Item.CreatorIdAsUuid, BaseMask = Item.BasePermissions, Description = Util.StringToBytes256(Item.Description), EveryoneMask = Item.EveryOnePermissions, FolderID = Item.Folder, InvType = (sbyte)Item.InvType, Name = Util.StringToBytes256(Item.Name), NextOwnerMask = Item.NextPermissions, OwnerID = Item.Owner, OwnerMask = Item.CurrentPermissions, Type = (sbyte)Item.AssetType, CallbackID = callbackId, GroupID = Item.GroupID, GroupOwned = Item.GroupOwned, GroupMask = Item.GroupPermissions, Flags = Item.Flags & 0x2000ff, SalePrice = Item.SalePrice, SaleType = Item.SaleType, 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 = m_agentId; remove.AgentData.SessionID = m_sessionId; remove.InventoryData = new RemoveInventoryItemPacket.InventoryDataBlock[] { new RemoveInventoryItemPacket.InventoryDataBlock { 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); scriptcontrol.Data = new ScriptControlChangePacket.DataBlock[] { new ScriptControlChangePacket.DataBlock { //ddata.Controls = adjustControls(controls); Controls = (uint)controls, PassToAgent = passToAgent, TakeControls = TakeControls } }; OutPacket(scriptcontrol, ThrottleOutPacketType.Task); } static private readonly byte[] ReplyTaskInventoryHeader = new byte[] { Helpers.MSG_RELIABLE, //| Helpers.MSG_ZEROCODED, not doing spec zeroencode on this 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 1, 34 // ID 90 (low frequency bigendian) }; public void SendTaskInventory(UUID taskID, short serial, byte[] fileName) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(ReplyTaskInventoryHeader, 0, data, 0, 10); taskID.ToBytes(data, 10); // 26 Utils.Int16ToBytes(serial, data, 26); // 28 data[28] = (byte)fileName.Length; if(data[28] > 0) Buffer.BlockCopy(fileName, 0, data, 29, data[28]); buf.DataLength = 29 + data[28]; //m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false, true); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } static private readonly byte[] SendXferPacketHeader = new byte[] { 0, //Helpers.MSG_RELIABLE, Xfer control must provide reliabialty 0, 0, 0, 0, // sequence number 0, // extra 18 // ID (high frequency bigendian) }; public void SendXferPacket(ulong xferID, uint packet, byte[] XferData, int XferDataOffset, int XferDatapktLen, bool isTaskInventory) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(SendXferPacketHeader, 0, data, 0, 7); Utils.UInt64ToBytesSafepos(xferID, data, 7); // 15 Utils.UIntToBytesSafepos(packet, data, 15); // 19 int len = XferDatapktLen; if (XferDataOffset == 0) // first packet needs to send the total xfer data len len += 4; if (len > LLUDPServer.MAXPAYLOAD) // should never happen len = LLUDPServer.MAXPAYLOAD; if (len == 0) { data[19] = 0; data[20] = 0; } else { data[19] = (byte)len; data[20] = (byte)(len >> 8); if(XferDataOffset == 0) { // need to send total xfer data len Utils.IntToBytesSafepos(XferData.Length, data, 21); if (XferDatapktLen > 0) Buffer.BlockCopy(XferData, XferDataOffset, data, 25, XferDatapktLen); } else Buffer.BlockCopy(XferData, XferDataOffset, data, 21, XferDatapktLen); } buf.DataLength = 21 + len; m_udpServer.SendUDPPacket(m_udpClient, buf, isTaskInventory ? ThrottleOutPacketType.Task : ThrottleOutPacketType.Asset); } static private readonly byte[] AbortXferHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 157 // ID 157 (low frequency bigendian) }; public void SendAbortXferPacket(ulong xferID) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(AbortXferHeader, 0, data, 0, 10); Utils.UInt64ToBytesSafepos(xferID, data, 10); // 18 Utils.IntToBytesSafepos(0, data, 18); // 22 reason TODO buf.DataLength = 22; m_udpServer.SendUDPPacket(m_udpClient, buf, 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); } static private readonly byte[] AvatarPickerReplyHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 28 // ID 28 (low frequency bigendian) }; public void SendAvatarPickerReply(UUID QueryID, List users) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(AvatarPickerReplyHeader, 0, data, 0, 10); m_agentId.ToBytes(data, 10); //26 QueryID.ToBytes(data, 26); //42 if (users.Count == 0) { data[42] = 0; buf.DataLength = 43; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); return; } int pos = 43; int count = 0; for(int u = 0; u < users.Count; ++u) { UserData user = users[u]; user.Id.ToBytes(data,pos); pos+= 16; byte[] tmp = Utils.StringToBytes(user.FirstName); data[pos++] = (byte)tmp.Length; if(tmp.Length > 0) { Buffer.BlockCopy(tmp, 0, data, pos, tmp.Length); pos += tmp.Length; } tmp = Utils.StringToBytes(user.LastName); data[pos++] = (byte)tmp.Length; if (tmp.Length > 0) { Buffer.BlockCopy(tmp, 0, data, pos, tmp.Length); pos += tmp.Length; } ++count; if (pos >= LLUDPServer.MAXPAYLOAD - 120) { data[42] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); if (u < users.Count - 1) { UDPPacketBuffer newbuf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] newdata = newbuf.Data; Buffer.BlockCopy(data, 0, newdata, 0, 42); buf = newbuf; data = newdata; pos = 43; } count = 0; } } if(count > 0) { data[42] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } } public void SendAgentDataUpdate(UUID agentid, UUID activegroupid, string firstname, string lastname, ulong grouppowers, string groupname, string grouptitle) { if (agentid.Equals(m_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.AlertData.Message = Util.StringToBytes256(message); alertPack.AlertInfo = Array.Empty(); OutPacket(alertPack, ThrottleOutPacketType.Task); } public void SendAlertMessage(string message, string info) { AlertMessagePacket alertPack = (AlertMessagePacket)PacketPool.Instance.GetPacket(PacketType.AlertMessage); alertPack.AlertData.Message = Util.StringToBytes256(message); alertPack.AlertInfo = new AlertMessagePacket.AlertInfoBlock[] { new AlertMessagePacket.AlertInfoBlock { Message = Util.StringToBytes256(info), ExtraParams = Array.Empty() } }; 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 = m_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 { ButtonLabel = Util.StringToBytes(buttonlabels[i], 24) }; } dialog.Buttons = buttons; dialog.OwnerData = new ScriptDialogPacket.OwnerDataBlock[1]; dialog.OwnerData[0] = new ScriptDialogPacket.OwnerDataBlock { 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 { ObjectID = objectID, OwnerID = ownerID, 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 SendViewerTime(Vector3 sunDir, float sunphase) { SimulatorViewerTimeMessagePacket viewertime = (SimulatorViewerTimeMessagePacket)PacketPool.Instance.GetPacket(PacketType.SimulatorViewerTimeMessage); viewertime.TimeInfo.UsecSinceStart = Util.UnixTimeSinceEpoch_uS(); viewertime.TimeInfo.SunDirection = sunDir; viewertime.TimeInfo.SunPhase = sunphase; viewertime.TimeInfo.SunAngVelocity = Vector3.Zero; //legacy viewertime.TimeInfo.SecPerDay = 14400; // legacy viewertime.TimeInfo.SecPerYear = 158400; // legacy viewertime.Header.Reliable = false; viewertime.Header.Zerocoded = true; OutPacket(viewertime, ThrottleOutPacketType.Task); } public void SendViewerEffect(ViewerEffectPacket.EffectBlock[] effectBlocks) { ViewerEffectPacket packet = (ViewerEffectPacket)PacketPool.Instance.GetPacket(PacketType.ViewerEffect); // packet.AgentData.AgentID = m_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 = m_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, m_agentId, 1, Message, false, new Vector3())); //SendInstantMessage(FromAvatarID, fromSessionID, Message, m_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 = m_agentId; logReply.AgentData.SessionID = m_sessionId; logReply.InventoryData = new LogoutReplyPacket.InventoryDataBlock[1]; logReply.InventoryData[0] = new LogoutReplyPacket.InventoryDataBlock { 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.AgentBlockBlock[] onpb = new OnlineNotificationPacket.AgentBlockBlock[agentIDs.Length]; for (int i = 0; i < agentIDs.Length; i++) { onpb[i] = new OnlineNotificationPacket.AgentBlockBlock { AgentID = agentIDs[i] }; } OnlineNotificationPacket onp = new() { AgentBlock = onpb }; onp.Header.Reliable = true; OutPacket(onp, ThrottleOutPacketType.Task); } public void SendAgentOffline(UUID[] agentIDs) { OfflineNotificationPacket offp = new(); OfflineNotificationPacket.AgentBlockBlock[] offpb = new OfflineNotificationPacket.AgentBlockBlock[agentIDs.Length]; for (int i = 0; i < agentIDs.Length; i++) { offpb[i] = new OfflineNotificationPacket.AgentBlockBlock { AgentID = agentIDs[i] }; } 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(); fap.AgentBlock.Hunter = HunterID; fap.AgentBlock.Prey = PreyID; fap.AgentBlock.SpaceIP = 0; fap.LocationBlock = new FindAgentPacket.LocationBlockBlock[] { new FindAgentPacket.LocationBlockBlock { GlobalX = GlobalX, 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(); 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(); respondPacket.AgentData.AgentID = m_agentId; respondPacket.AgentData.SessionID = m_sessionId; respondPacket.GrantData.GodLevel = (byte)AdminLevel; respondPacket.GrantData.Token = Token; OutPacket(respondPacket, ThrottleOutPacketType.Task); } public void SendGroupMembership(GroupMembershipData[] GroupMembership) { UpdateGroupMembership(GroupMembership); SendAgentGroupDataUpdate(m_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 is 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),m_agentId); */ } public void SendPartPhysicsProprieties(ISceneEntity entity) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) return; if (entity is not SceneObjectPart part) 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, m_agentId); } public void SendGroupNameReply(UUID groupLLUID, string GroupName) { UUIDGroupNameReplyPacket pack = new() { UUIDNameBlock = new UUIDGroupNameReplyPacket.UUIDNameBlockBlock[] { new UUIDGroupNameReplyPacket.UUIDNameBlockBlock() { ID = groupLLUID, GroupName = Util.StringToBytes256(GroupName) } } }; OutPacket(pack, ThrottleOutPacketType.Task); } public void SendLandStatReply(uint reportType, uint requestFlags, uint resultCount, LandStatReportItem[] lsrpia) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) { LandStatReplyPacket.ReportDataBlock[] lsrepdba = new LandStatReplyPacket.ReportDataBlock[lsrpia.Length]; for (int i = 0; i < lsrpia.Length; i++) { lsrepdba[i] = new LandStatReplyPacket.ReportDataBlock { LocationX = lsrpia[i].LocationX, LocationY = lsrpia[i].LocationY, LocationZ = lsrpia[i].LocationZ, Score = lsrpia[i].Score, TaskID = lsrpia[i].TaskID, TaskLocalID = lsrpia[i].TaskLocalID, TaskName = Util.StringToBytes256(lsrpia[i].TaskName), OwnerName = Util.StringToBytes256(lsrpia[i].OwnerName) }; } LandStatReplyPacket lsrp = new(); lsrp.RequestData.ReportType = reportType; lsrp.RequestData.RequestFlags = requestFlags; lsrp.RequestData.TotalObjectCount = resultCount; lsrp.ReportData = lsrepdba; OutPacket(lsrp, ThrottleOutPacketType.Task); } else { osUTF8 sb = eq.StartEvent("LandStatReply"); LLSDxmlEncode2.AddArrayAndMap("RequestData", sb); LLSDxmlEncode2.AddElem("ReportType", reportType, sb); LLSDxmlEncode2.AddElem("RequestFlags", requestFlags, sb); LLSDxmlEncode2.AddElem("TotalObjectCount", (uint)lsrpia.Length, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); if (lsrpia.Length > 0) { LLSDxmlEncode2.AddArray("ReportData", sb); foreach (var item in lsrpia) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("LocationX", item.LocationX, sb); LLSDxmlEncode2.AddElem("LocationY", item.LocationY, sb); LLSDxmlEncode2.AddElem("LocationZ", item.LocationZ, sb); LLSDxmlEncode2.AddElem("OwnerName", item.OwnerName, sb); LLSDxmlEncode2.AddElem("Score", item.Score, sb); LLSDxmlEncode2.AddElem("TaskID", item.TaskID, sb); LLSDxmlEncode2.AddElem("TaskLocalID", item.TaskLocalID, sb); LLSDxmlEncode2.AddElem("TaskName", item.TaskName, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); LLSDxmlEncode2.AddArray("DataExtended", sb); foreach (var item in lsrpia) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("MonoScore", 0.0f, sb); LLSDxmlEncode2.AddElem("OwnerID", item.OwnerID, sb); LLSDxmlEncode2.AddElem("ParcelName", item.Parcel, sb); LLSDxmlEncode2.AddElem("PublicURLs", item.Urls, sb); LLSDxmlEncode2.AddElem("Size", (float)item.Bytes, sb); LLSDxmlEncode2.AddElem("TimeStamp", item.Time, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); } eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } } public void SendScriptRunningReply(UUID objectID, UUID itemID, bool running) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) { ScriptRunningReplyPacket scriptRunningReply = new(); scriptRunningReply.Script.ObjectID = objectID; scriptRunningReply.Script.ItemID = itemID; scriptRunningReply.Script.Running = running; OutPacket(scriptRunningReply, ThrottleOutPacketType.Task); } else { eq.ScriptRunningEvent(objectID, itemID, running, m_agentId); } } public void SendAsset(AssetRequestToClient req) { if (req.AssetInf is null) { m_log.Error($"{LogHeader} Cannot send asset, because it is null"); return; } if (req.AssetInf.Data is null) { m_log.Error($"{LogHeader} Cannot send asset {req.AssetInf.ID} ({req.AssetInf.Metadata.ContentType}), asset data is null"); return; } TransferInfoPacket Transfer = new(); 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.Size = req.AssetInf.Data.Length; Transfer.TransferInfo.TransferID = req.TransferRequestID; Transfer.Header.Zerocoded = true; OutPacket(Transfer, ThrottleOutPacketType.Asset); if (req.NumPackets == 1) { TransferPacketPacket TransferPacket = new(); 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, 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(); 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, ThrottleOutPacketType.Asset); processedLength += chunkSize; packetNumber++; } } } public void SendAssetNotFound(AssetRequestToClient req) { TransferInfoPacket Transfer = new(); 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, 63); reply.Data.Desc = Utils.StringToBytes(land.Description, 255); 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 |= (1 << 7); if (land.IsGroupOwned) reply.Data.Flags |= 0x04; Vector3 pos = land.UserLocation; if (pos.IsZero()) { 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 = m_agentId; packet.QueryData[0].QueryID = queryID; DirPlacesReplyPacket.QueryRepliesBlock[] replies = Array.Empty(); DirPlacesReplyPacket.StatusDataBlock[] status = Array.Empty(); 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 = m_agentId; packet.QueryData[0].QueryID = queryID; replies = Array.Empty(); status = Array.Empty(); } } 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 { AgentID = m_agentId }; packet.QueryData = new DirPeopleReplyPacket.QueryDataBlock { QueryID = queryID }; packet.QueryReplies = new DirPeopleReplyPacket.QueryRepliesBlock[data.Length]; int i = 0; foreach (DirPeopleReplyData d in data) { packet.QueryReplies[i] = new DirPeopleReplyPacket.QueryRepliesBlock { AgentID = d.agentID, FirstName = Utils.StringToBytes(d.firstName), LastName = Utils.StringToBytes(d.lastName), Group = Utils.StringToBytes(d.group), Online = d.online, 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 { AgentID = m_agentId }; packet.QueryData = new DirEventsReplyPacket.QueryDataBlock { 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 { AgentID = m_agentId }; packet.QueryData = new DirGroupsReplyPacket.QueryDataBlock { QueryID = queryID }; packet.QueryReplies = new DirGroupsReplyPacket.QueryRepliesBlock[data.Length]; int i = 0; foreach (DirGroupsReplyData d in data) { packet.QueryReplies[i++] = new DirGroupsReplyPacket.QueryRepliesBlock { GroupID = d.groupID, GroupName = Util.StringToBytes(d.groupName, 35), Members = d.members, SearchOrder = d.searchOrder }; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirClassifiedReply(UUID queryID, DirClassifiedReplyData[] data) { DirClassifiedReplyPacket packet = (DirClassifiedReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirClassifiedReply); packet.AgentData = new DirClassifiedReplyPacket.AgentDataBlock { AgentID = m_agentId }; packet.QueryData = new DirClassifiedReplyPacket.QueryDataBlock { 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 { AgentID = m_agentId }; packet.QueryData = new DirLandReplyPacket.QueryDataBlock { QueryID = queryID }; packet.QueryReplies = new DirLandReplyPacket.QueryRepliesBlock[data.Length]; int i = 0; foreach (DirLandReplyData d in data) { packet.QueryReplies[i++] = new DirLandReplyPacket.QueryRepliesBlock { ParcelID = d.parcelID, Name = Utils.StringToBytes(d.name), Auction = d.auction, ForSale = d.forSale, SalePrice = d.salePrice, ActualArea = d.actualArea }; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendDirPopularReply(UUID queryID, DirPopularReplyData[] data) { DirPopularReplyPacket packet = (DirPopularReplyPacket)PacketPool.Instance.GetPacket(PacketType.DirPopularReply); packet.AgentData = new DirPopularReplyPacket.AgentDataBlock { AgentID = m_agentId }; packet.QueryData = new DirPopularReplyPacket.QueryDataBlock{ QueryID = queryID }; packet.QueryReplies = new DirPopularReplyPacket.QueryRepliesBlock[data.Length]; int i = 0; foreach (DirPopularReplyData d in data) { packet.QueryReplies[i++] = new DirPopularReplyPacket.QueryRepliesBlock { ParcelID = d.parcelID, Name = Utils.StringToBytes(d.name), Dwell = d.dwell }; } OutPacket(packet, ThrottleOutPacketType.Task); } public void SendEventInfoReply(EventData data) { EventInfoReplyPacket packet = (EventInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.EventInfoReply); packet.AgentData = new EventInfoReplyPacket.AgentDataBlock { AgentID = m_agentId }; packet.EventData = new EventInfoReplyPacket.EventDataBlock { EventID = data.eventID, Creator = Utils.StringToBytes(data.creator), Name = Utils.StringToBytes(data.name), Category = Utils.StringToBytes(data.category), Desc = Utils.StringToBytes(data.description), Date = Utils.StringToBytes(data.date), DateUTC = data.dateUTC, Duration = data.duration, Cover = data.cover, Amount = data.amount, SimName = Utils.StringToBytes(data.simName), GlobalPos = new Vector3d(data.globalPos), 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 = m_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 = m_agentId; p.AgentData.SessionID = UUID.Zero; p.FolderData = new AcceptCallingCardPacket.FolderDataBlock[1]; p.FolderData[0] = new AcceptCallingCardPacket.FolderDataBlock { FolderID = UUID.Zero }; OutPacket(p, ThrottleOutPacketType.Task); } public void SendDeclineCallingCard(UUID transactionID) { DeclineCallingCardPacket p = (DeclineCallingCardPacket)PacketPool.Instance.GetPacket(PacketType.DeclineCallingCard); p.AgentData.AgentID = m_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 = m_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 is null) return; // message template has a GroupData field AcceptNotices ignored by viewers // and a array NewGroupData also ignored osUTF8 sb = eq.StartEvent("AvatarGroupsReply"); LLSDxmlEncode2.AddArrayAndMap("AgentData", sb); LLSDxmlEncode2.AddElem("AgentID", m_agentId, sb); LLSDxmlEncode2.AddElem("AvatarID", avatarID, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); bool notSameAvatar = avatarID != m_agentId; if(data.Length == 0) LLSDxmlEncode2.AddEmptyArray("GroupData", sb); else { LLSDxmlEncode2.AddArray("GroupData", sb); GroupMembershipData m; for (int indx = 0; indx < data.Length; ++indx) { m = data[indx]; if(notSameAvatar && !m.ListInProfile) continue; LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("GroupPowers", m.GroupPowers, sb); LLSDxmlEncode2.AddElem("GroupTitle", m.GroupTitle, sb); LLSDxmlEncode2.AddElem("GroupID",m.GroupID, sb); LLSDxmlEncode2.AddElem("GroupName", m.GroupName, sb); LLSDxmlEncode2.AddElem("GroupInsigniaID", m.GroupPicture, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); } eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } public void SendAgentGroupDataUpdate(UUID avatarID, GroupMembershipData[] data) { if(avatarID != m_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.GroupDataBlock[] Groups = new AgentGroupDataUpdatePacket.GroupDataBlock[data.Length]; for (int i = 0; i < data.Length; ++i) { AgentGroupDataUpdatePacket.GroupDataBlock Group = new() { AcceptNotices = data[i].AcceptNotices, Contribution = data[i].Contribution, GroupID = data[i].GroupID, GroupInsigniaID = data[i].GroupPicture, GroupName = Util.StringToBytes256(data[i].GroupName), GroupPowers = data[i].GroupPowers }; Groups[i] = Group; } Groupupdate.GroupData = Groups; Groupupdate.AgentData = new AgentGroupDataUpdatePacket.AgentDataBlock { 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 { AgentID = m_agentId }; p.GroupData = new JoinGroupReplyPacket.GroupDataBlock { GroupID = groupID, 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 { AgentID = agentID }; p.GroupData = new EjectGroupMemberReplyPacket.GroupDataBlock { GroupID = groupID }; p.EjectData = new EjectGroupMemberReplyPacket.EjectDataBlock { 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 { AgentID = m_agentId }; p.GroupData = new LeaveGroupReplyPacket.GroupDataBlock { GroupID = groupID, 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 { AgentID = m_agentId, 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) { // fix classifiedFlags maturity if((classifiedFlags & 0x4e) == 0) // if none classifiedFlags |= 0x4; // pg ClassifiedInfoReplyPacket cr = (ClassifiedInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.ClassifiedInfoReply); cr.AgentData = new ClassifiedInfoReplyPacket.AgentDataBlock { AgentID = m_agentId }; cr.Data = new ClassifiedInfoReplyPacket.DataBlock { ClassifiedID = classifiedID, CreatorID = creatorID, CreationDate = creationDate, ExpirationDate = expirationDate, Category = category, Name = Utils.StringToBytes(name), Desc = Utils.StringToBytes(description), ParcelID = parcelID, ParentEstate = parentEstate, SnapshotID = snapshotID, SimName = Utils.StringToBytes(simName), PosGlobal = new Vector3d(globalPos), ParcelName = Utils.StringToBytes(parcelName), ClassifiedFlags = classifiedFlags, PriceForListing = price }; OutPacket(cr, ThrottleOutPacketType.Task); } public void SendAgentDropGroup(UUID groupID) { AgentDropGroupPacket dg = (AgentDropGroupPacket)PacketPool.Instance.GetPacket(PacketType.AgentDropGroup); dg.AgentData = new AgentDropGroupPacket.AgentDataBlock { AgentID = m_agentId, 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 { AgentID = m_agentId }; an.Data = new AvatarNotesReplyPacket.DataBlock { TargetID = targetID, 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 { AgentID = m_agentId, TargetID = targetID }; ap.Data = new AvatarPicksReplyPacket.DataBlock[picks.Count]; int i = 0; foreach (KeyValuePair pick in picks) { ap.Data[i++] = new AvatarPicksReplyPacket.DataBlock { PickID = pick.Key, PickName = Utils.StringToBytes(pick.Value) }; } OutPacket(ap, ThrottleOutPacketType.Task); } public void SendAvatarClassifiedReply(UUID targetID, Dictionary classifieds) { AvatarClassifiedReplyPacket ac = (AvatarClassifiedReplyPacket)PacketPool.Instance.GetPacket(PacketType.AvatarClassifiedReply); ac.AgentData = new AvatarClassifiedReplyPacket.AgentDataBlock { AgentID = m_agentId, TargetID = targetID }; ac.Data = new AvatarClassifiedReplyPacket.DataBlock[classifieds.Count]; int i = 0; foreach (KeyValuePair classified in classifieds) { ac.Data[i++] = new AvatarClassifiedReplyPacket.DataBlock { ClassifiedID = classified.Key, Name = Utils.StringToBytes(classified.Value) }; } 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 { AgentID = m_agentId }; pd.Data = new ParcelDwellReplyPacket.DataBlock { LocalID = localID, ParcelID = parcelID, Dwell = dwell }; OutPacket(pd, ThrottleOutPacketType.Land); } public void SendUserInfoReply(bool imViaEmail, bool visible, string email) { UserInfoReplyPacket ur = (UserInfoReplyPacket)PacketPool.Instance.GetPacket(PacketType.UserInfoReply); ur.AgentData = new UserInfoReplyPacket.AgentDataBlock { AgentID = m_agentId }; ur.UserData = new UserInfoReplyPacket.UserDataBlock { IMViaEMail = imViaEmail, DirectoryVisibility = Utils.StringToBytes(visible ? "default" : "hidden"), 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 { AgentID = m_agentId }; createGroupReply.ReplyData = new CreateGroupReplyPacket.ReplyDataBlock { GroupID = groupID, Success = success, Message = Utils.StringToBytes(message) }; OutPacket(createGroupReply, ThrottleOutPacketType.Task); } public void SendUseCachedMuteList() { UseCachedMuteListPacket useCachedMuteList = (UseCachedMuteListPacket)PacketPool.Instance.GetPacket(PacketType.UseCachedMuteList); useCachedMuteList.AgentData = new UseCachedMuteListPacket.AgentDataBlock { AgentID = m_agentId }; OutPacket(useCachedMuteList, ThrottleOutPacketType.Task); } public void SendEmpytMuteList() { GenericMessagePacket gmp = new(); gmp.AgentData.AgentID = m_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 { Parameter = Array.Empty() }; OutPacket(gmp, ThrottleOutPacketType.Task); } public void SendMuteListUpdate(string filename) { MuteListUpdatePacket muteListUpdate = (MuteListUpdatePacket)PacketPool.Instance.GetPacket(PacketType.MuteListUpdate); muteListUpdate.MuteData = new MuteListUpdatePacket.MuteDataBlock { AgentID = m_agentId, 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 { AgentID = m_agentId }; pickInfoReply.Data = new PickInfoReplyPacket.DataBlock { PickID = pickID, CreatorID = creatorID, TopPick = topPick, ParcelID = parcelID, Name = Utils.StringToBytes(name), Desc = Utils.StringToBytes(desc), SnapshotID = snapshotID, User = Utils.StringToBytes(user), OriginalName = Utils.StringToBytes(originalName), SimName = Utils.StringToBytes(simName), PosGlobal = new Vector3d(posGlobal), SortOrder = sortOrder, 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 = m_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 { WearableType = (byte)i, AssetID = wearables[i][j].AssetID, ItemID = wearables[i][j].ItemID }; aw.WearableData[idx++] = awb; //m_log.DebugFormat( // "[APPEARANCE]: Sending wearable item/asset {0} {1} (index {2}) for {3}", // awb.ItemID, awb.AssetID, i, Name); } } OutPacket(aw, ThrottleOutPacketType.Task); } 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, float hover) { // doing post zero encode, because odds of beeing bad are not that low UDPPacketBuffer buf = OpenSimUDPBase.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; // AppearanceHover vector 3 data[pos++] = 1; Utils.FloatToBytesSafepos(0, data, pos); pos += 4; Utils.FloatToBytesSafepos(0, data, pos); pos += 4; Utils.FloatToBytesSafepos(hover, data, pos); pos += 4; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, 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 = OpenSimUDPBase.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.Equals(m_agentId)) { List withobjects = new(animations.Length); List noobjects = new(animations.Length); for (int indx = 0; indx < animations.Length; ++indx) { if (objectIDs[indx].Equals(sourceAgentId) || objectIDs[indx].IsZero()) noobjects.Add(indx); else withobjects.Add(indx); } int i; // first the ones with corresponding objects for (int indx = 0; indx < withobjects.Count; ++indx) { i = withobjects[indx]; animations[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } // then the rest for (int indx = 0; indx < noobjects.Count; ++indx) { i = noobjects[indx]; animations[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } // object ids block data[pos++] = (byte)withobjects.Count; for (int indx = 0; indx < withobjects.Count; ++indx) { i = withobjects[indx]; 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); } 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 { ID = senderId }; ani.AnimationList = new ObjectAnimationPacket.AnimationListBlock[animations.Length]; for (int i = 0; i < animations.Length; ++i) { ani.AnimationList[i] = new ObjectAnimationPacket.AnimationListBlock { AnimID = animations[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 is not ScenePresence && ent is not SceneObjectPart) return; UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(objectUpdateHeader, 0, buf.Data, 0, 7); LLUDPZeroEncoder zc = new(buf.Data); zc.Position = 7; zc.AddUInt64(m_scene.RegionInfo.RegionHandle); zc.AddUInt16(Utils.FloatZeroOneToushort(m_scene.TimeDilation)); zc.AddByte(1); // block count if (ent is ScenePresence sp) CreateAvatarUpdateBlock(sp, zc); else CreatePrimUpdateBlock(ent as SceneObjectPart, (ScenePresence)SceneAgent, zc); buf.DataLength = zc.Finish(); m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } public unsafe void SendEntityTerseUpdateImmediate(ISceneEntity ent) { if (ent is null) return; UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); //setup header and regioninfo block Buffer.BlockCopy(terseUpdateHeader, 0, buf.Data, 0, 7); fixed(byte* bdata = &buf.Data[0]) { Utils.UInt16ToBytes(Utils.FloatZeroOneToushort(m_scene.TimeDilation), bdata + 15); bdata[17] = 1; byte* data = bdata + 18; if (ent is ScenePresence sp) { Utils.UInt64ToBytes(sp.RegionHandle, bdata + 7); CreateAvatartImprovedTerseBlock(sp, ref data); } else { Utils.UInt64ToBytes(m_scene.RegionInfo.RegionHandle, bdata + 7); CreatePartImprovedTerseBlock((SceneObjectPart)ent, ref data, false); } buf.DataLength = (int)(data - bdata); } m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, true); } //UUID m_courseLocationPrey = UUID.Zero; bool m_couseLocationLastEmpty = false; 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); if(totalLocations == 0) { if(m_couseLocationLastEmpty) return; m_couseLocationLastEmpty = true; } else m_couseLocationLastEmpty = false; int totalAgents = Math.Min(users.Count, 60); if(totalAgents > totalLocations) totalAgents = totalLocations; int selfindex = -1; int preyindex = -1; //bool doprey = m_courseLocationPrey != UUID.Zero; UDPPacketBuffer buf = OpenSimUDPBase.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] == m_agentId) selfindex = i; //if (doprey && users[i] == m_courseLocationPrey) // preyindex = i; } } 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); } #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 is SceneObjectPart p) { SceneObjectGroup g = p.ParentGroup; if (g.HasPrivateAttachmentPoint && g.OwnerID != m_agentId) return; // Don't send updates for other people's HUDs if((updateFlags ^ PrimUpdateFlags.SendInTransit) == 0) { List partIDs = (new List {p.LocalId}); m_entityProps.Remove(partIDs); m_entityUpdates.Remove(partIDs); return; } } int priority = m_prioritizer.GetUpdatePriority(this, entity); m_entityUpdates.Enqueue(priority, EntityUpdatesPool.Get(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 int priority = m_prioritizer.GetUpdatePriority(this, update.Entity); 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); if (oPacket.Buffer is null) return; // 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); 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) }; static private readonly byte[] ObjectAnimationHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 30 // ID (high frequency) }; static private readonly byte[] CompressedObjectHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 13 // ID (high frequency) }; static private readonly byte[] ObjectUpdateCachedHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 14 // ID (high frequency) }; private void ProcessEntityUpdates(int maxUpdatesBytes) { if (!IsActive) return; ScenePresence mysp = (ScenePresence)SceneAgent; if (mysp is null) return; List objectUpdates = null; List objectUpdateProbes = null; List compressedUpdates = null; List terseUpdates = null; List ObjectAnimationUpdates = null; List needMaterials = null; // Check to see if this is a flush if (maxUpdatesBytes <= 0) { maxUpdatesBytes = Int32.MaxValue; } EntityUpdate update; bool viewerCache = m_supportViewerCache;// && mysp.IsChildAgent; // only on child agents 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(); bool useCompressUpdate = false; if (doCulling) { cullingrange = mysp.DrawDistance + m_scene.ReprioritizationDistance + 16f; mypos = mysp.AbsolutePosition; } while (maxUpdatesBytes > 0) { if (!IsActive) return; if(orderedDequeue) { if (!m_entityUpdates.TryOrderedDequeue(out update)) break; } else { if (!m_entityUpdates.TryDequeue(out update)) break; } PrimUpdateFlags updateFlags = update.Flags; if ((updateFlags & PrimUpdateFlags.Kill) != 0) { m_killRecord.Add(update.Entity.LocalId); maxUpdatesBytes -= 30; update.Free(); continue; } useCompressUpdate = false; bool istree = false; bool hasMaterialOverride = false; if (update.Entity is SceneObjectPart part) { SceneObjectGroup grp = part.ParentGroup; if (grp.inTransit && ((update.Flags & PrimUpdateFlags.SendInTransit) == 0)) { update.Free(); 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; } update.Free(); continue; } if (grp.IsAttachment) { // animated attachments are nasty if not supported by viewer if(!m_SupportObjectAnimations && grp.RootPart.Shape.MeshFlagEntry) { update.Free(); continue; } // Someone else's HUD, why are we getting these? if (grp.HasPrivateAttachmentPoint && grp.OwnerID != m_agentId) { update.Free(); continue; } // if owner gone don't update it to anyone if (!m_scene.TryGetScenePresence(part.OwnerID, out ScenePresence sp)) { update.Free(); 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) { update.Free(); continue; } // It's an attachment of a valid avatar, but // doesn't seem to be attached, skip List atts = sp.GetAttachments(); bool found = false; for (int indx = 0; indx < atts.Count; ++indx) { if (atts[indx] == grp) { found = true; break; } } if (!found) { update.Free(); 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)) { update.Free(); 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) { update.Free(); continue; } if (!viewerCache || ((updateFlags & PrimUpdateFlags.UpdateProbe) == 0)) { GroupsNeedFullUpdate.Add(grp); update.Free(); continue; } } } if ((updateFlags & PrimUpdateFlags.UpdateProbe) != 0) { if (objectUpdateProbes is null) { objectUpdateProbes = new List(64); maxUpdatesBytes -= 18; } objectUpdateProbes.Add(update); maxUpdatesBytes -= 12; continue; } if ((updateFlags & PrimUpdateFlags.Animations) != 0) { if (m_SupportObjectAnimations && part.Animations != null) { ObjectAnimationUpdates ??= new List(8); ObjectAnimationUpdates.Add(part); maxUpdatesBytes -= 20 * part.Animations.Count + 24; } } if(viewerCache) useCompressUpdate = grp.IsViewerCachable; istree = (part.Shape.PCode == (byte)PCode.Grass || part.Shape.PCode == (byte)PCode.NewTree || part.Shape.PCode == (byte)PCode.Tree); if((m_SupportPBR && (updateFlags & PrimUpdateFlags.MaterialOvr) != 0)) hasMaterialOverride = !istree && part.Shape.RenderMaterials is not null; } else if (update.Entity is ScenePresence presence) { if (presence.IsDeleted) { update.Free(); 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.IsZero() && presence.ParentID == 0) { update.Free(); continue; } } else // what is this update ? { update.Free(); continue; } #region UpdateFlags to packet type conversion updateFlags &= PrimUpdateFlags.FullUpdate; // clear other control bits already handled if(updateFlags == PrimUpdateFlags.None) { update.Free(); continue; } 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 if ((updateFlags & canNotUseImprovedMask) == 0) { if (terseUpdates is null) { terseUpdates = new List(16); 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 if (objectUpdates is null) { objectUpdates = new List(16); maxUpdatesBytes -= 18; } objectUpdates.Add(update); } else { if (useCompressUpdate) { if (istree) maxUpdatesBytes -= 64; else maxUpdatesBytes -= 120; // crude estimation if (compressedUpdates is null) { compressedUpdates = new List(16); maxUpdatesBytes -= 18; } compressedUpdates.Add(update); } else { if (istree) maxUpdatesBytes -= 70; else maxUpdatesBytes -= 150; // crude estimation if (objectUpdates is null) { objectUpdates = new List(16); maxUpdatesBytes -= 18; } objectUpdates.Add(update); } if(hasMaterialOverride) { needMaterials ??= new List(16); needMaterials.Add((SceneObjectPart)update.Entity); } } } #endregion Block Construction } #region Packet Sending ushort timeDilation; if (!IsActive) return; timeDilation = Utils.FloatZeroOneToushort(m_scene.TimeDilation); if (objectUpdates is not null) { //List tau = new List(30); UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(objectUpdateHeader, 0, buf.Data, 0, 7); LLUDPZeroEncoder zc = new(buf.Data); zc.Position = 7; zc.AddUInt64(m_scene.RegionInfo.RegionHandle); zc.AddUInt16(timeDilation); zc.AddByte(1); // tmp block count int countposition = zc.Position - 1; int lastpos = 0; int lastzc = 0; int count = 0; bool shouldCreateSelected = false; //mantis 8639 EntityUpdate eu; for(int indx = 0; indx < objectUpdates.Count; ++indx) { eu = objectUpdates[indx]; lastpos = zc.Position; lastzc = zc.ZeroCount; if (eu.Entity is ScenePresence sp) CreateAvatarUpdateBlock(sp, zc); else { SceneObjectPart part = (SceneObjectPart)eu.Entity; shouldCreateSelected = part.CreateSelected; CreatePrimUpdateBlock(part, mysp, zc); } if (zc.Position < LLUDPServer.MAXPAYLOAD - 300) { //tau.Add(eu); ++count; } else { // we need more packets UDPPacketBuffer newbuf = OpenSimUDPBase.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); null, 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 eusp) CreateAvatarUpdateBlock(eusp, zc); else { if(shouldCreateSelected) //mantis 8639 recover selected state ((SceneObjectPart)eu.Entity).CreateSelected = true; CreatePrimUpdateBlock((SceneObjectPart)eu.Entity, mysp, zc); } //tau = new List(30); //tau.Add(eu); count = 1; } eu.Free(); //remove if using resend } 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); null, false); } } if (compressedUpdates is not null) { //List tau = new List(30); UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; Buffer.BlockCopy(CompressedObjectHeader, 0, data, 0, 7); data[0] |= Helpers.MSG_ZEROCODED; LLUDPZeroEncoder zc = new(buf.Data); zc.Position = 7; zc.AddUInt64(m_scene.RegionInfo.RegionHandle); zc.AddUInt16(timeDilation); zc.AddByte(1); // tmp block count int countposition = zc.Position - 1; int lastpos = 0; int lastzc = 0; int count = 0; bool shouldCreateSelected = false; //mantis 8639 EntityUpdate eu; for (int indx = 0; indx < compressedUpdates.Count; ++indx) { eu = compressedUpdates[indx]; SceneObjectPart sop = (SceneObjectPart)eu.Entity; if (sop.ParentGroup is null || sop.ParentGroup.IsDeleted) continue; shouldCreateSelected = sop.CreateSelected; lastpos = zc.Position; lastzc = zc.ZeroCount; CreateCompressedUpdateBlockZC(sop, mysp, zc); if (zc.Position < LLUDPServer.MAXPAYLOAD - 200) { //tau.Add(eu); ++count; } else { // we need more packets UDPPacketBuffer newbuf = OpenSimUDPBase.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); null, false); buf = newbuf; zc.Data = buf.Data; data[0] |= Helpers.MSG_ZEROCODED; zc.ZeroCount = 0; zc.Position = countposition + 1; if (shouldCreateSelected) //mantis 8639 recover selected state sop.CreateSelected = true; // im lazy now, just do last again CreateCompressedUpdateBlockZC(sop, mysp, zc); //tau = new List(30); //tau.Add(eu); count = 1; } eu.Free(); //remove if using resend } 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); null, false); } } if (objectUpdateProbes is not null) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; Buffer.BlockCopy(ObjectUpdateCachedHeader, 0, data, 0, 7); Utils.UInt64ToBytesSafepos(m_scene.RegionInfo.RegionHandle, data, 7); // 15 Utils.UInt16ToBytes(timeDilation, data, 15); // 17 const int countposition = 17; // blocks count index int pos = countposition + 1; int count = 0; EntityUpdate eu; int indx = 0; while (indx < objectUpdateProbes.Count) { unsafe { fixed(byte* bdata = &data[countposition + 1]) { byte* ptr = bdata; while (indx < objectUpdateProbes.Count) { eu = objectUpdateProbes[indx++]; SceneObjectPart sop = (SceneObjectPart)eu.Entity; if (sop.ParentGroup is null || sop.ParentGroup.IsDeleted) continue; uint primflags = m_scene.Permissions.GenerateClientFlags(sop, mysp); if (mysp.UUID.NotEqual(sop.OwnerID)) primflags &= ~(uint)PrimFlags.CreateSelected; else { if (sop.CreateSelected) primflags |= (uint)PrimFlags.CreateSelected; else primflags &= ~(uint)PrimFlags.CreateSelected; } Utils.UIntToBytes(sop.LocalId, ptr); ptr += 4; Utils.UIntToBytes((uint)sop.PseudoCRC, ptr); ptr += 4; //WRONG Utils.UIntToBytes(primflags, ptr); ptr += 4; eu.Free(); ++count; pos += 12; if (pos > (LLUDPServer.MAXPAYLOAD - 13)) { // we need more packets UDPPacketBuffer newbuf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(data, 0, newbuf.Data, 0, countposition); // start is the same data[countposition] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false); buf = newbuf; data = buf.Data; count = 0; pos = countposition + 1; break; } } } } } if (count > 0) { data[countposition] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, false); } } if (terseUpdates is not null) { int blocks = terseUpdates.Count; //List tau = new List(30); UDPPacketBuffer buf = OpenSimUDPBase.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); const int COUNTINDEX = 17; int pos = COUNTINDEX + 1; int lastpos = 0; int count = 0; EntityUpdate eu; int indx = 0; while (indx < terseUpdates.Count) { unsafe { fixed(byte* bdata = &buf.Data[0]) { byte* data = bdata + pos; while (indx < terseUpdates.Count) { eu = terseUpdates[indx++]; lastpos = pos; if(eu.Entity is ScenePresence sp) CreateAvatartImprovedTerseBlock(sp, ref data); else CreatePartImprovedTerseBlock((SceneObjectPart)eu.Entity, ref data, (eu.Flags & PrimUpdateFlags.Textures) != 0); eu.Free(); pos = (int)(data - bdata); --blocks; if (pos < LLUDPServer.MAXPAYLOAD) { //tau.Add(eu); ++count; } else if (blocks > 0) { // we need more packets UDPPacketBuffer newbuf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(buf.Data, 0, newbuf.Data, 0, COUNTINDEX); // 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 = COUNTINDEX + 1 + extralen; buf.Data[COUNTINDEX] = (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); }, true); null, true); //tau = new List(30); //tau.Add(eu); count = 1; buf = newbuf; break; } } } } } if (count > 0) { buf.Data[COUNTINDEX] = (byte)count; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, //delegate (OutgoingPacket oPacket) { ResendPrimUpdates(tau, oPacket); }, true); null, true); } } if (ObjectAnimationUpdates is not null) { SceneObjectPart sop; for (int indx = 0; indx < ObjectAnimationUpdates.Count; ++indx) { sop = ObjectAnimationUpdates[indx]; if (sop.Animations is null) continue; SceneObjectGroup sog = sop.ParentGroup; if (sog is null || sog.IsDeleted) continue; SceneObjectPart root = sog.RootPart; if (root is null || root.Shape is null || !root.Shape.MeshFlagEntry) continue; int count = sop.GetAnimations(out UUID[] ids, out int[] seqs); UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(ObjectAnimationHeader, 0, data , 0, 7); // sender block sop.UUID.ToBytes(data, 7); // 23 //animations block if (count > 255) count = 255; data[23] = (byte)count; int pos = 24; for(int i = 0; i < count; i++) { ids[i].ToBytes(data, pos); pos += 16; Utils.IntToBytesSafepos(seqs[i], data, pos); pos += 4; } buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } } if (needMaterials is not null) { foreach (SceneObjectPart sop in needMaterials) { if(sop.Shape.RenderMaterials is null) continue; string inner; if (sop.Shape.RenderMaterials.overrides is not null && sop.Shape.RenderMaterials.overrides.Length > 0) { OSDMap data = new(3) { ["id"] = (int)sop.LocalId }; OSDArray sides = new(); OSDArray sidesdata = new(); foreach (Primitive.RenderMaterials.RenderMaterialOverrideEntry ovr in sop.Shape.RenderMaterials.overrides) { sides.Add(ovr.te_index); sidesdata.Add(string.IsNullOrEmpty(ovr.data) ? new OSD() : new OSDllsdxml(ovr.data)); } data["te"] = sides; data["od"] = sidesdata; inner = OSDParser.SerializeLLSDNotation(data); if (inner.Length < 4) continue; } else inner = $"{{'id':i{sop.LocalId},'od':[],'te':[]}}"; byte[] innerB = Util.UTF8NBGetbytes(inner); if (innerB.Length > UDPPacketBuffer.BUFFER_SIZE - 16) { m_log.Debug($"[LLCLIENTVIEW]: GenericStreamingMessage packet too large ({innerB.Length})"); continue; } UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] dataptr = buf.Data; //setup header Buffer.BlockCopy(GenericStreamingMessageHeader, 0, dataptr, 0, 7); Utils.UInt16ToBytes(16757, dataptr, 7); dataptr[9] = (byte)(innerB.Length); dataptr[10] = (byte)(innerB.Length >> 8); Buffer.BlockCopy(innerB, 0, dataptr, 11, innerB.Length); buf.DataLength = innerB.Length + 11; // send it m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } } #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); PrimUpdateFlags flags = PrimUpdateFlags.CancelKill; if(viewerCache && grp.IsViewerCachable) flags |= PrimUpdateFlags.UpdateProbe; foreach (SceneObjectPart p in grp.Parts) SendEntityUpdate(p, flags); } } #endregion } static private readonly byte[] GenericStreamingMessageHeader = new byte[] { Helpers.MSG_RELIABLE, 0, 0, 0, 0, // sequence number 0, // extra 31 // ID 31 (high frequency) }; // 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() { m_entityUpdates.Reprioritize(UpdatePriorityHandler); CheckGroupsInView(); } private bool CheckGroupsInViewBusy = false; public void CheckGroupsInView() { if(!m_scene.ObjectsCullingByDistance) return; if (!IsActive) return; if (CheckGroupsInViewBusy) return; if(SceneAgent is not ScenePresence mysp || mysp.IsDeleted) return; CheckGroupsInViewBusy = true; float cullingrange = mysp.DrawDistance + m_scene.ReprioritizationDistance + 16f; Vector3 mypos = mysp.AbsolutePosition; HashSet NewGroupsInView = new(); HashSet GroupsNeedFullUpdate = new(); List kills = new(); EntityBase[] entities = m_scene.Entities.GetEntities(); foreach (EntityBase e in entities.AsSpan()) { if (!IsActive) return; if (e is SceneObjectGroup grp) { if(grp.IsDeleted || grp.IsAttachment ) continue; bool inviewgroups; lock (GroupsInView) inviewgroups = GroupsInView.Contains(grp); //temp handling of sits if(grp.GetSittingAvatarsCount() > 0) { if (!inviewgroups) GroupsNeedFullUpdate.Add(grp); NewGroupsInView.Add(grp); } else { 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(); foreach(SceneObjectGroup grp in CollectionsMarshal.AsSpan(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) { m_entityProps.Remove(partIDs); m_entityUpdates.Remove(partIDs); } } if(GroupsNeedFullUpdate.Count > 0) { bool sendProbes = m_supportViewerCache && (m_viewerHandShakeFlags & 2) == 0; if(sendProbes) { foreach (SceneObjectGroup grp in GroupsNeedFullUpdate) { PrimUpdateFlags flags = PrimUpdateFlags.CancelKill; if (grp.IsViewerCachable) flags |= PrimUpdateFlags.UpdateProbe; foreach (SceneObjectPart p in grp.Parts) SendEntityUpdate(p, flags); } } else { m_viewerHandShakeFlags &= ~2U; // nexttime send probes PrimUpdateFlags flags = PrimUpdateFlags.CancelKill; foreach (SceneObjectGroup grp in GroupsNeedFullUpdate) { foreach (SceneObjectPart p in grp.Parts) SendEntityUpdate(p, flags); } } } CheckGroupsInViewBusy = false; } private bool UpdatePriorityHandler(ref int priority, ISceneEntity entity) { if (!IsActive) 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 is 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(); 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(); 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(); 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(); newPack.AgentData.AgentID = m_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(); 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(); 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); } static private readonly byte[] SimStatsHeader = new byte[] { 0, 0, 0, 0, 0, // sequence number 0, // extra 0xff, 0xff, 0, 140 // ID 140 (low frequency bigendian) }; public void SendSimStats(SimStats stats) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); byte[] data = buf.Data; //setup header Buffer.BlockCopy(SimStatsHeader, 0, data, 0, 10); // Region Block Utils.UIntToBytesSafepos(stats.RegionX, data, 10); Utils.UIntToBytesSafepos(stats.RegionY, data, 14); Utils.UIntToBytesSafepos(stats.RegionFlags, data, 18); Utils.UIntToBytesSafepos(stats.ObjectCapacity, data, 22); // 26 // stats data[26] = (byte)StatsIndex.ViewerArraySize; int pos = 27; int i = 0; for (; i < (int)StatsIndex.UnAckedBytes; ++i) { Utils.UIntToBytesSafepos(SimStats.StatsIndexID[i], data, pos); pos += 4; Utils.FloatToBytesSafepos(stats.StatsValues[i], data, pos); pos += 4; } // unack Bytes is in KB Utils.UIntToBytesSafepos(SimStats.StatsIndexID[i], data, pos); pos += 4; Utils.FloatToBytesSafepos(stats.StatsValues[i] / 1024, data, pos); pos += 4; ++i; for (; i < (int)StatsIndex.ViewerArraySize; ++i) { Utils.UIntToBytesSafepos(SimStats.StatsIndexID[i], data, pos); pos += 4; Utils.FloatToBytesSafepos(stats.StatsValues[i], data, pos); pos += 4; } //no PID Utils.IntToBytesSafepos(0, data, pos); pos += 4; // no regioninfo (extended flags) data[pos++] = 0; // = 1; //Utils.UInt64ToBytesSafepos(RegionFlagsExtended, data, pos); pos += 8; buf.DataLength = pos; m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task); } public void SendObjectPropertiesFamilyData(ISceneEntity entity, uint requestFlags) { m_entityProps.Enqueue(0, EntityUpdatesPool.Get(entity, (PrimUpdateFlags)requestFlags, true, false)); } private void ResendPropertyUpdate(EntityUpdate update) { m_entityProps.Enqueue(0, 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 (EntityUpdate update in updates) ResendPropertyUpdate(update); } public void SendObjectPropertiesReply(ISceneEntity entity) { m_entityProps.Enqueue(0, EntityUpdatesPool.Get(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 used = new(64); List needPhysics = null; // bool orderedDequeue = m_scene.UpdatePrioritizationScheme == UpdatePrioritizationSchemes.SimpleAngularDistance; bool orderedDequeue = false; // for now EntityUpdate update; while (maxUpdateBytes > 0) { if(orderedDequeue) { if (!m_entityProps.TryOrderedDequeue(out update)) break; } else { if (!m_entityProps.TryDequeue(out update)) break; } if (update.PropsFlags == 0) { update.Free(); continue; } if(update.Entity is not SceneObjectPart sop) { update.Free(); continue; } used.Add(update); if ((update.PropsFlags & ObjectPropertyUpdateFlags.Family) != 0) { objectPropertiesFamilyUpdates ??= new List(); objectPropertiesFamilyUpdates.Add(update); maxUpdateBytes -= 100; } if ((update.PropsFlags & ObjectPropertyUpdateFlags.Object) != 0) { needPhysics ??= new List(); needPhysics.Add(sop); objectPropertiesUpdates ??= new List(); objectPropertiesUpdates.Add(update); maxUpdateBytes -= 200; // aprox } } if (objectPropertiesUpdates != null) { int blocks = objectPropertiesUpdates.Count; //List tau = new List(30); UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(ObjectPropertyUpdateHeader, 0, buf.Data, 0, 8); LLUDPZeroEncoder zc = new(buf.Data); zc.Position = 8; zc.AddByte(1); // tmp block count int countposition = zc.Position - 1; int lastpos; int lastzc; 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 = OpenSimUDPBase.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); 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); } } if (objectPropertiesFamilyUpdates != null) { foreach (EntityUpdate eu in objectPropertiesFamilyUpdates) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); Buffer.BlockCopy(ObjectFamilyUpdateHeader, 0, buf.Data, 0, 8); LLUDPZeroEncoder zc = new(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); } } if (needPhysics != null) { IEventQueue eq = Scene.RequestModuleInterface(); if(eq != null) { osUTF8 sb = eq.StartEvent("ObjectPhysicsProperties"); LLSDxmlEncode2.AddArray("ObjectData", sb); foreach (SceneObjectPart sop in needPhysics) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("LocalID",(int)sop.LocalId, sb); LLSDxmlEncode2.AddElem("Density", sop.Density, sb); LLSDxmlEncode2.AddElem("Friction", sop.Friction, sb); LLSDxmlEncode2.AddElem("GravityMultiplier", sop.GravityModifier, sb); LLSDxmlEncode2.AddElem("Restitution", sop.Restitution, sb); LLSDxmlEncode2.AddElem("PhysicsShapeType", (int)sop.PhysicsShapeType, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } } foreach(EntityUpdate eu in used) eu.Free(); } private static 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.AddShortLimitedUTF8(sop.osUTF8Name); //Description zc.AddShortLimitedUTF8(sop.osUTF8Description); } private static 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.AddShortLimitedUTF8(sop.osUTF8Name); //Description zc.AddShortLimitedUTF8(sop.osUTF8Description); // touch name zc.AddShortLimitedUTF8(sop.osUTF8TouchName); // sit name zc.AddShortLimitedUTF8(sop.osUTF8SitName); //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(); packet.AgentData.TransactionID = UUID.Random(); packet.AgentData.AgentID = m_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(); for (int i = 0; i < bl.Length; i++) { if (bl[i] is null) continue; if (bl[i].BannedUserID.IsZero()) continue; BannedUsers.Add(bl[i].BannedUserID); } SendEstateList(invoice, 4, BannedUsers.ToArray(), estateID); } public void SendRegionInfoToEstateMenu(RegionInfoForEstateMenuArgs args) { RegionInfoPacket rinfopack = new(); RegionInfoPacket.RegionInfoBlock rinfoblk = new(); rinfopack.AgentData.AgentID = m_agentId; rinfopack.AgentData.SessionID = SessionId; rinfoblk.BillableFactor = args.billableFactor; rinfoblk.EstateID = args.estateID; rinfoblk.MaxAgents = (byte)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 { HardMaxAgents = (uint)args.AgentCapacity, HardMaxObjects = (uint)args.ObjectsCapacity, MaxAgents32 = (uint)args.maxAgents, ProductName = Util.StringToBytes256(args.regionType), ProductSKU = Utils.EmptyBytes }; rinfopack.HasVariableBlocks = true; rinfopack.RegionInfo = rinfoblk; rinfopack.AgentData = new RegionInfoPacket.AgentDataBlock { AgentID = m_agentId, SessionID = m_sessionId }; rinfopack.RegionInfo3 = Array.Empty(); 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(); einfopack.Data.CovenantID = covenant; einfopack.Data.CovenantTimestamp = (uint)m_scene.RegionInfo.RegionSettings.CovenantChangedDateTime; einfopack.Data.EstateOwnerID = m_scene.RegionInfo.EstateSettings.EstateOwner; einfopack.Data.EstateName = Utils.StringToBytes(m_scene.RegionInfo.EstateSettings.EstateName); 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(); 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.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 is null) { m_log.Warn("[LLCLIENTVIEW]: No EQ Interface when sending parcel data."); return; } LandData landData = lo.LandData; IPrimCounts pc = lo.PrimCounts; int cap = 4 * landData.Bitmap.Length / 3 + 2048; osUTF8 sb = eq.StartEvent("ParcelProperties", cap); LLSDxmlEncode2.AddArrayAndMap("ParcelData", sb); LLSDxmlEncode2.AddElem("LocalID", landData.LocalID, sb); LLSDxmlEncode2.AddElem("AABBMax", landData.AABBMax, sb); LLSDxmlEncode2.AddElem("AABBMin", landData.AABBMin, sb); LLSDxmlEncode2.AddElem("Area", landData.Area, sb); LLSDxmlEncode2.AddElem("AuctionID", (int)landData.AuctionID, sb); LLSDxmlEncode2.AddElem("AuthBuyerID", landData.AuthBuyerID, sb); LLSDxmlEncode2.AddElem("Bitmap", landData.Bitmap, sb); LLSDxmlEncode2.AddElem("Category", (int)landData.Category, sb); LLSDxmlEncode2.AddElem("ClaimDate", Util.ToDateTime(landData.ClaimDate), sb); LLSDxmlEncode2.AddElem("ClaimPrice", landData.ClaimPrice, sb); LLSDxmlEncode2.AddElem("Desc", landData.Description, sb); LLSDxmlEncode2.AddElem("ParcelFlags", landData.Flags, sb); LLSDxmlEncode2.AddElem("GroupID", landData.GroupID, sb); LLSDxmlEncode2.AddElem("GroupPrims", pc.Group, sb); LLSDxmlEncode2.AddElem("IsGroupOwned", landData.IsGroupOwned, sb); LLSDxmlEncode2.AddElem("LandingType", (int)landData.LandingType, sb); if (landData.Area > 0) LLSDxmlEncode2.AddElem("MaxPrims", parcelObjectCapacity, sb); else LLSDxmlEncode2.AddElem("MaxPrims", (int)0, sb); LLSDxmlEncode2.AddElem("MediaID", landData.MediaID, sb); LLSDxmlEncode2.AddElem("MediaURL", landData.MediaURL, sb); LLSDxmlEncode2.AddElem("MediaAutoScale", landData.MediaAutoScale != 0, sb); LLSDxmlEncode2.AddElem("MusicURL", landData.MusicURL, sb); LLSDxmlEncode2.AddElem("Name", landData.Name, sb); LLSDxmlEncode2.AddElem("OtherCleanTime", landData.OtherCleanTime, sb); LLSDxmlEncode2.AddElem("OtherCount", (int)0 , sb); //TODO LLSDxmlEncode2.AddElem("OtherPrims", pc.Others, sb); LLSDxmlEncode2.AddElem("OwnerID", landData.OwnerID, sb); LLSDxmlEncode2.AddElem("OwnerPrims", pc.Owner, sb); LLSDxmlEncode2.AddElem("ParcelPrimBonus", simObjectBonusFactor, sb); LLSDxmlEncode2.AddElem("PassHours", landData.PassHours, sb); LLSDxmlEncode2.AddElem("PassPrice", landData.PassPrice, sb); LLSDxmlEncode2.AddElem("PublicCount", (int)0, sb); //TODO LLSDxmlEncode2.AddElem("RegionDenyAnonymous", (regionFlags & (uint)RegionFlags.DenyAnonymous) != 0, sb); LLSDxmlEncode2.AddElem("RegionDenyIdentified", false, sb); LLSDxmlEncode2.AddElem("RegionDenyTransacted", false, sb); LLSDxmlEncode2.AddElem("RegionPushOverride", (regionFlags & (uint)RegionFlags.RestrictPushObject) != 0, sb); LLSDxmlEncode2.AddElem("RentPrice", (int) 0, sb); LLSDxmlEncode2.AddElem("RequestResult", request_result, sb); LLSDxmlEncode2.AddElem("SalePrice", landData.SalePrice, sb); LLSDxmlEncode2.AddElem("SelectedPrims", pc.Selected, sb); LLSDxmlEncode2.AddElem("SelfCount", (int)0, sb); //TODO LLSDxmlEncode2.AddElem("SequenceID", sequence_id, sb); if (landData.SimwideArea > 0) LLSDxmlEncode2.AddElem("SimWideMaxPrims", lo.GetSimulatorMaxPrimCount(), sb); else LLSDxmlEncode2.AddElem("SimWideMaxPrims", (int)0, sb); LLSDxmlEncode2.AddElem("SimWideTotalPrims", pc.Simulator, sb); LLSDxmlEncode2.AddElem("SnapSelection", snap_selection, sb); LLSDxmlEncode2.AddElem("SnapshotID", landData.SnapshotID, sb); LLSDxmlEncode2.AddElem("Status", (int)landData.Status, sb); LLSDxmlEncode2.AddElem("TotalPrims", pc.Total, sb); LLSDxmlEncode2.AddElem("UserLocation", landData.UserLocation, sb); LLSDxmlEncode2.AddElem("UserLookAt", landData.UserLookAt, sb); LLSDxmlEncode2.AddElem("SeeAVs", landData.SeeAVs, sb); LLSDxmlEncode2.AddElem("AnyAVSounds", landData.AnyAVSounds, sb); LLSDxmlEncode2.AddElem("GroupAVSounds", landData.GroupAVSounds, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); LLSDxmlEncode2.AddArrayAndMap("MediaData", sb); LLSDxmlEncode2.AddElem("MediaDesc", landData.MediaDescription, sb); LLSDxmlEncode2.AddElem("MediaHeight", landData.MediaHeight, sb); LLSDxmlEncode2.AddElem("MediaWidth", landData.MediaWidth, sb); LLSDxmlEncode2.AddElem("MediaLoop", landData.MediaLoop, sb); LLSDxmlEncode2.AddElem("MediaType", landData.MediaType, sb); //LLSDxmlEncode2.AddElem("ObscureMedia", landData.ObscureMedia, sb); LLSDxmlEncode2.AddElem("ObscureMedia", false, sb); //obsolete //LLSDxmlEncode2.AddElem("ObscureMusic", landData.ObscureMusic, sb); LLSDxmlEncode2.AddElem("ObscureMusic", false, sb); //obsolete LLSDxmlEncode2.AddEndMapAndArray(sb); LLSDxmlEncode2.AddArrayAndMap("ParcelExtendedFlags", sb); // obscure moap LLSDxmlEncode2.AddElem("Flags", (uint)(landData.ObscureMedia ? 1 : (int)0), sb); LLSDxmlEncode2.AddEndMapAndArray(sb); LLSDxmlEncode2.AddArrayAndMap("AgeVerificationBlock", sb); LLSDxmlEncode2.AddElem("RegionDenyAgeUnverified", (regionFlags & (uint)RegionFlags.DenyAgeUnverified) != 0, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); bool allowenvovr = (regionFlags & (uint)RegionFlags.AllowEnvironmentOverride) != 0; int envVersion; if(allowenvovr) { if (SceneAgent is ScenePresence sp && sp.EnvironmentVersion > 0) envVersion = -1; else envVersion = landData.EnvironmentVersion; } else envVersion = -1; LLSDxmlEncode2.AddArrayAndMap("ParcelEnvironmentBlock", sb); LLSDxmlEncode2.AddElem("ParcelEnvironmentVersion", envVersion, sb); LLSDxmlEncode2.AddElem("RegionAllowEnvironmentOverride", allowenvovr, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); bool accessovr = !Scene.RegionInfo.EstateSettings.TaxFree; LLSDxmlEncode2.AddArrayAndMap("RegionAllowAccessBlock", sb); LLSDxmlEncode2.AddElem("RegionAllowAccessOverride", accessovr, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } public void SendLandAccessListData(List accessList, uint accessFlag, int localLandID) { ParcelAccessListReplyPacket replyPacket = (ParcelAccessListReplyPacket)PacketPool.Instance.GetPacket(PacketType.ParcelAccessListReply); replyPacket.Data.AgentID = m_agentId; replyPacket.Data.Flags = accessFlag; replyPacket.Data.LocalID = localLandID; replyPacket.Data.SequenceID = 0; List list = new(); foreach (LandAccessEntry entry in accessList) { ParcelAccessListReplyPacket.ListBlock block = new() { Flags = accessFlag, ID = entry.AgentID, 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 { 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 { 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) { ParcelObjectOwnersReplyPacket.DataBlock[] dataBlock = new ParcelObjectOwnersReplyPacket.DataBlock[notifyCount]; int num = 0; foreach (UUID owner in ownersAndCount.Keys) { dataBlock[num++] = new ParcelObjectOwnersReplyPacket.DataBlock { Count = ownersAndCount[owner], OnlineStatus = true, //TODO: fix me later OwnerID = owner, IsGroupOwned = land.GroupID.Equals(owner) || groups.Contains(owner) }; if (num >= notifyCount) break; } pack.Data = dataBlock; } else { pack.Data = Array.Empty(); } pack.Header.Zerocoded = true; OutPacket(pack, ThrottleOutPacketType.Task); } #endregion #region Helper Methods private static void ClampVectorForUint(ref Vector3 v, float max) { float a, b; a = MathF.Abs(v.X); b = MathF.Abs(v.Y); if (b > a) a = b; b = MathF.Abs(v.Z); if (b > a) a = b; if (a > max) { a = max / a; v.X *= a; v.Y *= a; v.Z *= a; } } protected static void CreateImprovedTerseBlock(ISceneEntity entity, byte[] data, ref int pos, bool includeTexture) { #region ScenePresence/SOP Handling bool avatar = (entity is ScenePresence); 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(entity.LocalId, data, pos); pos += 4; data[pos++] = (byte)attachPoint; // Avatar/CollisionPlane if (avatar) { data[pos++] = 1; //m_log.DebugFormat("CollisionPlane: {0}",collisionPlane); if (collisionPlane == Vector4.Zero) Vector4.UnitW.ToBytes(data, pos); else 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 is 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 static void CreatePartImprovedTerseBlock(SceneObjectPart part, byte[] data, ref int pos, bool includeTexture) { //object block size data[pos++] = 44; // LocalID Utils.UIntToBytesSafepos(part.LocalId, data, pos); pos += 4; if (part.ParentGroup.AttachmentPoint == 0) data[pos++] = 0; else { uint attachPoint = 0xff & part.ParentGroup.AttachmentPoint; data[pos++] = (byte)(attachPoint << 4 | attachPoint >> 4); } // no Avatar/CollisionPlane data[pos++] = 0; // Position part.RelativePosition.ToBytes(data, pos); pos += 12; // Velocity part.Velocity.ClampedToShortsBytes(128f, data, pos); pos += 6; // Acceleration part.Acceleration.ClampedToShortsBytes(64f, data, pos); pos += 6; // Rotation part.RotationOffset.ToShortsBytes(data, pos); pos += 8; // Angular Velocity part.AngularVelocity.ClampedToShortsBytes(64f, data, pos); pos += 6; // texture entry block size if (includeTexture && part.Shape.TextureEntry != null) { byte[] te = part.Shape.TextureEntry; 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; } else { data[pos++] = 0; data[pos++] = 0; } // total size 47 + (texture size + 4) } protected unsafe void CreatePartImprovedTerseBlock(SceneObjectPart part, ref byte* data, bool includeTexture) { //object block size *data++ = 44; // LocalID Utils.UIntToBytes(part.LocalId, data); data += 4; if (part.ParentGroup.AttachmentPoint == 0) *data++ = 0; else { uint attachPoint = 0xff & part.ParentGroup.AttachmentPoint; *data++ = (byte)(attachPoint << 4 | attachPoint >> 4); } // no Avatar/CollisionPlane *data++ = 0; // Position part.RelativePosition.ToBytes(data); data += 12; // Velocity part.Velocity.ClampedToShortsBytes(128f, data); data += 6; // Acceleration part.Acceleration.ClampedToShortsBytes(64f, data); data += 6; // Rotation part.RotationOffset.ToShortsBytes(data); data += 8; // Angular Velocity part.AngularVelocity.ClampedToShortsBytes(64f, data); data += 6; // texture entry block size if (includeTexture && part.Shape.TextureEntry != null) { byte[] te = part.Shape.TextureEntry; uint len = (uint)te.Length & 0x7fff; ushort totlen = (ushort)(len + 4); Utils.UInt16ToBytes(totlen, data); data += 2; Utils.UIntToBytes(len, data); data += 4; fixed(byte* t = &te[0]) Buffer.MemoryCopy(t, data, 4096, (long)len); data += len; } else { *data++ = 0; *data++ = 0; } // total size 47 + (texture size + 4) } protected static void CreateAvatartImprovedTerseBlock(ScenePresence presence, byte[] data, ref int pos) { //m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending terse update to {0} with position {1} in {2}", Name, presence.OffsetPosition, m_scene.Name); //object block size data[pos++] = 60; // LocalID Utils.UIntToBytesSafepos(presence.LocalId, data, pos); pos += 4; data[pos++] = presence.State; // Avatar/CollisionPlane data[pos++] = 1; //m_log.DebugFormat("CollisionPlane: {0}",collisionPlane); if (presence.CollisionPlane.IsZero()) Vector4.UnitW.ToBytes(data, pos); else presence.CollisionPlane.ToBytes(data, pos); pos += 16; // Position presence.OffsetPosition.ToBytes(data, pos); pos += 12; // Velocity presence.Velocity.ClampedToShortsBytes(128f, data, pos); pos += 6; // Acceleration is zero Utils.UIntToBytesSafepos(0x7fff7fff, data, pos); pos += 4; Utils.UInt16ToBytes(0x7fff, data, pos); pos += 2; // Rotation // tpvs can only see rotations around Z in some cases Quaternion rotation = presence.Rotation; if (!presence.Flying && !presence.IsSatOnObject) { Utils.UIntToBytesSafepos(0x7fff7fff, data, pos); pos += 4; float rz = rotation.Z; float rw = rotation.W; float a = rz * rz + rw * rw; if (a > -1e-6f && a < 1e-6f) { Utils.UIntToBytesSafepos(0xffff7fff, data, pos); pos += 4; } else { a = 1.0f / MathF.Sqrt(a); Utils.FloatToUInt16Bytes(rz * a, 1.0f, data, pos); pos += 2; Utils.FloatToUInt16Bytes(rw * a, 1.0f, data, pos); pos += 2; } } else { rotation.ToShortsBytes(data, pos); pos += 8; } // Angular Velocity presence.AngularVelocity.ClampedToShortsBytes(64f, data, pos); pos += 6; //texture data[pos++] = 0; data[pos++] = 0; // total size 63 } protected unsafe void CreateAvatartImprovedTerseBlock(ScenePresence presence, ref byte* data) { //m_log.DebugFormat( // "[LLCLIENTVIEW]: Sending terse update to {0} with position {1} in {2}", Name, presence.OffsetPosition, m_scene.Name); //object block size *data++ = 60; // LocalID Utils.UIntToBytes(presence.LocalId, data); data += 4; *data++ = presence.State; // Avatar/CollisionPlane *data++ = 1; //m_log.DebugFormat("CollisionPlane: {0}",collisionPlane); if (presence.CollisionPlane.IsZero()) Vector4.UnitW.ToBytes(data); else presence.CollisionPlane.ToBytes(data); data += 16; // Position presence.OffsetPosition.ToBytes(data); data += 12; // Velocity presence.Velocity.ClampedToShortsBytes(128f, data); data += 6; // Acceleration is zero Utils.UIntToBytes(0x7fff7fff, data); data += 4; Utils.UInt16ToBytes(0x7fff, data); data += 2; // Rotation // tpvs can only see rotations around Z in some cases Quaternion rotation = presence.Rotation; if (!presence.Flying && !presence.IsSatOnObject) { Utils.UIntToBytes(0x7fff7fff, data); data += 4; float rz = rotation.Z; float rw = rotation.W; float a = rz * rz + rw * rw; if (a > -1e-6f && a < 1e-6f) { Utils.UIntToBytes(0xffff7fff, data); data += 4; } else { a = 1.0f / MathF.Sqrt(a); Utils.UInt16ToBytes(Utils.FloatToUnitUInt16(rz * a), data); data += 2; Utils.UInt16ToBytes(Utils.FloatToUnitUInt16(rw * a), data); data += 2; } } else { rotation.ToShortsBytes(data); data += 8; } // Angular Velocity presence.AngularVelocity.ClampedToShortsBytes(64f, data); data += 6; //texture *data++ = 0; *data++ = 0; // total size 63 } protected static void CreateAvatarUpdateBlock(ScenePresence data, byte[] dest, ref int pos) { 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; 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.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; if (data.HideTitle) nv = Utils.StringToBytes("FirstName STRING RW SV " + data.Firstname + "\nLastName STRING RW SV " + data.Lastname + "\nTitle STRING RW SV "); else 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 static 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; } 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; if (data.HideTitle) nv = Utils.StringToBytes("FirstName STRING RW SV " + data.Firstname + "\nLastName STRING RW SV " + data.Lastname + "\nTitle STRING RW SV "); else 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.AddUInt((uint)part.PseudoCRC); 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); zc.AddNormQuat(part.RotationOffset); 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) { if (part.ParentGroup.FromItemID.IsZero()) nv = Util.StringToBytes256("AttachItemID STRING RW SV " + part.UUID.ToString()); else nv = Util.StringToBytes256("AttachItemID STRING RW SV " + part.ParentGroup.FromItemID.ToString()); } int st = 0xff & (int)part.ParentGroup.AttachmentPoint; state = (byte)((st >> 4) | (st << 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.AddUInt((uint)part.PseudoCRC); 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); zc.AddNormQuat(part.RotationOffset); 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 is 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 is null) zc.AddZeros(1); else { int len = tanim.Length; zc.AddByte((byte)len); zc.AddBytes(tanim, len); } //NameValue if(nv is 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 is null) zc.AddZeros(2); else { int len = data.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddBytes(data, len); } //text osUTF8 osUTF8PartText = part.osUTF8Text; if (osUTF8PartText is null || osUTF8PartText.Length == 0) zc.AddZeros(5); else { zc.AddShortLimitedUTF8(osUTF8PartText); //textcolor byte[] tc = part.GetTextColor().GetBytes(false); zc.AddBytes(tc, 4); } //media url zc.AddShortLimitedUTF8(part.osUTFMediaUrl); bool hasps = false; //particle system byte[] ps = part.ParticleSystem; if (ps is 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 is null || ep.Length < 2) zc.AddZeros(1); else { int len = ep.Length; zc.AddByte((byte)len); zc.AddBytes(ep, len); } bool hassound = part.Sound.IsNotZero() || 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); } [Flags] private enum CompressedFlags : uint { None = 0x00, /// Unknown ScratchPad = 0x01, /// Whether the object has a TreeSpecies Tree = 0x02, /// Whether the object has floating text ala llSetText HasText = 0x04, /// Whether the object has an active particle system HasParticlesLegacy = 0x08, /// Whether the object has sound attached to it HasSound = 0x10, /// Whether the object is attached to a root object or not HasParent = 0x20, /// Whether the object has texture animation settings TextureAnimation = 0x40, /// Whether the object has an angular velocity HasAngularVelocity = 0x80, /// Whether the object has a name value pairs string HasNameValues = 0x100, /// Whether the object has a Media URL set MediaURL = 0x200, HasParticlesNew = 0x400 } /* protected void CreateCompressedUpdateBlock(SceneObjectPart part, ScenePresence sp, byte[] dest, ref int pos) { // prepare data CompressedFlags cflags = CompressedFlags.None; // 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; } } byte state = part.Shape.State; PCode pcode = (PCode)part.Shape.PCode; bool hastree = false; if (pcode == PCode.Grass || pcode == PCode.Tree || pcode == PCode.NewTree) { cflags |= CompressedFlags.Tree; hastree = true; } //NameValue and state byte[] nv = null; if (part.ParentGroup.IsAttachment) { if (part.IsRoot) { UUID fromID = part.ParentGroup.FromItemID; if (fromID == UUID.Zero) fromID = part.UUID; nv = Util.StringToBytes256("AttachItemID STRING RW SV " + fromID.ToString()); } int st = 0xff & (int)part.ParentGroup.AttachmentPoint; state = (byte)((st & >> 4) | (st << 4)); } bool hastext = part.Text != null && part.Text.Length > 0; bool hassound = part.Sound != UUID.Zero || part.SoundFlags != 0; bool hasps = part.ParticleSystem != null && part.ParticleSystem.Length > 1; bool hastexanim = part.TextureAnimation != null && part.TextureAnimation.Length > 0; bool hasangvel = part.AngularVelocity.LengthSquared() > 1e-8f; bool hasmediaurl = part.MediaUrl != null && part.MediaUrl.Length > 1; bool haspsnew = false; if (hastext) cflags |= CompressedFlags.HasText; if (hasps) { if(part.ParticleSystem.Length > 86) { hasps= false; cflags |= CompressedFlags.HasParticlesNew; haspsnew = true; } else cflags |= CompressedFlags.HasParticlesLegacy; } if (hassound) cflags |= CompressedFlags.HasSound; if (part.ParentID != 0) cflags |= CompressedFlags.HasParent; if (hastexanim) cflags |= CompressedFlags.TextureAnimation; if (hasangvel) cflags |= CompressedFlags.HasAngularVelocity; if (hasmediaurl) cflags |= CompressedFlags.MediaURL; if (nv != null) cflags |= CompressedFlags.HasNameValues; // 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; } // first is primFlags Utils.UIntToBytesSafepos((uint)primflags, dest, pos); pos += 4; // datablock len to fill later int lenpos = pos; pos += 2; // data block part.UUID.ToBytes(dest, pos); pos += 16; Utils.UIntToBytesSafepos(part.LocalId, dest, pos); pos += 4; dest[pos++] = (byte)pcode; dest[pos++] = state; Utils.UIntToBytesSafepos((uint)part.ParentGroup.PseudoCRC, dest, pos); pos += 4; dest[pos++] = part.Material; dest[pos++] = part.ClickAction; part.Shape.Scale.ToBytes(dest, pos); pos += 12; part.RelativePosition.ToBytes(dest, pos); pos += 12; if(pcode == PCode.Grass) Vector3.Zero.ToBytes(dest, pos); else { Quaternion rotation = part.RotationOffset; rotation.Normalize(); rotation.ToBytes(dest, pos); } pos += 12; Utils.UIntToBytesSafepos((uint)cflags, dest, pos); pos += 4; if (hasps || haspsnew || hassound) part.OwnerID.ToBytes(dest, pos); else UUID.Zero.ToBytes(dest, pos); pos += 16; if (hasangvel) { part.AngularVelocity.ToBytes(dest, pos); pos += 12; } if (part.ParentID != 0) { Utils.UIntToBytesSafepos(part.ParentID, dest, pos); pos += 4; } if (hastree) dest[pos++] = state; if (hastext) { byte[] text = Util.StringToBytes256(part.Text); // must be null term Buffer.BlockCopy(text, 0, dest, pos, text.Length); pos += text.Length; byte[] tc = part.GetTextColor().GetBytes(false); Buffer.BlockCopy(tc, 0, dest, pos, tc.Length); pos += tc.Length; } if (hasmediaurl) { byte[] mu = Util.StringToBytes256(part.MediaUrl); // must be null term Buffer.BlockCopy(mu, 0, dest, pos, mu.Length); pos += mu.Length; } if (hasps) { byte[] ps = part.ParticleSystem; Buffer.BlockCopy(ps, 0, dest, pos, ps.Length); pos += ps.Length; } byte[] ex = part.Shape.ExtraParams; if (ex is null || ex.Length < 2) dest[pos++] = 0; else { Buffer.BlockCopy(ex, 0, dest, pos, ex.Length); pos += ex.Length; } if (hassound) { part.Sound.ToBytes(dest, pos); pos += 16; Utils.FloatToBytesSafepos((float)part.SoundGain, dest, pos); pos += 4; dest[pos++] = part.SoundFlags; Utils.FloatToBytesSafepos((float)part.SoundRadius, dest, pos); pos += 4; } if (nv != null) { Buffer.BlockCopy(nv, 0, dest, pos, nv.Length); pos += nv.Length; } dest[pos++] = part.Shape.PathCurve; Utils.UInt16ToBytes(part.Shape.PathBegin, dest, pos); pos += 2; Utils.UInt16ToBytes(part.Shape.PathEnd, dest, pos); pos += 2; dest[pos++] = part.Shape.PathScaleX; dest[pos++] = pathScaleY; dest[pos++] = part.Shape.PathShearX; dest[pos++] = part.Shape.PathShearY; dest[pos++] = (byte)part.Shape.PathTwist; dest[pos++] = (byte)part.Shape.PathTwistBegin; dest[pos++] = (byte)part.Shape.PathRadiusOffset; dest[pos++] = (byte)part.Shape.PathTaperX; dest[pos++] = (byte)part.Shape.PathTaperY; dest[pos++] = part.Shape.PathRevolutions; dest[pos++] = (byte)part.Shape.PathSkew; dest[pos++] = profileCurve; Utils.UInt16ToBytes(profileBegin, dest, pos); pos += 2; Utils.UInt16ToBytes(part.Shape.ProfileEnd, dest, pos); pos += 2; Utils.UInt16ToBytes(profileHollow, dest, pos); pos += 2; byte[] te = part.Shape.TextureEntry; if (te is null) { dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; dest[pos++] = 0; } else { int len = te.Length & 0x7fff; dest[pos++] = (byte)len; dest[pos++] = (byte)(len >> 8); dest[pos++] = 0; dest[pos++] = 0; Buffer.BlockCopy(te, 0, dest, pos, len); pos += len; } if (hastexanim) { byte[] ta = part.TextureAnimation; int len = ta.Length & 0x7fff; dest[pos++] = (byte)len; dest[pos++] = (byte)(len >> 8); dest[pos++] = 0; dest[pos++] = 0; Buffer.BlockCopy(ta, 0, dest, pos, len); pos += len; } if (haspsnew) { byte[] ps = part.ParticleSystem; Buffer.BlockCopy(ps, 0, dest, pos, ps.Length); pos += ps.Length; } int totlen = pos - lenpos - 2; dest[lenpos++] = (byte)totlen; dest[lenpos++] = (byte)(totlen >> 8); } */ protected void CreateCompressedUpdateBlockZC(SceneObjectPart part, ScenePresence sp, LLUDPZeroEncoder zc) { // prepare data CompressedFlags cflags = CompressedFlags.None; // 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.Equals(part.OwnerID)) { if (part.CreateSelected) { // Only send this flag once, then unset it primflags |= PrimFlags.CreateSelected; part.CreateSelected = false; } } byte state = part.Shape.State; PCode pcode = (PCode)part.Shape.PCode; // trees and grass are a lot more compact if (pcode == PCode.Grass || pcode == PCode.Tree || pcode == PCode.NewTree) { // first is primFlags zc.AddUInt((uint)primflags); // datablock len zc.AddByte(113); zc.AddZeros(1); // data block zc.AddUUID(part.UUID); zc.AddUInt(part.LocalId); zc.AddByte((byte)pcode); zc.AddByte(state); zc.AddUInt((uint)part.PseudoCRC); zc.AddZeros(2); // material and click action zc.AddVector3(part.Shape.Scale); zc.AddVector3(part.RelativePosition); if (pcode == PCode.Grass) zc.AddZeros(12); else zc.AddNormQuat(part.RotationOffset); zc.AddUInt((uint)CompressedFlags.Tree); // cflags zc.AddZeros(16); // owner id zc.AddByte(state); // tree parameter zc.AddZeros(28); //extraparameters 1, pbs 23, texture 4 return; } //NameValue and state byte[] nv = null; if (part.ParentGroup.IsAttachment) { if (part.IsRoot) { if (part.ParentGroup.FromItemID.IsZero()) nv = Util.StringToBytes256("AttachItemID STRING RW SV " + part.UUID.ToString()); else nv = Util.StringToBytes256("AttachItemID STRING RW SV " + part.ParentGroup.FromItemID.ToString()); } int st = 0xff & (int)part.ParentGroup.AttachmentPoint; state = (byte)((st >> 4) | (st << 4)); } bool hastext = false; bool hassound = false; bool hasps = false; bool hastexanim = false; bool hasangvel = false; bool hasmediaurl = false; bool haspsnew = false; int BlockLengh = 111; byte[] extraParamBytes = part.Shape.ExtraParams; if (extraParamBytes is null || extraParamBytes.Length < 2) { ++BlockLengh; extraParamBytes = null; } else BlockLengh += extraParamBytes.Length; byte[] hoverTextColor = null; osUTF8 osUTF8PartText = part.osUTF8Text; if (osUTF8PartText != null && osUTF8PartText.Length > 0) { cflags |= CompressedFlags.HasText; BlockLengh += osUTF8PartText.Length; if (osUTF8PartText[^1] != 0) ++BlockLengh; hoverTextColor = part.GetTextColor().GetBytes(false); BlockLengh += hoverTextColor.Length; hastext = true; } if (part.ParticleSystem != null && part.ParticleSystem.Length > 1) { BlockLengh += part.ParticleSystem.Length; if (part.ParticleSystem.Length > 86) { hasps = false; cflags |= CompressedFlags.HasParticlesNew; haspsnew = true; } else { cflags |= CompressedFlags.HasParticlesLegacy; hasps = true; } } if (part.Sound.IsNotZero() || part.SoundFlags != 0) { BlockLengh += 25; cflags |= CompressedFlags.HasSound; hassound = true; } if (part.ParentID != 0) { BlockLengh += 4; cflags |= CompressedFlags.HasParent; } if (part.TextureAnimation != null && part.TextureAnimation.Length > 0) { BlockLengh += part.TextureAnimation.Length + 4; cflags |= CompressedFlags.TextureAnimation; hastexanim = true; } if (part.AngularVelocity.LengthSquared() > 1e-8f) { BlockLengh += 12; cflags |= CompressedFlags.HasAngularVelocity; hasangvel = true; } osUTF8 osUTFMediaUrl = part.osUTFMediaUrl; if (osUTFMediaUrl != null && osUTFMediaUrl.Length > 0) { BlockLengh += osUTFMediaUrl.Length; if (osUTFMediaUrl[^1] != 0) ++BlockLengh; cflags |= CompressedFlags.MediaURL; hasmediaurl = true; } if (nv != null) { BlockLengh += nv.Length; cflags |= CompressedFlags.HasNameValues; } byte[] textureEntry = part.Shape.TextureEntry; if(textureEntry != null) BlockLengh += textureEntry.Length; // 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; } // first is primFlags zc.AddUInt((uint)primflags); // datablock len zc.AddByte((byte)BlockLengh); zc.AddByte((byte)(BlockLengh >> 8)); // data block zc.AddUUID(part.UUID); zc.AddUInt(part.LocalId); zc.AddByte((byte)pcode); zc.AddByte(state); zc.AddUInt((uint)part.PseudoCRC); zc.AddByte(part.Material); zc.AddByte(part.ClickAction); zc.AddVector3(part.Shape.Scale); zc.AddVector3(part.RelativePosition); zc.AddNormQuat(part.RotationOffset); zc.AddUInt((uint)cflags); if (hasps || haspsnew || hassound) zc.AddUUID(part.OwnerID); else zc.AddZeros(16); if (hasangvel) { zc.AddVector3(part.AngularVelocity); } if (part.ParentID != 0) { zc.AddUInt(part.ParentID); } if (hastext) { zc.AddBytes(osUTF8PartText.GetArray(), osUTF8PartText.Length); if (osUTF8PartText[^1] != 0) zc.AddZeros(1); zc.AddBytes(hoverTextColor, hoverTextColor.Length); } if (hasmediaurl) { zc.AddBytes(osUTFMediaUrl.GetArray(), osUTFMediaUrl.Length); if (osUTFMediaUrl[^1] != 0) zc.AddZeros(1); } if (hasps) { byte[] ps = part.ParticleSystem; zc.AddBytes(ps, ps.Length); } if (extraParamBytes is null) zc.AddZeros(1); else { zc.AddBytes(extraParamBytes, extraParamBytes.Length); } if (hassound) { zc.AddUUID(part.Sound); zc.AddFloat((float)part.SoundGain); zc.AddByte(part.SoundFlags); zc.AddFloat((float)part.SoundRadius); } if (nv != null) { zc.AddBytes(nv, nv.Length); } zc.AddByte(part.Shape.PathCurve); 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.AddByte(profileCurve); zc.AddUInt16(profileBegin); zc.AddUInt16(part.Shape.ProfileEnd); zc.AddUInt16(profileHollow); if (textureEntry is null) { zc.AddZeros(4); } else { int len = textureEntry.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddZeros(2); zc.AddBytes(textureEntry, len); } if (hastexanim) { byte[] ta = part.TextureAnimation; int len = ta.Length; zc.AddByte((byte)len); zc.AddByte((byte)(len >> 8)); zc.AddZeros(2); zc.AddBytes(ta, len); } if (haspsnew) { byte[] ps = part.ParticleSystem; zc.AddBytes(ps, ps.Length); } } 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 { ID = profileId, FirstName = Util.StringToBytes256(firstname), 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 private static readonly FrozenDictionary m_staticHandlers = new SortedDictionary() //private static readonly Dictionary m_staticHandlers = new() { { PacketType.AgentUpdate, new(HandleAgentUpdate, false) }, { PacketType.ViewerEffect, new(HandleViewerEffect , false) }, { PacketType.VelocityInterpolateOn, new(HandleVelocityInterpolateOn, false) }, { PacketType.VelocityInterpolateOff, new(HandleVelocityInterpolateOff, false) }, { PacketType.AgentCachedTexture, new(HandleAgentTextureCached, false) }, { PacketType.MultipleObjectUpdate, new(HandleMultipleObjUpdate, false) }, { PacketType.MoneyTransferRequest, new(HandleMoneyTransferRequest, false) }, { PacketType.ParcelBuy, new(HandleParcelBuyRequest, false) }, { PacketType.RegionHandshakeReply, new(HandlerRegionHandshakeReply, false) }, { PacketType.SetAlwaysRun, new(HandleSetAlwaysRun, false) }, { PacketType.AgentAnimation, new(HandleAgentAnimation, false) }, { PacketType.AgentThrottle, new(HandleAgentThrottle, false) }, { PacketType.AgentPause, new(HandleAgentPause, false) }, { PacketType.AgentResume, new(HandleAgentResume, false) }, { PacketType.ObjectImage, new(HandleObjectImage, false) }, { PacketType.ObjectGrab, new(HandleObjectGrab, false) }, { PacketType.ObjectGrabUpdate, new(HandleObjectGrabUpdate, false) }, { PacketType.ObjectSpinStart, new(HandleObjectSpinStart, false) }, { PacketType.ObjectSpinUpdate, new(HandleObjectSpinUpdate, false) }, { PacketType.ObjectSpinStop, new(HandleObjectSpinStop, false) }, { PacketType.ObjectDescription, new(HandleObjectDescription, false) }, { PacketType.ObjectName, new(HandleObjectName, false) }, { PacketType.ObjectPermissions, new(HandleObjectPermissions, false) }, { PacketType.Undo, new(HandleUndo, false) }, { PacketType.UndoLand, new(HandleLandUndo, false) }, { PacketType.Redo, new(HandleRedo, false) }, { PacketType.RequestObjectPropertiesFamily, new(HandleRequestObjectPropertiesFamily, false) }, { PacketType.ScriptAnswerYes, new(HandleScriptAnswerYes, false) }, { PacketType.ObjectClickAction, new(HandleObjectClickAction, false) }, { PacketType.ObjectMaterial, new(HandleObjectMaterial, false) }, { PacketType.RequestImage, new(HandleRequestImage, false) }, { PacketType.TransferRequest, new(HandleTransferRequest, false) }, { PacketType.ConfirmXferPacket, new(HandleConfirmXferPacket, false) }, { PacketType.UUIDNameRequest, new(HandleUUIDNameRequest, false) }, { PacketType.RegionHandleRequest, new(HandleRegionHandleRequest, false) }, { PacketType.ParcelAccessListRequest, new(HandleParcelAccessListRequest, false) }, { PacketType.ParcelAccessListUpdate, new(HandleParcelAccessListUpdate, false) }, { PacketType.ParcelPropertiesRequest, new(HandleParcelPropertiesRequest, false) }, { PacketType.RequestRegionInfo, new(HandleRequestRegionInfo, false) }, { PacketType.AgentFOV, new(HandleAgentFOV, false) }, { PacketType.MapItemRequest, new(HandleMapItemRequest, false) }, { PacketType.TransferAbort, new(HandleTransferAbort, false) }, { PacketType.AgentHeightWidth, new(HandleAgentHeightWidth, false) }, { PacketType.LogoutRequest, new(HandleLogout, true) }, { PacketType.UUIDGroupNameRequest, new(HandleUUIDGroupNameRequest, true) }, { PacketType.ObjectGroup, new(HandleObjectGroupRequest, true) }, { PacketType.GenericMessage, new(HandleGenericMessage, true) }, { PacketType.MuteListRequest, new(HandleMuteListRequest, false) }, { PacketType.UpdateMuteListEntry, new(HandleUpdateMuteListEntry, true) }, { PacketType.RemoveMuteListEntry, new(HandleRemoveMuteListEntry, true) }, { PacketType.AvatarPropertiesRequest, new(HandleAvatarPropertiesRequest, true) }, { PacketType.ChatFromViewer, new(HandleChatFromViewer, true) }, { PacketType.AvatarPropertiesUpdate, new(HandlerAvatarPropertiesUpdate, true) }, { PacketType.ScriptDialogReply, new(HandlerScriptDialogReply, true) }, { PacketType.ImprovedInstantMessage, new(HandlerImprovedInstantMessage, true) }, { PacketType.AcceptFriendship, new(HandlerAcceptFriendship, true) }, { PacketType.DeclineFriendship, new(HandlerDeclineFriendship, true) }, { PacketType.TerminateFriendship, new(HandlerTerminateFriendship, true) }, { PacketType.RezObject, new(HandlerRezObject, true) }, { PacketType.DeRezObject, new(HandlerDeRezObject, true) }, { PacketType.RezRestoreToWorld, new(HandlerRezRestoreToWorld, true) }, { PacketType.ModifyLand, new(HandlerModifyLand, true) }, { PacketType.AgentWearablesRequest, new(HandlerAgentWearablesRequest, true) }, { PacketType.AgentSetAppearance, new(HandlerAgentSetAppearance, true) }, { PacketType.AgentIsNowWearing, new(HandlerAgentIsNowWearing, true) }, { PacketType.RezSingleAttachmentFromInv, new(HandlerRezSingleAttachmentFromInv, true) }, { PacketType.RezMultipleAttachmentsFromInv, new(HandleRezMultipleAttachmentsFromInv, true) }, { PacketType.DetachAttachmentIntoInv, new(HandleDetachAttachmentIntoInv, true) }, { PacketType.ObjectAttach, new(HandleObjectAttach, true) }, { PacketType.ObjectDetach, new(HandleObjectDetach, true) }, { PacketType.ObjectDrop, new(HandleObjectDrop, true) }, { PacketType.CompleteAgentMovement, new(HandleCompleteAgentMovement, true) }, { PacketType.AgentRequestSit, new(HandleAgentRequestSit, true) }, { PacketType.AgentSit, new(HandleAgentSit, true) }, { PacketType.SoundTrigger, new(HandleSoundTrigger, true) }, { PacketType.AvatarPickerRequest, new(HandleAvatarPickerRequest, true) }, { PacketType.AgentDataUpdateRequest, new(HandleAgentDataUpdateRequest, true) }, { PacketType.UserInfoRequest, new(HandleUserInfoRequest, true) }, { PacketType.UpdateUserInfo, new(HandleUpdateUserInfo, true) }, { PacketType.SetStartLocationRequest, new(HandleSetStartLocationRequest, true) }, { PacketType.ForceScriptControlRelease, new(HandleForceScriptControlRelease, true) }, { PacketType.ObjectLink, new(HandleObjectLink, true) }, { PacketType.ObjectDelink, new(HandleObjectDelink, true) }, { PacketType.ObjectAdd, new(HandleObjectAdd, true) }, { PacketType.ObjectShape, new(HandleObjectShape, true) }, { PacketType.ObjectExtraParams, new(HandleObjectExtraParams, true) }, { PacketType.ObjectDuplicate, new(HandleObjectDuplicate, true) }, { PacketType.RequestMultipleObjects, new(HandleRequestMultipleObjects, true) }, { PacketType.ObjectSelect, new(HandleObjectSelect, true) }, { PacketType.ObjectDeselect, new(HandleObjectDeselect, true) }, { PacketType.ObjectPosition, new(HandleObjectPosition, true) }, { PacketType.ObjectScale, new(HandleObjectScale, true) }, { PacketType.ObjectRotation, new(HandleObjectRotation, true) }, { PacketType.ObjectFlagUpdate, new(HandleObjectFlagUpdate, true) }, { PacketType.ObjectDeGrab, new(HandleObjectDeGrab, true) }, { PacketType.ObjectDuplicateOnRay, new(HandleObjectDuplicateOnRay, true) }, { PacketType.ObjectIncludeInSearch, new(HandleObjectIncludeInSearch, true) }, { PacketType.AssetUploadRequest, new(HandleAssetUploadRequest, true) }, { PacketType.RequestXfer, new(HandleRequestXfer, true) }, { PacketType.SendXferPacket, new(HandleSendXferPacket, true) }, { PacketType.AbortXfer, new(HandleAbortXfer, true) }, { PacketType.CreateInventoryFolder, new (HandleCreateInventoryFolder, true) }, { PacketType.UpdateInventoryFolder, new(HandleUpdateInventoryFolder, true) }, { PacketType.MoveInventoryFolder, new(HandleMoveInventoryFolder, true) }, { PacketType.CreateInventoryItem, new(HandleCreateInventoryItem, true) }, { PacketType.LinkInventoryItem, new(HandleLinkInventoryItem, true) }, { PacketType.FetchInventory, new(HandleFetchInventory, true) }, { PacketType.FetchInventoryDescendents, new(HandleFetchInventoryDescendents, true) }, { PacketType.InventoryDescendents, new (HandleInventoryDescendents, true) }, { PacketType.PurgeInventoryDescendents, new(HandlePurgeInventoryDescendents, true) }, { PacketType.UpdateInventoryItem, new(HandleUpdateInventoryItem, true) }, { PacketType.CopyInventoryItem, new(HandleCopyInventoryItem, true) }, { PacketType.MoveInventoryItem, new(HandleMoveInventoryItem, true) }, { PacketType.RemoveInventoryItem, new(HandleRemoveInventoryItem, true) }, { PacketType.RemoveInventoryFolder, new(HandleRemoveInventoryFolder, true) }, { PacketType.RemoveInventoryObjects, new(HandleRemoveInventoryObjects, true) }, { PacketType.RequestTaskInventory, new(HandleRequestTaskInventory, true) }, { PacketType.UpdateTaskInventory, new(HandleUpdateTaskInventory, true) }, { PacketType.RemoveTaskInventory, new(HandleRemoveTaskInventory, true) }, { PacketType.MoveTaskInventory, new(HandleMoveTaskInventory, true) }, { PacketType.RezScript, new(HandleRezScript, true) }, { PacketType.MapLayerRequest, new(HandleMapLayerRequest, true) }, { PacketType.MapBlockRequest, new(HandleMapBlockRequest, true) }, { PacketType.MapNameRequest, new(HandleMapNameRequest, true) }, { PacketType.TeleportLandmarkRequest, new(HandleTeleportLandmarkRequest, true) }, { PacketType.TeleportCancel, new(HandleTeleportCancel, true) }, { PacketType.TeleportLocationRequest, new(HandleTeleportLocationRequest, true) }, { PacketType.ParcelInfoRequest, new (HandleParcelInfoRequest, true) }, { PacketType.ParcelDivide, new(HandleParcelDivide, true) }, { PacketType.ParcelJoin, new(HandleParcelJoin, true) }, { PacketType.ParcelPropertiesUpdate, new(HandleParcelPropertiesUpdate, true) }, { PacketType.ParcelSelectObjects, new(HandleParcelSelectObjects, true) }, { PacketType.ParcelObjectOwnersRequest, new(HandleParcelObjectOwnersRequest, true) }, { PacketType.ParcelGodForceOwner, new(HandleParcelGodForceOwner, true) }, { PacketType.ParcelRelease, new(HandleParcelRelease, true) }, { PacketType.ParcelReclaim, new(HandleParcelReclaim, true) }, { PacketType.ParcelReturnObjects, new(HandleParcelReturnObjects, true) }, { PacketType.ParcelSetOtherCleanTime, new(HandleParcelSetOtherCleanTime, true) }, { PacketType.ParcelDeedToGroup, new (HandleParcelDeedToGroup, true) }, { PacketType.LandStatRequest, new(HandleLandStatRequest, true) }, { PacketType.ParcelDwellRequest, new(HandleParcelDwellRequest, true) }, { PacketType.EstateOwnerMessage, new(HandleEstateOwnerMessage, true) }, { PacketType.EstateCovenantRequest, new(HandleEstateCovenantRequest, true) }, { PacketType.RequestGodlikePowers, new(HandleRequestGodlikePowers, true) }, { PacketType.GodKickUser, new(HandleGodKickUser, true) }, { PacketType.MoneyBalanceRequest, new(HandleMoneyBalanceRequest, true) }, { PacketType.EconomyDataRequest, new(HandleEconomyDataRequest, true) }, { PacketType.RequestPayPrice, new(HandleRequestPayPrice, true) }, { PacketType.ObjectSaleInfo, new(HandleObjectSaleInfo, true) }, { PacketType.ObjectBuy, new(HandleObjectBuy, true) }, { PacketType.GetScriptRunning, new(HandleGetScriptRunning, true) }, { PacketType.SetScriptRunning, new(HandleSetScriptRunning, true) }, { PacketType.ScriptReset, new(HandleScriptReset, true) }, { PacketType.ActivateGestures, new(HandleActivateGestures, true) }, { PacketType.DeactivateGestures, new(HandleDeactivateGestures, true) }, { PacketType.ObjectOwner, new(HandleObjectOwner, true) }, { PacketType.ViewerStats, new(HandleViewerStats, true) }, { PacketType.UseCircuitCode, new(HandleUseCircuitCode, true) }, { PacketType.CreateNewOutfitAttachments, new(HandleCreateNewOutfitAttachments, true) }, { PacketType.DirFindQuery, new(HandleDirFindQuery, true) }, { PacketType.DirPlacesQuery, new(HandleDirPlacesQuery, true) }, { PacketType.DirClassifiedQuery, new(HandleDirClassifiedQuery, true) }, { PacketType.DirLandQuery, new(HandleDirLandQuery, true) }, { PacketType.DirPopularQuery, new(HandleDirPopularQuery, true) }, { PacketType.EventInfoRequest, new(HandleEventInfoRequest, true) }, { PacketType.OfferCallingCard, new(HandleOfferCallingCard, true) }, { PacketType.AcceptCallingCard, new(HandleAcceptCallingCard, true) }, { PacketType.DeclineCallingCard, new(HandleDeclineCallingCard, true) }, { PacketType.ActivateGroup, new(HandleActivateGroup, true) }, { PacketType.GroupTitlesRequest, new(HandleGroupTitlesRequest, true) }, { PacketType.GroupProfileRequest, new(HandleGroupProfileRequest, true) }, { PacketType.GroupMembersRequest, new(HandleGroupMembersRequest, true) }, { PacketType.GroupRoleDataRequest, new(HandleGroupRoleDataRequest, true) }, { PacketType.GroupRoleMembersRequest, new(HandleGroupRoleMembersRequest, true) }, { PacketType.CreateGroupRequest, new(HandleCreateGroupRequest, true) }, { PacketType.UpdateGroupInfo, new(HandleUpdateGroupInfo, true) }, { PacketType.SetGroupAcceptNotices, new(HandleSetGroupAcceptNotices, true) }, { PacketType.GroupTitleUpdate, new(HandleGroupTitleUpdate, true) }, { PacketType.GroupNoticesListRequest, new(HandleGroupNoticesListRequest, true) }, { PacketType.GroupNoticeRequest, new(HandleGroupNoticeRequest, true) }, { PacketType.GroupRoleUpdate, new(HandleGroupRoleUpdate, true) }, { PacketType.GroupRoleChanges, new(HandleGroupRoleChanges, true) }, { PacketType.JoinGroupRequest, new(HandleJoinGroupRequest, true) }, { PacketType.LeaveGroupRequest, new(HandleLeaveGroupRequest, true) }, { PacketType.EjectGroupMemberRequest, new(HandleEjectGroupMemberRequest, true) }, { PacketType.InviteGroupRequest, new(HandleInviteGroupRequest, true) }, { PacketType.GroupAccountDetailsRequest, new(HandleGroupAccountDetailsRequest, true) }, { PacketType.GroupAccountSummaryRequest, new(HandleGroupAccountSummaryRequest, true) }, { PacketType.GroupAccountTransactionsRequest, new(HandleGroupTransactionsDetailsRequest, true) }, { PacketType.GroupActiveProposalsRequest, new(HandleGroupActiveProposalsRequest, true) }, { PacketType.GroupVoteHistoryRequest, new(HandleGroupVoteHistoryRequest, true) }, { PacketType.StartLure, new(HandleStartLure, true) }, { PacketType.TeleportLureRequest, new(HandleTeleportLureRequest, true) }, { PacketType.ClassifiedInfoRequest, new(HandleClassifiedInfoRequest, true) }, { PacketType.ClassifiedInfoUpdate, new(HandleClassifiedInfoUpdate, true) }, { PacketType.ClassifiedDelete, new(HandleClassifiedDelete, true) }, { PacketType.ClassifiedGodDelete, new(HandleClassifiedGodDelete, true) }, { PacketType.EventGodDelete, new(HandleEventGodDelete, true) }, { PacketType.EventNotificationAddRequest, new(HandleEventNotificationAddRequest, true) }, { PacketType.EventNotificationRemoveRequest, new(HandleEventNotificationRemoveRequest, true) }, { PacketType.RetrieveInstantMessages, new(HandleRetrieveInstantMessages, true) }, { PacketType.PickDelete, new(HandlePickDelete, true) }, { PacketType.PickGodDelete, new(HandlePickGodDelete, true) }, { PacketType.PickInfoUpdate, new(HandlePickInfoUpdate, true) }, { PacketType.AvatarNotesUpdate, new(HandleAvatarNotesUpdate, true) }, { PacketType.AvatarInterestsUpdate, new(HandleAvatarInterestsUpdate, true) }, { PacketType.GrantUserRights, new(HandleGrantUserRights, true) }, { PacketType.PlacesQuery, new(HandlePlacesQuery, true) }, { PacketType.UserReport, new(HandleUserReport, true) }, { PacketType.FindAgent, new(HandleFindAgent, true) }, { PacketType.TrackAgent, new(HandleTrackAgent, true) }, { PacketType.GodUpdateRegionInfo, new(HandleGodUpdateRegionInfoUpdate, true) }, { PacketType.GodlikeMessage, new(HandleGodlikeMessage, true) }, { PacketType.StateSave, new(HandleSaveStatePacket, true) }, { PacketType.RevokePermissions, new(HandleRevokePermissions, true) }, { PacketType.FreezeUser, new(HandleFreezeUser, true) }, { PacketType.EjectUser, new(HandleEjectUser, true) }, { PacketType.ParcelBuyPass, new(HandleParcelBuyPass, true) }, { PacketType.ParcelGodMarkAsContent, new(HandleParcelGodMarkAsContent, true) }, { PacketType.SimWideDeletes, new(HandleSimWideDeletes, true) }, { PacketType.SendPostcard, new(HandleSendPostcard, true) }, { PacketType.ChangeInventoryItemFlags, new(HandleChangeInventoryItemFlags, true) }, }.ToFrozenDictionary(); //}; protected virtual void RegisterLocalPacketHandlers() { 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 movement/state update significance against the last update made. /// /// Can only be called by one thread at a time /// /// [MethodImpl(MethodImplOptions.AggressiveInlining)] private bool CheckAgentMovementUpdateSignificance(AgentUpdatePacket.AgentDataBlock x, double now) { if((x.ControlFlags & ~(uint)AgentManager.ControlFlags.AGENT_CONTROL_FINISH_ANIM) != (uint)AgentManager.ControlFlags.NONE && now > m_thisAgentUpdateArgs.lastMoveUpdateTS + 20) return true; if ( (x.ControlFlags != m_thisAgentUpdateArgs.ControlFlags) // significant if control flags changed || (x.Flags != m_thisAgentUpdateArgs.Flags) // significant if Flags changed || (x.State != m_thisAgentUpdateArgs.State) // significant if Stats changed || (MathF.Abs(x.Far - m_thisAgentUpdateArgs.Far) >= 32f) // significant if far distance changed ) return true; float qdelta = MathF.Abs(x.BodyRotation.Dot(m_thisAgentUpdateArgs.BodyRotation)); return qdelta < QDELTABody; // significant if body rotation above(below cos) threshold } /// /// This checks the camera update significance against the last update made. /// /// Can only be called by one thread at a time /// /// [MethodImpl(MethodImplOptions.AggressiveInlining)] private bool CheckAgentCameraUpdateSignificance(AgentUpdatePacket.AgentDataBlock x) { return (MathF.Abs(x.CameraCenter.X - m_thisAgentUpdateArgs.CameraCenter.X) > VDELTA || MathF.Abs(x.CameraCenter.Y - m_thisAgentUpdateArgs.CameraCenter.Y) > VDELTA || MathF.Abs(x.CameraCenter.Z - m_thisAgentUpdateArgs.CameraCenter.Z) > VDELTA || MathF.Abs(x.CameraAtAxis.X - m_thisAgentUpdateArgs.CameraAtAxis.X) > VDELTA || MathF.Abs(x.CameraAtAxis.Y - m_thisAgentUpdateArgs.CameraAtAxis.Y) > VDELTA || MathF.Abs(x.CameraUpAxis.X - m_thisAgentUpdateArgs.CameraUpAxis.X) > VDELTA || MathF.Abs(x.CameraUpAxis.Y - m_thisAgentUpdateArgs.CameraUpAxis.Y) > VDELTA ); } private static void HandleAgentUpdate(LLClientView c, Packet packet) { if(c.OnAgentUpdate is null) return; AgentUpdatePacket agentUpdate = (AgentUpdatePacket)packet; AgentUpdatePacket.AgentDataBlock x = agentUpdate.AgentData; uint seq = packet.Header.Sequence; c.TotalAgentUpdates++; // dont let ignored updates pollute this throttles if(c.SceneAgent is null || c.SceneAgent.IsChildAgent || c.SceneAgent.IsInTransit || seq <= c.m_thisAgentUpdateArgs.lastpacketSequence ) { // throttle reset is done at MoveAgentIntoRegion() // called by scenepresence on completemovement //PacketPool.Instance.ReturnPacket(packet); return; } c.m_thisAgentUpdateArgs.lastpacketSequence = seq; bool movement; bool camera; double now = Util.GetTimeStampMS(); if (now - c.m_thisAgentUpdateArgs.lastUpdateTS > 500.0) // at least 2 per sec { movement = true; camera = true; } else { movement = c.CheckAgentMovementUpdateSignificance(x, now); camera = c.CheckAgentCameraUpdateSignificance(x); } c.OnPreAgentUpdate?.Invoke(c, c.m_thisAgentUpdateArgs); // Was there a significant movement/state change? if (movement) { c.m_thisAgentUpdateArgs.lastMoveUpdateTS = now; c.m_thisAgentUpdateArgs.BodyRotation = x.BodyRotation; c.m_thisAgentUpdateArgs.ControlFlags = x.ControlFlags; c.m_thisAgentUpdateArgs.Far = x.Far; c.m_thisAgentUpdateArgs.Flags = x.Flags; c.m_thisAgentUpdateArgs.HeadRotation = x.HeadRotation; c.m_thisAgentUpdateArgs.State = x.State; c.m_thisAgentUpdateArgs.NeedsCameraCollision = !camera; c.OnAgentUpdate?.Invoke(c, c.m_thisAgentUpdateArgs); } // Was there a significant camera(s) change? if (camera) { c.m_thisAgentUpdateArgs.CameraAtAxis = x.CameraAtAxis; c.m_thisAgentUpdateArgs.CameraCenter = x.CameraCenter; c.m_thisAgentUpdateArgs.CameraLeftAxis = x.CameraLeftAxis; c.m_thisAgentUpdateArgs.CameraUpAxis = x.CameraUpAxis; c.m_thisAgentUpdateArgs.NeedsCameraCollision = true; c.OnAgentCameraUpdate?.Invoke(c, c.m_thisAgentUpdateArgs); } if(movement && camera) c.m_thisAgentUpdateArgs.lastUpdateTS = now; } private static void HandleMoneyTransferRequest(LLClientView c, Packet Pack) { if(c.OnMoneyTransferRequest is null) return; MoneyTransferRequestPacket money = (MoneyTransferRequestPacket)Pack; c.OnMoneyTransferRequest?.Invoke(money.MoneyData.SourceID, money.MoneyData.DestID, money.MoneyData.Amount, money.MoneyData.TransactionType, Util.FieldToString(money.MoneyData.Description)); } private static void HandleParcelGodMarkAsContent(LLClientView c, Packet packet) { if(c.OnParcelGodMark is null) return; ParcelGodMarkAsContentPacket ParcelGodMarkAsContent = (ParcelGodMarkAsContentPacket)packet; c.OnParcelGodMark?.Invoke(c, c.m_agentId, ParcelGodMarkAsContent.ParcelData.LocalID); } private static void HandleFreezeUser(LLClientView c, Packet packet) { if(c.OnParcelFreezeUser is null) return; FreezeUserPacket FreezeUser = (FreezeUserPacket)packet; c.OnParcelFreezeUser?.Invoke(c, c.m_agentId, FreezeUser.Data.Flags, FreezeUser.Data.TargetID); } private static void HandleEjectUser(LLClientView c, Packet packet) { if(c.OnParcelEjectUser is null) return; EjectUserPacket EjectUser = (EjectUserPacket)packet; c.OnParcelEjectUser?.Invoke(c, c.m_agentId, EjectUser.Data.Flags, EjectUser.Data.TargetID); } private static void HandleParcelBuyPass(LLClientView c, Packet packet) { if(c.OnParcelBuyPass is null) return; ParcelBuyPassPacket ParcelBuyPass = (ParcelBuyPassPacket)packet; c.OnParcelBuyPass?.Invoke(c, c.m_agentId, ParcelBuyPass.ParcelData.LocalID); } private static void HandleParcelBuyRequest(LLClientView c, Packet Pack) { ParcelBuyPacket parcel = (ParcelBuyPacket)Pack; c.OnParcelBuy?.Invoke(c.m_agentId, parcel.Data.GroupID, parcel.Data.Final, parcel.Data.IsGroupOwned, parcel.Data.RemoveContribution, parcel.Data.LocalID, parcel.ParcelData.Area, parcel.ParcelData.Price, false); } private static void HandleUUIDGroupNameRequest(LLClientView c, Packet Pack) { if(c.OnUUIDGroupNameRequest is null) return; ScenePresence sp = (ScenePresence)c.SceneAgent; if(sp is null || sp.IsDeleted || (sp.IsInTransit && !sp.IsInLocalTransit)) return; UUIDGroupNameRequestPacket upack = (UUIDGroupNameRequestPacket)Pack; for (int i = 0; i < upack.UUIDNameBlock.Length; i++) { c.OnUUIDGroupNameRequest?.Invoke(upack.UUIDNameBlock[i].ID, c); } } public static void HandleGenericMessage(LLClientView c, Packet pack) { if (c.m_genericPacketHandlers.Count == 0) return; GenericMessagePacket gmpack = (GenericMessagePacket)pack; string method = Util.FieldToString(gmpack.MethodData.Method).ToLower().Trim(); if (c.m_genericPacketHandlers.TryGetValue(method, out GenericMessage handlerGenericMessage)) { List msg = new(); List msgBytes = new(); foreach (GenericMessagePacket.ParamListBlock block in gmpack.ParamList) { msg.Add(Util.FieldToString(block.Parameter)); msgBytes.Add(block.Parameter); } try { c.OnBinaryGenericMessage?.Invoke(c, method, msgBytes.ToArray()); handlerGenericMessage?.Invoke(c, method, msg); } catch (Exception e) { m_log.ErrorFormat( "[LLCLIENTVIEW]: Exception when handling generic message {0}{1}", e.Message, e.StackTrace); } } } public static void HandleObjectGroupRequest(LLClientView c, Packet Pack) { if(c.OnObjectGroupRequest is null) return; ObjectGroupPacket ogpack = (ObjectGroupPacket)Pack; for (int i = 0; i < ogpack.ObjectData.Length; i++) c.OnObjectGroupRequest?.Invoke(c, ogpack.AgentData.GroupID, ogpack.ObjectData[i].ObjectLocalID, UUID.Zero); } private static void HandleViewerEffect(LLClientView c, Packet Pack) { if(c.OnViewerEffect is null) return; ViewerEffectPacket viewer = (ViewerEffectPacket)Pack; List args = new(viewer.Effect.Length); for (int i = 0; i < viewer.Effect.Length; i++) { //copy the effects block arguments into the event handler arg. ViewerEffectEventHandlerArg argument = new() { AgentID = viewer.Effect[i].AgentID, Color = viewer.Effect[i].Color, Duration = viewer.Effect[i].Duration, ID = viewer.Effect[i].ID, Type = viewer.Effect[i].Type, TypeData = viewer.Effect[i].TypeData }; args.Add(argument); } c.OnViewerEffect?.Invoke(c, args); } private static void HandleVelocityInterpolateOff(LLClientView c, Packet Pack) { c.m_VelocityInterpolate = false; } private static void HandleVelocityInterpolateOn(LLClientView c, Packet Pack) { c.m_VelocityInterpolate = true; } private static void HandleAvatarPropertiesRequest(LLClientView c, Packet Pack) { AvatarPropertiesRequestPacket avatarProperties = (AvatarPropertiesRequestPacket)Pack; c.OnRequestAvatarProperties?.Invoke(c, avatarProperties.AgentData.AvatarID); } private static void HandleChatFromViewer(LLClientView c, Packet Pack) { if (c.OnChatFromClient is null) return; ChatFromViewerPacket inchatpack = (ChatFromViewerPacket)Pack; ChatFromViewerPacket.ChatDataBlock packdata = inchatpack.ChatData; OSChatMessage args = new() { Channel = packdata.Channel, Message = Utils.BytesToString(packdata.Message), Type = (ChatTypeEnum)packdata.Type, Position = c.SceneAgent.AbsolutePosition, Scene = c.Scene, Sender = c }; c.OnChatFromClient?.Invoke(c, args); } private static void HandlerAvatarPropertiesUpdate(LLClientView c, Packet Pack) { if (c.OnUpdateAvatarProperties is null) return; AvatarPropertiesUpdatePacket avatarProps = (AvatarPropertiesUpdatePacket)Pack; AvatarPropertiesUpdatePacket.PropertiesDataBlock Properties = avatarProps.PropertiesData; UserProfileProperties UserProfile = new() { UserId = c.AgentId, WebUrl = Utils.BytesToString(Properties.ProfileURL), ImageId = Properties.ImageID, FirstLifeImageId = Properties.FLImageID, AboutText = Utils.BytesToString(Properties.AboutText), FirstLifeText = Utils.BytesToString(Properties.FLAboutText), PublishProfile = Properties.AllowPublish, PublishMature = Properties.MaturePublish }; c.OnUpdateAvatarProperties?.Invoke(c, UserProfile); } private static void HandlerScriptDialogReply(LLClientView c, Packet Pack) { if (c.OnChatFromClient is null) return; ScriptDialogReplyPacket rdialog = (ScriptDialogReplyPacket)Pack; //m_log.DebugFormat("[CLIENT]: Received ScriptDialogReply from {0}", rdialog.Data.ObjectID); ScriptDialogReplyPacket.DataBlock rdialogData = rdialog.Data; OSChatMessage args = new() { Channel = rdialogData.ChatChannel, Message = Utils.BytesToString(rdialogData.ButtonLabel), Type = ChatTypeEnum.Region, //Behaviour in SL is that the response can be heard from any distance Scene = c.Scene, Sender = c }; c.OnChatFromClient?.Invoke(c, args); } private static void HandlerImprovedInstantMessage(LLClientView c, Packet Pack) { if(c.OnInstantMessage is null) return; ImprovedInstantMessagePacket msgpack = (ImprovedInstantMessagePacket)Pack; string IMfromName = Util.FieldToString(msgpack.MessageBlock.FromAgentName); string IMmessage = Utils.BytesToString(msgpack.MessageBlock.Message); GridInstantMessage im = new(c.Scene, msgpack.AgentData.AgentID, IMfromName, msgpack.MessageBlock.ToAgentID, msgpack.MessageBlock.Dialog, msgpack.MessageBlock.FromGroup, IMmessage, msgpack.MessageBlock.ID, msgpack.MessageBlock.Offline != 0, msgpack.MessageBlock.Position, msgpack.MessageBlock.BinaryBucket, true); c.OnInstantMessage?.Invoke(c, im); } private static void HandlerAcceptFriendship(LLClientView c, Packet Pack) { if(c.OnApproveFriendRequest is null) return; AcceptFriendshipPacket afriendpack = (AcceptFriendshipPacket)Pack; // My guess is this is the folder to stick the calling card into List callingCardFolders = new(); UUID transactionID = afriendpack.TransactionBlock.TransactionID; for (int fi = 0; fi < afriendpack.FolderData.Length; fi++) { callingCardFolders.Add(afriendpack.FolderData[fi].FolderID); } c.OnApproveFriendRequest?.Invoke(c, transactionID, callingCardFolders); } private static void HandlerDeclineFriendship(LLClientView c, Packet Pack) { DeclineFriendshipPacket dfriendpack = (DeclineFriendshipPacket)Pack; c.OnDenyFriendRequest?.Invoke(c, dfriendpack.TransactionBlock.TransactionID, null); } private static void HandlerTerminateFriendship(LLClientView c, Packet Pack) { TerminateFriendshipPacket tfriendpack = (TerminateFriendshipPacket)Pack; UUID exFriendID = tfriendpack.ExBlock.OtherID; FriendshipTermination TerminateFriendshipHandler = c.OnTerminateFriendship; TerminateFriendshipHandler?.Invoke(c, exFriendID); } private static void HandleFindAgent(LLClientView c, Packet packet) { FindAgentPacket FindAgent = (FindAgentPacket)packet; c.OnFindAgent?.Invoke(c, FindAgent.AgentBlock.Hunter,FindAgent.AgentBlock.Prey); } private static void HandleTrackAgent(LLClientView c, Packet packet) { TrackAgentPacket TrackAgent = (TrackAgentPacket)packet; c.OnTrackAgent?.Invoke(c, TrackAgent.AgentData.AgentID, TrackAgent.TargetData.PreyID); } private static void HandlerRezObject(LLClientView c, Packet Pack) { RezObjectPacket rezPacket = (RezObjectPacket)Pack; UUID rezGroupID = rezPacket.AgentData.GroupID; if(!c.IsGroupMember(rezGroupID)) rezGroupID = UUID.Zero; c.OnRezObject?.Invoke(c, 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); } private class DeRezObjectInfo { public List objectids; public HashSet rcvedpackets; } private Dictionary m_DeRezObjectDelayed; private static void HandlerDeRezObject(LLClientView c, Packet Pack) { if (c.OnDeRezObject is null) return; DeRezObjectPacket DeRezPacket = (DeRezObjectPacket)Pack; List deRezIDs; DeRezObjectPacket.AgentBlockBlock DeRezPacketAgentBlock = DeRezPacket.AgentBlock; DeRezAction action = (DeRezAction)DeRezPacketAgentBlock.Destination; int numberPackets = DeRezPacketAgentBlock.PacketCount; int curPacket = DeRezPacketAgentBlock.PacketNumber; UUID id = DeRezPacketAgentBlock.TransactionID; if (numberPackets > 1) { c.m_DeRezObjectDelayed ??= new Dictionary(); if (!c.m_DeRezObjectDelayed.TryGetValue(id, out DeRezObjectInfo info)) { deRezIDs = new List(DeRezPacket.ObjectData.Length); info = new DeRezObjectInfo { rcvedpackets = new HashSet() { curPacket }, objectids = deRezIDs }; c.m_DeRezObjectDelayed[id] = info; } else { if(info.rcvedpackets.Contains(curPacket)) return; info.rcvedpackets.Add(curPacket); deRezIDs = info.objectids; } foreach (DeRezObjectPacket.ObjectDataBlock data in DeRezPacket.ObjectData) { deRezIDs.Add(data.ObjectLocalID); } if (info.rcvedpackets.Count < numberPackets) return; c.m_DeRezObjectDelayed.Remove(id); info.objectids = null; info.rcvedpackets = null; } else { deRezIDs = new List(DeRezPacket.ObjectData.Length); foreach (DeRezObjectPacket.ObjectDataBlock data in DeRezPacket.ObjectData) { deRezIDs.Add(data.ObjectLocalID); } } c.OnDeRezObject?.Invoke(c, deRezIDs, DeRezPacketAgentBlock.GroupID, action, DeRezPacketAgentBlock.DestinationID); } private static void HandlerRezRestoreToWorld(LLClientView c, Packet Pack) { RezRestoreToWorldPacket restore = (RezRestoreToWorldPacket)Pack; c.OnRezRestoreToWorld?.Invoke(c, restore.InventoryData.ItemID); } private static void HandlerModifyLand(LLClientView c, Packet Pack) { if (c.OnModifyTerrain is null) return; ModifyLandPacket modify = (ModifyLandPacket)Pack; if (modify.ParcelData.Length == 0) return; //m_log.Info("[LAND]: LAND:" + modify.ToString()); for (int i = 0; i < modify.ParcelData.Length; i++) { c.OnModifyTerrain?.Invoke(c.m_agentId, modify.ModifyBlock.Height, modify.ModifyBlock.Seconds, modify.ModifyBlockExtended[i].BrushSize, modify.ModifyBlock.Action, modify.ParcelData[i].North, modify.ParcelData[i].West, modify.ParcelData[i].South, modify.ParcelData[i].East, modify.ParcelData[i].LocalID); } } public uint m_viewerHandShakeFlags = 0; private static void HandlerRegionHandshakeReply(LLClientView c, Packet Pack) { if (c.OnRegionHandShakeReply is null) return; // silence the warning RegionHandshakeReplyPacket rsrpkt = (RegionHandshakeReplyPacket)Pack; if(c.m_supportViewerCache) c.m_viewerHandShakeFlags = rsrpkt.RegionInfo.Flags; else c.m_viewerHandShakeFlags = 0; c.OnRegionHandShakeReply?.Invoke(c); } private static void HandlerAgentWearablesRequest(LLClientView c, Packet Pack) { c.OnRequestWearables?.Invoke(c); c.OnRequestAvatarsData?.Invoke(c); } private static void HandlerAgentSetAppearance(LLClientView c, Packet Pack) { if(c.OnSetAppearance is null) return; AgentSetAppearancePacket appear = (AgentSetAppearancePacket)Pack; try { Vector3 avSize = appear.AgentData.Size; 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(); foreach (RezMultipleAttachmentsFromInvPacket.ObjectDataBlock obj in rez.ObjectData) rezlist.Add(new KeyValuePair(obj.ItemID, obj.AttachmentPt)); c.OnRezMultipleAttachmentsFromInv?.Invoke(c, rezlist); } private static void HandleDetachAttachmentIntoInv(LLClientView c, Packet Pack) { if (c.OnDetachAttachmentIntoInv is null) return; DetachAttachmentIntoInvPacket detachtoInv = (DetachAttachmentIntoInvPacket)Pack; if(detachtoInv.ObjectData.AgentID.NotEqual(c.m_agentId)) return; c.OnDetachAttachmentIntoInv?.Invoke(detachtoInv.ObjectData.ItemID, c); } private static void HandleObjectAttach(LLClientView c, Packet Pack) { if (c.OnObjectAttach is null) return; ObjectAttachPacket att = (ObjectAttachPacket)Pack; if (att.ObjectData.Length > 0) c.OnObjectAttach?.Invoke(c, att.ObjectData[0].ObjectLocalID, att.AgentData.AttachmentPoint, false); } private static void HandleObjectDetach(LLClientView c, Packet Pack) { if(c.OnObjectDetach is null) return; ObjectDetachPacket dett = (ObjectDetachPacket)Pack; for (int j = 0; j < dett.ObjectData.Length; j++) { uint obj = dett.ObjectData[j].ObjectLocalID; c.OnObjectDetach?.Invoke(obj, c); } } private static void HandleObjectDrop(LLClientView c, Packet Pack) { if(c.OnObjectDrop is null) return; ObjectDropPacket dropp = (ObjectDropPacket)Pack; for (int j = 0; j < dropp.ObjectData.Length; j++) { uint obj = dropp.ObjectData[j].ObjectLocalID; c.OnObjectDrop?.Invoke(obj, c); } } private static void HandleSetAlwaysRun(LLClientView c, Packet Pack) { SetAlwaysRunPacket run = (SetAlwaysRunPacket)Pack; c.OnSetAlwaysRun?.Invoke(c, run.AgentData.AlwaysRun); } private static void HandleCompleteAgentMovement(LLClientView c, Packet Pack) { //m_log.DebugFormat("[LLClientView] HandleCompleteAgentMovement"); CompleteAgentMovementPacket cmp = (CompleteAgentMovementPacket)Pack; if(cmp.AgentData.CircuitCode == c.m_circuitCode) c.OnCompleteMovementToRegion?.Invoke(c, true); } private static void HandleAgentAnimation(LLClientView c, Packet Pack) { if(c.OnChangeAnim is null) return; AgentAnimationPacket AgentAni = (AgentAnimationPacket)Pack; for (int i = 0; i < AgentAni.AnimationList.Length; i++) c.OnChangeAnim?.Invoke(AgentAni.AnimationList[i].AnimID, AgentAni.AnimationList[i].StartAnim, false); c.OnChangeAnim?.Invoke(UUID.Zero, false, true); } private static void HandleAgentRequestSit(LLClientView c, Packet Pack) { if (c.OnAgentRequestSit is null) return; if (c.SceneAgent.IsChildAgent) { c.SendCantSitBecauseChildAgentResponse(); return; } AgentRequestSitPacket agentRequestSit = (AgentRequestSitPacket)Pack; c.OnAgentRequestSit?.Invoke(c, agentRequestSit.AgentData.AgentID, agentRequestSit.TargetObject.TargetID, agentRequestSit.TargetObject.Offset); } private static void HandleAgentSit(LLClientView c, Packet Pack) { if (c.OnAgentSit is null) return; if (c.SceneAgent.IsChildAgent) { c.SendCantSitBecauseChildAgentResponse(); return; } AgentSitPacket agentSit = (AgentSitPacket)Pack; c.OnAgentSit?.Invoke(c, agentSit.AgentData.AgentID); } /// /// Used when a child agent gets a sit response which should not be fulfilled. /// private void SendCantSitBecauseChildAgentResponse() { SendAlertMessage("Can't sit on object because it is not in the same region as you."); } private static void HandleSoundTrigger(LLClientView c, Packet Pack) { SoundTriggerPacket soundTriggerPacket = (SoundTriggerPacket)Pack; // UUIDS are sent as zeroes by the client, substitute agent's id c.OnSoundTrigger?.Invoke(soundTriggerPacket.SoundData.SoundID, c.m_agentId, c.m_agentId, c.m_agentId, soundTriggerPacket.SoundData.Gain, soundTriggerPacket.SoundData.Position, soundTriggerPacket.SoundData.Handle); } private static void HandleAvatarPickerRequest(LLClientView c, Packet Pack) { AvatarPickerRequestPacket avRequestQuery = (AvatarPickerRequestPacket)Pack; AvatarPickerRequestPacket.AgentDataBlock Requestdata = avRequestQuery.AgentData; AvatarPickerRequestPacket.DataBlock querydata = avRequestQuery.Data; //m_log.Debug("Agent Sends:" + Utils.BytesToString(querydata.Name)); c.OnAvatarPickerRequest?.Invoke(c, Requestdata.AgentID, Requestdata.QueryID, Utils.BytesToString(querydata.Name)); } private static void HandleAgentDataUpdateRequest(LLClientView c, Packet Pack) { AgentDataUpdateRequestPacket avRequestDataUpdatePacket = (AgentDataUpdateRequestPacket)Pack; c.OnAgentDataUpdateRequest?.Invoke(c, avRequestDataUpdatePacket.AgentData.AgentID, avRequestDataUpdatePacket.AgentData.SessionID); } private static void HandleUserInfoRequest(LLClientView c, Packet _) { if (c.OnUserInfoRequest != null) { c.OnUserInfoRequest(c); } else { c.SendUserInfoReply(false, true, ""); } } private static void HandleUpdateUserInfo(LLClientView c, Packet Pack) { if(c.OnUpdateUserInfo is null) return; UpdateUserInfoPacket updateUserInfo = (UpdateUserInfoPacket)Pack; bool visible = true; string DirectoryVisibility = Utils.BytesToString(updateUserInfo.UserData.DirectoryVisibility); if (DirectoryVisibility == "hidden") visible = false; c.OnUpdateUserInfo?.Invoke( updateUserInfo.UserData.IMViaEMail, visible, c); } private static void HandleSetStartLocationRequest(LLClientView c, Packet Pack) { SetStartLocationRequestPacket avSetStartLocationRequestPacket = (SetStartLocationRequestPacket)Pack; float packX = avSetStartLocationRequestPacket.StartLocationData.LocationPos.X; float packY = avSetStartLocationRequestPacket.StartLocationData.LocationPos.Y; // Linden Client limitation.. if (packX == 255.5f || packY == 255.5f) { if (c.m_scene.TryGetScenePresence(c.m_agentId, out ScenePresence avatar)) { if (packX == 255.5f) { avSetStartLocationRequestPacket.StartLocationData.LocationPos.X = avatar.AbsolutePosition.X; } if (packY == 255.5f) { avSetStartLocationRequestPacket.StartLocationData.LocationPos.Y = avatar.AbsolutePosition.Y; } } } c.OnSetStartLocationRequest?.Invoke(c, 0, avSetStartLocationRequestPacket.StartLocationData.LocationPos, avSetStartLocationRequestPacket.StartLocationData.LocationLookAt, avSetStartLocationRequestPacket.StartLocationData.LocationID); } private static void HandleAgentThrottle(LLClientView c, Packet Pack) { AgentThrottlePacket atpack = (AgentThrottlePacket)Pack; c.m_udpClient.SetThrottles(atpack.Throttle.Throttles); c.OnUpdateThrottles?.Invoke(); } private static void HandleAgentPause(LLClientView c, Packet Pack) { c.m_udpClient.IsPaused = true; } private static void HandleAgentResume(LLClientView c, Packet Pack) { c.m_udpClient.IsPaused = false; c.m_udpServer.SendPing(c.m_udpClient); } private static void HandleForceScriptControlRelease(LLClientView c, Packet _) { c.OnForceReleaseControls?.Invoke(c, c.m_agentId); } #endregion Scene/Avatar #region Objects/m_sceneObjects private static void HandleObjectLink(LLClientView c, Packet Pack) { ObjectLinkPacket link = (ObjectLinkPacket)Pack; uint parentprimid = 0; List childrenprims = new(); 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); } } c.OnLinkObjects?.Invoke(c, parentprimid, childrenprims); } private static void HandleObjectDelink(LLClientView c, Packet Pack) { ObjectDelinkPacket delink = (ObjectDelinkPacket)Pack; // 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(); for (int i = 0; i < delink.ObjectData.Length; i++) { prims.Add(delink.ObjectData[i].ObjectLocalID); } c.OnDelinkObjects?.Invoke(prims, c); } private static void HandleObjectAdd(LLClientView c, Packet Pack) { ObjectAddPacket addPacket = (ObjectAddPacket)Pack; if (addPacket.AgentData.SessionID.NotEqual(c.m_sessionId) || addPacket.AgentData.AgentID.NotEqual(c.m_agentId)) return; ObjectAddPacket.ObjectDataBlock datablk = addPacket.ObjectData; PrimitiveBaseShape shape = GetShapeFromAddPacket(addPacket); c.OnAddPrim?.Invoke(c.m_agentId, addPacket.AgentData.GroupID, datablk.RayEnd, datablk.Rotation, shape, datablk.BypassRaycast, datablk.RayStart, datablk.RayTargetID, datablk.RayEndIsIntersection, datablk.AddFlags); } private static void HandleObjectShape(LLClientView c, Packet Pack) { if(c.OnUpdatePrimShape is null) return; ObjectShapePacket shapePacket = (ObjectShapePacket)Pack; for (int i = 0; i < shapePacket.ObjectData.Length; i++) { uint id = shapePacket.ObjectData[i].ObjectLocalID; UpdateShapeArgs shapeData = new() { ObjectLocalID = id, PathBegin = shapePacket.ObjectData[i].PathBegin, PathCurve = shapePacket.ObjectData[i].PathCurve, PathEnd = shapePacket.ObjectData[i].PathEnd, PathRadiusOffset = shapePacket.ObjectData[i].PathRadiusOffset, PathRevolutions = shapePacket.ObjectData[i].PathRevolutions, PathScaleX = shapePacket.ObjectData[i].PathScaleX, PathScaleY = shapePacket.ObjectData[i].PathScaleY, PathShearX = shapePacket.ObjectData[i].PathShearX, PathShearY = shapePacket.ObjectData[i].PathShearY, PathSkew = shapePacket.ObjectData[i].PathSkew, PathTaperX = shapePacket.ObjectData[i].PathTaperX, PathTaperY = shapePacket.ObjectData[i].PathTaperY, PathTwist = shapePacket.ObjectData[i].PathTwist, PathTwistBegin = shapePacket.ObjectData[i].PathTwistBegin, ProfileBegin = shapePacket.ObjectData[i].ProfileBegin, ProfileCurve = shapePacket.ObjectData[i].ProfileCurve, ProfileEnd = shapePacket.ObjectData[i].ProfileEnd, ProfileHollow = shapePacket.ObjectData[i].ProfileHollow }; c.OnUpdatePrimShape?.Invoke(c.m_agentId, id, shapeData); } } private static void HandleObjectExtraParams(LLClientView c, Packet Pack) { ObjectExtraParamsPacket extraPar = (ObjectExtraParamsPacket)Pack; if (c.OnUpdateExtraParams != null) { for (int i = 0; i < extraPar.ObjectData.Length; i++) { c.OnUpdateExtraParams?.Invoke(c.m_agentId, extraPar.ObjectData[i].ObjectLocalID, extraPar.ObjectData[i].ParamType, extraPar.ObjectData[i].ParamInUse, extraPar.ObjectData[i].ParamData); } } } private static void HandleObjectDuplicate(LLClientView c, Packet Pack) { if(c.OnObjectDuplicate is null) return; ObjectDuplicatePacket dupe = (ObjectDuplicatePacket)Pack; UUID rezGroupID = c.IsGroupMember(dupe.AgentData.GroupID) ? dupe.AgentData.GroupID : UUID.Zero; //ObjectDuplicatePacket.AgentDataBlock AgentandGroupData = dupe.AgentData; for (int i = 0; i < dupe.ObjectData.Length; i++) { c.OnObjectDuplicate?.Invoke(dupe.ObjectData[i].ObjectLocalID, dupe.SharedData.Offset, dupe.SharedData.DuplicateFlags, c.m_agentId, rezGroupID); } } private static void HandleRequestMultipleObjects(LLClientView c, Packet Pack) { if (c.OnObjectRequest is null) return; RequestMultipleObjectsPacket incomingRequest = (RequestMultipleObjectsPacket)Pack; for (int i = 0; i < incomingRequest.ObjectData.Length; i++) c.OnObjectRequest?.Invoke(incomingRequest.ObjectData[i].ID, c); } private static void HandleObjectSelect(LLClientView c, Packet Pack) { if(c.OnObjectSelect is null) return; ObjectSelectPacket incomingselect = (ObjectSelectPacket)Pack; List thisSelection = new(); for (int i = 0; i < incomingselect.ObjectData.Length; i++) thisSelection.Add(incomingselect.ObjectData[i].ObjectLocalID); c.OnObjectSelect?.Invoke(thisSelection, c); } private static void HandleObjectDeselect(LLClientView c, Packet Pack) { if(c.OnObjectDeselect is null) return; ObjectDeselectPacket incomingdeselect = (ObjectDeselectPacket)Pack; for (int i = 0; i < incomingdeselect.ObjectData.Length; i++) { c.OnObjectDeselect?.Invoke(incomingdeselect.ObjectData[i].ObjectLocalID, c); } } private static void HandleObjectPosition(LLClientView c, Packet Pack) { if (c.OnUpdatePrimGroupPosition is null) return; // DEPRECATED: but till libsecondlife removes it, people will use it ObjectPositionPacket position = (ObjectPositionPacket)Pack; for (int i = 0; i < position.ObjectData.Length; i++) c.OnUpdatePrimGroupPosition?.Invoke(position.ObjectData[i].ObjectLocalID, position.ObjectData[i].Position, c); } private static void HandleObjectScale(LLClientView c, Packet Pack) { if (c.OnUpdatePrimGroupScale is null) return; // DEPRECATED: but till libsecondlife removes it, people will use it ObjectScalePacket scale = (ObjectScalePacket)Pack; for (int i = 0; i < scale.ObjectData.Length; i++) c.OnUpdatePrimGroupScale?.Invoke(scale.ObjectData[i].ObjectLocalID, scale.ObjectData[i].Scale, c); } private static void HandleObjectRotation(LLClientView c, Packet Pack) { if (c.OnUpdatePrimGroupRotation is null) return; // DEPRECATED: but till libsecondlife removes it, people will use it ObjectRotationPacket rotation = (ObjectRotationPacket)Pack; for (int i = 0; i < rotation.ObjectData.Length; i++) c.OnUpdatePrimGroupRotation?.Invoke(rotation.ObjectData[i].ObjectLocalID, rotation.ObjectData[i].Rotation, c); } private static void HandleObjectFlagUpdate(LLClientView c, Packet Pack) { if(c.OnUpdatePrimFlags is null) return; ObjectFlagUpdatePacket flags = (ObjectFlagUpdatePacket)Pack; // 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(); if (blocks is 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; } c.OnUpdatePrimFlags?.Invoke(flags.AgentData.ObjectLocalID, UsePhysics, IsTemporary, IsPhantom, physdata, c); } Dictionary objImageSeqs = null; double lastobjImageSeqsMS = 0.0; private static void HandleObjectImage(LLClientView c, Packet Pack) { if (c.OnUpdatePrimTexture is null) return; ObjectImagePacket imagePack = (ObjectImagePacket)Pack; double now = Util.GetTimeStampMS(); if(c.objImageSeqs is null || ( now - c.lastobjImageSeqsMS > 30000.0)) c.objImageSeqs = new Dictionary(16); c.lastobjImageSeqsMS = now; uint seq = Pack.Header.Sequence; uint id; ObjectImagePacket.ObjectDataBlock o; for (int i = 0; i < imagePack.ObjectData.Length; i++) { o = imagePack.ObjectData[i]; id = o.ObjectLocalID; if(c.objImageSeqs.TryGetValue(id, out uint lastseq)) { if(seq <= lastseq) continue; } c.objImageSeqs[id] = seq; c.OnUpdatePrimTexture?.Invoke(id, o.TextureEntry, c); } } private static void HandleObjectGrab(LLClientView c, Packet Pack) { if(c.OnGrabObject is null) return; ObjectGrabPacket grab = (ObjectGrabPacket)Pack; List touchArgs = new(); if ((grab.SurfaceInfo != null) && (grab.SurfaceInfo.Length > 0)) { foreach (ObjectGrabPacket.SurfaceInfoBlock surfaceInfo in grab.SurfaceInfo) { SurfaceTouchEventArgs arg = new() { Binormal = surfaceInfo.Binormal, FaceIndex = surfaceInfo.FaceIndex, Normal = surfaceInfo.Normal, Position = surfaceInfo.Position, STCoord = surfaceInfo.STCoord, UVCoord = surfaceInfo.UVCoord }; touchArgs.Add(arg); } } c.OnGrabObject?.Invoke(grab.ObjectData.LocalID, grab.ObjectData.GrabOffset, c, touchArgs); } private static void HandleObjectGrabUpdate(LLClientView c, Packet Pack) { if (c.OnGrabUpdate is null) return; ObjectGrabUpdatePacket grabUpdate = (ObjectGrabUpdatePacket)Pack; List touchArgs = new(); if ((grabUpdate.SurfaceInfo != null) && (grabUpdate.SurfaceInfo.Length > 0)) { foreach (ObjectGrabUpdatePacket.SurfaceInfoBlock surfaceInfo in grabUpdate.SurfaceInfo) { SurfaceTouchEventArgs arg = new() { Binormal = surfaceInfo.Binormal, FaceIndex = surfaceInfo.FaceIndex, Normal = surfaceInfo.Normal, Position = surfaceInfo.Position, STCoord = surfaceInfo.STCoord, UVCoord = surfaceInfo.UVCoord }; touchArgs.Add(arg); } } c.OnGrabUpdate?.Invoke(grabUpdate.ObjectData.ObjectID, grabUpdate.ObjectData.GrabOffsetInitial, grabUpdate.ObjectData.GrabPosition, c, touchArgs); } private static void HandleObjectDeGrab(LLClientView c, Packet Pack) { if (c.OnDeGrabObject is null) return; ObjectDeGrabPacket deGrab = (ObjectDeGrabPacket)Pack; List touchArgs = new(); if ((deGrab.SurfaceInfo != null) && (deGrab.SurfaceInfo.Length > 0)) { foreach (ObjectDeGrabPacket.SurfaceInfoBlock surfaceInfo in deGrab.SurfaceInfo) { SurfaceTouchEventArgs arg = new() { Binormal = surfaceInfo.Binormal, FaceIndex = surfaceInfo.FaceIndex, Normal = surfaceInfo.Normal, Position = surfaceInfo.Position, STCoord = surfaceInfo.STCoord, UVCoord = surfaceInfo.UVCoord }; touchArgs.Add(arg); } } c.OnDeGrabObject?.Invoke(deGrab.ObjectData.LocalID, c, touchArgs); } private static void HandleObjectSpinStart(LLClientView c, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinStart packet"); ObjectSpinStartPacket spinStart = (ObjectSpinStartPacket)Pack; if (spinStart.AgentData.SessionID.Equals(c.m_sessionId) && spinStart.AgentData.AgentID.Equals(c.m_agentId)) c.OnSpinStart?.Invoke(spinStart.ObjectData.ObjectID, c); } private static void HandleObjectSpinUpdate(LLClientView c, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinUpdate packet"); ObjectSpinUpdatePacket spinUpdate = (ObjectSpinUpdatePacket)Pack; spinUpdate.ObjectData.Rotation.GetAxisAngle(out Vector3 axis, out float angle); //m_log.Warn("[CLIENT]: ObjectSpinUpdate packet rot axis:" + axis + " angle:" + angle); c.OnSpinUpdate?.Invoke(spinUpdate.ObjectData.ObjectID, spinUpdate.ObjectData.Rotation, c); } private static void HandleObjectSpinStop(LLClientView c, Packet Pack) { //m_log.Warn("[CLIENT]: unhandled ObjectSpinStop packet"); ObjectSpinStopPacket spinStop = (ObjectSpinStopPacket)Pack; if (spinStop.AgentData.SessionID.Equals(c.m_sessionId) && spinStop.AgentData.AgentID.Equals(c.m_agentId)) c.OnSpinStop?.Invoke(spinStop.ObjectData.ObjectID, c); } private static void HandleObjectDescription(LLClientView c, Packet Pack) { if(c.OnObjectDescription is null) return; ObjectDescriptionPacket objDes = (ObjectDescriptionPacket)Pack; for (int i = 0; i < objDes.ObjectData.Length; i++) c.OnObjectDescription?.Invoke(c, objDes.ObjectData[i].LocalID, Util.FieldToString(objDes.ObjectData[i].Description)); } private static void HandleObjectName(LLClientView c, Packet Pack) { if(c.OnObjectName is null) return; ObjectNamePacket objName = (ObjectNamePacket)Pack; for (int i = 0; i < objName.ObjectData.Length; i++) c.OnObjectName?.Invoke(c, objName.ObjectData[i].LocalID, Util.FieldToString(objName.ObjectData[i].Name)); } private static void HandleObjectPermissions(LLClientView c, Packet Pack) { if (c.OnObjectPermissions is null) return; ObjectPermissionsPacket newobjPerms = (ObjectPermissionsPacket)Pack; 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; c.OnObjectPermissions?.Invoke(c, newobjPerms.AgentData.AgentID, newobjPerms.AgentData.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. } private static void HandleUndo(LLClientView c, Packet Pack) { if(c.OnUndo is null) return; UndoPacket undoitem = (UndoPacket)Pack; for (int i = 0; i < undoitem.ObjectData.Length; i++) c.OnUndo?.Invoke(c, undoitem.ObjectData[i].ObjectID); } private static void HandleLandUndo(LLClientView c, Packet Pack) { UndoLandPacket undolanditem = (UndoLandPacket)Pack; c.OnLandUndo?.Invoke(c); } private static void HandleRedo(LLClientView c, Packet Pack) { if(c.OnRedo is null) return; RedoPacket redoitem = (RedoPacket)Pack; for (int i = 0; i < redoitem.ObjectData.Length; i++) c.OnRedo?.Invoke(c, redoitem.ObjectData[i].ObjectID); } private static void HandleObjectDuplicateOnRay(LLClientView c, Packet Pack) { if(c.OnObjectDuplicateOnRay is null) return; ObjectDuplicateOnRayPacket dupeOnRay = (ObjectDuplicateOnRayPacket)Pack; UUID rezGroupID = dupeOnRay.AgentData.GroupID; if (!c.IsGroupMember(rezGroupID)) rezGroupID = UUID.Zero; for (int i = 0; i < dupeOnRay.ObjectData.Length; i++) { c.OnObjectDuplicateOnRay?.Invoke(dupeOnRay.ObjectData[i].ObjectLocalID, dupeOnRay.AgentData.DuplicateFlags, c.m_agentId, rezGroupID, dupeOnRay.AgentData.RayTargetID, dupeOnRay.AgentData.RayEnd, dupeOnRay.AgentData.RayStart, dupeOnRay.AgentData.BypassRaycast, dupeOnRay.AgentData.RayEndIsIntersection, dupeOnRay.AgentData.CopyCenters, dupeOnRay.AgentData.CopyRotates); } } private static void HandleRequestObjectPropertiesFamily(LLClientView c, Packet Pack) { //This powers the little tooltip that appears when you move your mouse over an object RequestObjectPropertiesFamilyPacket packToolTip = (RequestObjectPropertiesFamilyPacket)Pack; RequestObjectPropertiesFamilyPacket.ObjectDataBlock packObjBlock = packToolTip.ObjectData; c.OnRequestObjectPropertiesFamily?.Invoke(c, c.m_agentId, packObjBlock.RequestFlags, packObjBlock.ObjectID); } private static void HandleObjectIncludeInSearch(LLClientView c, Packet Pack) { if(c.OnObjectIncludeInSearch is null) return; //This lets us set objects to appear in search (stuff like DataSnapshot, etc) ObjectIncludeInSearchPacket packInSearch = (ObjectIncludeInSearchPacket)Pack; foreach (ObjectIncludeInSearchPacket.ObjectDataBlock objData in packInSearch.ObjectData) c.OnObjectIncludeInSearch?.Invoke(c, objData.IncludeInSearch, objData.ObjectLocalID); } private static void HandleScriptAnswerYes(LLClientView c, Packet Pack) { ScriptAnswerYesPacket scriptAnswer = (ScriptAnswerYesPacket)Pack; c.OnScriptAnswer?.Invoke(c, scriptAnswer.Data.TaskID, scriptAnswer.Data.ItemID, scriptAnswer.Data.Questions); } private static void HandleObjectClickAction(LLClientView c, Packet Pack) { if(c.OnObjectClickAction is null) return; ObjectClickActionPacket ocpacket = (ObjectClickActionPacket)Pack; foreach (ObjectClickActionPacket.ObjectDataBlock odata in ocpacket.ObjectData) { byte action = odata.ClickAction; uint localID = odata.ObjectLocalID; c.OnObjectClickAction?.Invoke(c, localID, action.ToString()); } } private static void HandleObjectMaterial(LLClientView c, Packet Pack) { if(c.OnObjectMaterial is null) return; ObjectMaterialPacket ompacket = (ObjectMaterialPacket)Pack; foreach (ObjectMaterialPacket.ObjectDataBlock odata in ompacket.ObjectData) { byte material = odata.Material; uint localID = odata.ObjectLocalID; c.OnObjectMaterial?.Invoke(c, localID, material.ToString()); } } #endregion Objects/m_sceneObjects #region Inventory/Asset/Other related packets private static void HandleRequestImage(LLClientView c, Packet Pack) { RequestImagePacket imageRequest = (RequestImagePacket)Pack; //handlerTextureRequest = null; for (int i = 0; i < imageRequest.RequestImage.Length; i++) { TextureRequestArgs args = new(); 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; c.ImageManager.EnqueueReq(args); } } /// /// 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 static void HandleTransferRequest(LLClientView c, 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 (!c.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 c.HandleSimInventoryTransferRequestWithPermsCheck(c, transfer); }, null, "LLClientView.HandleTransferRequest"); return; } } else if (transfer.TransferInfo.SourceType == (int)SourceType.SimEstate) { //TransferRequestPacket does not include covenant uuid? //get scene covenant uuid taskID = c.m_scene.RegionInfo.RegionSettings.Covenant; } // This is non-blocking c.MakeAssetRequest(transfer, taskID); } private void HandleSimInventoryTransferRequestWithPermsCheck(IClientAPI sender, TransferRequestPacket transfer) { UUID taskID = new(transfer.TransferInfo.Params, 48); UUID itemID = new(transfer.TransferInfo.Params, 64); UUID requestID = new(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.IsZero()) // Prim { SceneObjectPart part = m_scene.GetSceneObjectPart(taskID); if (part is 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 is 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, m_agentId)) return; } else if (tii.Type == (int)AssetType.Notecard) { if (!((Scene)m_scene).Permissions.CanEditNotecard(itemID, taskID, m_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 != m_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 != m_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 static void HandleAssetUploadRequest(LLClientView c, 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, c.SecureSessionId); c.OnAssetUploadRequest?.Invoke(c, temp, request.AssetBlock.TransactionID, request.AssetBlock.Type, request.AssetBlock.AssetData, request.AssetBlock.StoreLocal, request.AssetBlock.Tempfile); } private static void HandleRequestXfer(LLClientView c, Packet Pack) { RequestXferPacket xferReq = (RequestXferPacket)Pack; c.OnRequestXfer?.Invoke(c, xferReq.XferID.ID, Util.FieldToString(xferReq.XferID.Filename)); } private static void HandleSendXferPacket(LLClientView c, Packet Pack) { SendXferPacketPacket xferRec = (SendXferPacketPacket)Pack; c.OnXferReceive?.Invoke(c, xferRec.XferID.ID, xferRec.XferID.Packet, xferRec.DataPacket.Data); } private static void HandleConfirmXferPacket(LLClientView c, Packet Pack) { ConfirmXferPacketPacket confirmXfer = (ConfirmXferPacketPacket)Pack; c.OnConfirmXfer?.Invoke(c, confirmXfer.XferID.ID, confirmXfer.XferID.Packet); } private static void HandleAbortXfer(LLClientView c, Packet Pack) { AbortXferPacket abortXfer = (AbortXferPacket)Pack; c.OnAbortXfer?.Invoke(c, abortXfer.XferID.ID); } private static void HandleCreateInventoryFolder(LLClientView c, Packet Pack) { CreateInventoryFolderPacket invFolder = (CreateInventoryFolderPacket)Pack; c.OnCreateNewInventoryFolder?.Invoke(c, invFolder.FolderData.FolderID, (ushort)invFolder.FolderData.Type, Util.FieldToString(invFolder.FolderData.Name), invFolder.FolderData.ParentID); } private static void HandleUpdateInventoryFolder(LLClientView c, Packet Pack) { if (c.OnUpdateInventoryFolder is null) return; UpdateInventoryFolderPacket invFolderx = (UpdateInventoryFolderPacket)Pack; if (invFolderx.AgentData.SessionID.NotEqual(c.m_sessionId) || invFolderx.AgentData.AgentID.NotEqual(c.m_agentId)) return; for (int i = 0; i < invFolderx.FolderData.Length; i++) { c.OnUpdateInventoryFolder?.Invoke(c, invFolderx.FolderData[i].FolderID, (ushort)invFolderx.FolderData[i].Type, Util.FieldToString(invFolderx.FolderData[i].Name), invFolderx.FolderData[i].ParentID); } } private static void HandleMoveInventoryFolder(LLClientView c, Packet Pack) { if (c.OnMoveInventoryFolder is null) return; MoveInventoryFolderPacket invFoldery = (MoveInventoryFolderPacket)Pack; for (int i = 0; i < invFoldery.InventoryData.Length; i++) { c.OnMoveInventoryFolder?.Invoke(c, invFoldery.InventoryData[i].FolderID, invFoldery.InventoryData[i].ParentID); } } private static void HandleCreateInventoryItem(LLClientView c, Packet Pack) { CreateInventoryItemPacket createItem = (CreateInventoryItemPacket)Pack; if (createItem.AgentData.SessionID.NotEqual(c.m_sessionId) || createItem.AgentData.AgentID.NotEqual(c.m_agentId)) return; c.OnCreateNewInventoryItem?.Invoke(c, 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()); } private static void HandleLinkInventoryItem(LLClientView c, Packet Pack) { LinkInventoryItemPacket createLink = (LinkInventoryItemPacket)Pack; c.OnLinkInventoryItem?.Invoke( c, 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); } private static void HandleFetchInventory(LLClientView c, Packet Pack) { if (c.OnFetchInventory is null) return; FetchInventoryPacket FetchInventoryx = (FetchInventoryPacket)Pack; FetchInventoryPacket.InventoryDataBlock[] data = FetchInventoryx.InventoryData; UUID[] items = new UUID[data.Length]; UUID[] owners = new UUID[data.Length]; for (int i = 0; i < data.Length; ++i) { items[i] =data[i].ItemID; owners[i] = data[i].OwnerID; } c.OnFetchInventory?.Invoke(c, items, owners); } private static void HandleFetchInventoryDescendents(LLClientView c, Packet Pack) { FetchInventoryDescendentsPacket Fetch = (FetchInventoryDescendentsPacket)Pack; FetchInventoryDescendentsPacket.InventoryDataBlock data = Fetch.InventoryData; c.OnFetchInventoryDescendents?.Invoke(c, data.FolderID, data.OwnerID, data.FetchFolders, data.FetchItems, data.SortOrder); } private static void HandlePurgeInventoryDescendents(LLClientView c, Packet Pack) { PurgeInventoryDescendentsPacket Purge = (PurgeInventoryDescendentsPacket)Pack; c.OnPurgeInventoryDescendents?.Invoke(c, Purge.InventoryData.FolderID); } private static void HandleUpdateInventoryItem(LLClientView c, Packet Pack) { if (c.OnUpdateInventoryItem is null) return; UpdateInventoryItemPacket inventoryItemUpdate = (UpdateInventoryItemPacket)Pack; for (int i = 0; i < inventoryItemUpdate.InventoryData.Length; i++) { InventoryItemBase itemUpd = new() { ID = inventoryItemUpdate.InventoryData[i].ItemID, Name = Util.FieldToString(inventoryItemUpdate.InventoryData[i].Name), Description = Util.FieldToString(inventoryItemUpdate.InventoryData[i].Description), GroupID = inventoryItemUpdate.InventoryData[i].GroupID, GroupOwned = inventoryItemUpdate.InventoryData[i].GroupOwned, GroupPermissions = inventoryItemUpdate.InventoryData[i].GroupMask, NextPermissions = inventoryItemUpdate.InventoryData[i].NextOwnerMask, EveryOnePermissions = inventoryItemUpdate.InventoryData[i].EveryoneMask, CreationDate = inventoryItemUpdate.InventoryData[i].CreationDate, Folder = inventoryItemUpdate.InventoryData[i].FolderID, InvType = inventoryItemUpdate.InventoryData[i].InvType, SalePrice = inventoryItemUpdate.InventoryData[i].SalePrice, SaleType = inventoryItemUpdate.InventoryData[i].SaleType, Flags = inventoryItemUpdate.InventoryData[i].Flags }; c.OnUpdateInventoryItem?.Invoke(c, inventoryItemUpdate.InventoryData[i].TransactionID, inventoryItemUpdate.InventoryData[i].ItemID, itemUpd); } } private static void HandleCopyInventoryItem(LLClientView c, Packet Pack) { if(c.OnCopyInventoryItem is null) return; CopyInventoryItemPacket copyitem = (CopyInventoryItemPacket)Pack; foreach (CopyInventoryItemPacket.InventoryDataBlock datablock in copyitem.InventoryData) { c.OnCopyInventoryItem?.Invoke(c, datablock.CallbackID, datablock.OldAgentID, datablock.OldItemID, datablock.NewFolderID, Util.FieldToString(datablock.NewName)); } } private static void HandleMoveInventoryItem(LLClientView c, Packet Pack) { if (c.OnMoveInventoryItem is null) return; MoveInventoryItemPacket moveitem = (MoveInventoryItemPacket)Pack; InventoryItemBase itm = null; List items = new(); foreach (MoveInventoryItemPacket.InventoryDataBlock datablock in moveitem.InventoryData) { itm = new InventoryItemBase(datablock.ItemID, c.m_agentId) { Folder = datablock.FolderID, Name = Util.FieldToString(datablock.NewName) }; // weird, comes out as empty string //m_log.DebugFormat("[XXX] new name: {0}", itm.Name); items.Add(itm); } c.OnMoveInventoryItem?.Invoke(c, items); } private static void HandleRemoveInventoryItem(LLClientView c, Packet Pack) { if(c.OnRemoveInventoryItem is null) return; RemoveInventoryItemPacket removeItem = (RemoveInventoryItemPacket)Pack; List uuids = new(removeItem.InventoryData.Length); foreach (RemoveInventoryItemPacket.InventoryDataBlock datablock in removeItem.InventoryData) { uuids.Add(datablock.ItemID); } c.OnRemoveInventoryItem?.Invoke(c, uuids); } private static void HandleRemoveInventoryFolder(LLClientView c, Packet Pack) { if (c.OnRemoveInventoryFolder is null) return; RemoveInventoryFolderPacket removeFolder = (RemoveInventoryFolderPacket)Pack; List uuids = new(removeFolder.FolderData.Length); foreach (RemoveInventoryFolderPacket.FolderDataBlock datablock in removeFolder.FolderData) { uuids.Add(datablock.FolderID); } c.OnRemoveInventoryFolder?.Invoke(c, uuids); } private static void HandleRemoveInventoryObjects(LLClientView c, Packet Pack) { if (c.OnRemoveInventoryFolder is null || c.OnRemoveInventoryItem is null) return; RemoveInventoryObjectsPacket removeObject = (RemoveInventoryObjectsPacket)Pack; List uuids = new(removeObject.FolderData.Length); foreach (RemoveInventoryObjectsPacket.FolderDataBlock datablock in removeObject.FolderData) uuids.Add(datablock.FolderID); c.OnRemoveInventoryFolder?.Invoke(c, uuids); uuids.Clear(); foreach (RemoveInventoryObjectsPacket.ItemDataBlock datablock in removeObject.ItemData) uuids.Add(datablock.ItemID); c.OnRemoveInventoryItem?.Invoke(c, uuids); } private static void HandleRequestTaskInventory(LLClientView c, Packet Pack) { RequestTaskInventoryPacket requesttask = (RequestTaskInventoryPacket)Pack; c.OnRequestTaskInventory?.Invoke(c, requesttask.InventoryData.LocalID); } private static void HandleUpdateTaskInventory(LLClientView c, Packet Pack) { if (c.OnUpdateTaskInventory is null) return; UpdateTaskInventoryPacket updatetask = (UpdateTaskInventoryPacket)Pack; if (updatetask.UpdateData.Key != 0) return; // only do inventory not assets TaskInventoryItem newTaskItem = new() { ItemID = updatetask.InventoryData.ItemID, ParentID = updatetask.InventoryData.FolderID, CreatorID = updatetask.InventoryData.CreatorID, OwnerID = updatetask.InventoryData.OwnerID, GroupID = updatetask.InventoryData.GroupID, BasePermissions = updatetask.InventoryData.BaseMask, CurrentPermissions = updatetask.InventoryData.OwnerMask, GroupPermissions = updatetask.InventoryData.GroupMask, EveryonePermissions = updatetask.InventoryData.EveryoneMask, NextPermissions = updatetask.InventoryData.NextOwnerMask, // Unused? Clicking share with group sets GroupPermissions instead, so perhaps this is something // different //newTaskItem.GroupOwned=updatetask.InventoryData.GroupOwned, Type = updatetask.InventoryData.Type, InvType = updatetask.InventoryData.InvType, Flags = updatetask.InventoryData.Flags, //newTaskItem.SaleType=updatetask.InventoryData.SaleType, //newTaskItem.SalePrice=updatetask.InventoryData.SalePrice, Name = Util.FieldToString(updatetask.InventoryData.Name), Description = Util.FieldToString(updatetask.InventoryData.Description), CreationDate = (uint)updatetask.InventoryData.CreationDate }; c.OnUpdateTaskInventory?.Invoke(c, updatetask.InventoryData.TransactionID, newTaskItem, updatetask.UpdateData.LocalID); } private static void HandleRemoveTaskInventory(LLClientView c, Packet Pack) { RemoveTaskInventoryPacket removeTask = (RemoveTaskInventoryPacket)Pack; c.OnRemoveTaskItem?.Invoke(c, removeTask.InventoryData.ItemID, removeTask.InventoryData.LocalID); } private static void HandleMoveTaskInventory(LLClientView c, Packet Pack) { MoveTaskInventoryPacket moveTaskInventoryPacket = (MoveTaskInventoryPacket)Pack; c.OnMoveTaskItem?.Invoke( c, moveTaskInventoryPacket.AgentData.FolderID, moveTaskInventoryPacket.InventoryData.LocalID, moveTaskInventoryPacket.InventoryData.ItemID); } private static void HandleRezScript(LLClientView c, Packet Pack) { if(c.OnRezScript is null) return; //m_log.Debug(Pack.ToString()); RezScriptPacket rezScriptx = (RezScriptPacket)Pack; if (rezScriptx.AgentData.SessionID.NotEqual(c.m_sessionId) || rezScriptx.AgentData.AgentID.NotEqual(c.m_agentId)) return; InventoryItemBase item = new() { ID = rezScriptx.InventoryBlock.ItemID, Folder = rezScriptx.InventoryBlock.FolderID, CreatorId = rezScriptx.InventoryBlock.CreatorID.ToString(), Owner = rezScriptx.InventoryBlock.OwnerID, BasePermissions = rezScriptx.InventoryBlock.BaseMask, CurrentPermissions = rezScriptx.InventoryBlock.OwnerMask, EveryOnePermissions = rezScriptx.InventoryBlock.EveryoneMask, NextPermissions = rezScriptx.InventoryBlock.NextOwnerMask, GroupPermissions = rezScriptx.InventoryBlock.GroupMask, GroupOwned = rezScriptx.InventoryBlock.GroupOwned, GroupID = rezScriptx.InventoryBlock.GroupID, AssetType = rezScriptx.InventoryBlock.Type, InvType = rezScriptx.InventoryBlock.InvType, Flags = rezScriptx.InventoryBlock.Flags, SaleType = rezScriptx.InventoryBlock.SaleType, SalePrice = rezScriptx.InventoryBlock.SalePrice, Name = Util.FieldToString(rezScriptx.InventoryBlock.Name), Description = Util.FieldToString(rezScriptx.InventoryBlock.Description), CreationDate = rezScriptx.InventoryBlock.CreationDate }; c.OnRezScript?.Invoke(c, item, rezScriptx.InventoryBlock.TransactionID, rezScriptx.UpdateBlock.ObjectLocalID); } private static void HandleMapLayerRequest(LLClientView c, Packet Pack) { c.RequestMapLayer(); } private static void HandleMapBlockRequest(LLClientView c, Packet Pack) { MapBlockRequestPacket MapRequest = (MapBlockRequestPacket)Pack; c.OnRequestMapBlocks?.Invoke(c, MapRequest.PositionData.MinX, MapRequest.PositionData.MinY, MapRequest.PositionData.MaxX, MapRequest.PositionData.MaxY, MapRequest.AgentData.Flags); } private static void HandleMapNameRequest(LLClientView c, Packet Pack) { MapNameRequestPacket map = (MapNameRequestPacket)Pack; string mapName = map.NameData.Name.Length == 0 ? c.m_scene.RegionInfo.RegionName : Util.UTF8.GetString(map.NameData.Name, 0, map.NameData.Name.Length - 1); c.OnMapNameRequest?.Invoke(c, mapName, map.AgentData.Flags); } private static void HandleTeleportLandmarkRequest(LLClientView c, Packet Pack) { TeleportLandmarkRequestPacket tpReq = (TeleportLandmarkRequestPacket)Pack; UUID lmid = tpReq.Info.LandmarkID; AssetLandmark lm; if (!lmid.IsZero()) { //AssetBase lma = m_assetCache.GetAsset(lmid, false); AssetBase lma = c.m_assetService.Get(lmid.ToString()); if (lma is null) { // Failed to find landmark // Let's try to search in the user's home asset server lma = c.FindAssetInUserAssetServer(lmid.ToString()); if (lma is null) { // Really doesn't exist m_log.WarnFormat("[llClient]: landmark asset {0} not found",lmid.ToString()); c.SendTeleportFailed("Could not find the landmark asset data"); return; } } try { lm = new AssetLandmark(lma); } catch (NullReferenceException) { // asset not found generates null ref inside the assetlandmark constructor. c.SendTeleportFailed("Could not find the landmark asset data"); return; } } else { // Teleport home request c.OnTeleportHomeRequest?.Invoke(c.m_agentId, c); return; } c.OnTeleportLandmarkRequest?.Invoke(c, lm); } private static void HandleTeleportCancel(LLClientView c, Packet Pack) { TeleportCancelPacket pkt = (TeleportCancelPacket) Pack; c.OnTeleportCancel?.Invoke(c); } 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(id, assetServer, false); } return null; } private static void HandleTeleportLocationRequest(LLClientView c, Packet Pack) { if(c.OnTeleportLocationRequest is null) { c.SendTeleportFailed("Could not process the teleport"); return; } TeleportLocationRequestPacket tpLocReq = (TeleportLocationRequestPacket)Pack; // Adjust teleport location to base of a larger region if requested to teleport to a sub-region Util.RegionHandleToWorldLoc(tpLocReq.Info.RegionHandle, out uint locX, out uint locY); RegionInfo ri = c.m_scene.RegionInfo; if ((locX >= ri.WorldLocX) && (locX < (ri.WorldLocX + ri.RegionSizeX)) && (locY >= ri.WorldLocY) && (locY < (ri.WorldLocY + ri.RegionSizeY))) { tpLocReq.Info.RegionHandle = ri.RegionHandle; tpLocReq.Info.Position.X += locX - ri.WorldLocX; tpLocReq.Info.Position.Y += locY - ri.WorldLocY; } c.OnTeleportLocationRequest?.Invoke(c, tpLocReq.Info.RegionHandle, tpLocReq.Info.Position, tpLocReq.Info.LookAt, 16); } #endregion Inventory/Asset/Other related packets private static void HandleUUIDNameRequest(LLClientView c, Packet Pack) { if(c.OnNameFromUUIDRequest is null) return; ScenePresence sp = (ScenePresence)c.SceneAgent; if(sp is null || sp.IsDeleted || (sp.IsInTransit && !sp.IsInLocalTransit)) return; UUIDNameRequestPacket incoming = (UUIDNameRequestPacket)Pack; foreach (UUIDNameRequestPacket.UUIDNameBlockBlock UUIDBlock in incoming.UUIDNameBlock) c.OnNameFromUUIDRequest?.Invoke(UUIDBlock.ID, c); } #region Parcel related packets private static void HandleRegionHandleRequest(LLClientView c, Packet Pack) { RegionHandleRequestPacket rhrPack = (RegionHandleRequestPacket)Pack; c.OnRegionHandleRequest?.Invoke(c, rhrPack.RequestBlock.RegionID); } private static void HandleParcelInfoRequest(LLClientView c, Packet Pack) { if(c.OnParcelInfoRequest is null) return; ParcelInfoRequestPacket pirPack = (ParcelInfoRequestPacket)Pack; c.OnParcelInfoRequest?.Invoke(c, pirPack.Data.ParcelID); } private static void HandleParcelAccessListRequest(LLClientView c, Packet Pack) { ParcelAccessListRequestPacket requestPacket = (ParcelAccessListRequestPacket)Pack; c.OnParcelAccessListRequest?.Invoke(requestPacket.AgentData.AgentID, requestPacket.AgentData.SessionID, requestPacket.Data.Flags, requestPacket.Data.SequenceID, requestPacket.Data.LocalID, c); } private static void HandleParcelAccessListUpdate(LLClientView c, Packet Pack) { if(c.OnParcelAccessListUpdateRequest is null) return; ParcelAccessListUpdatePacket updatePacket = (ParcelAccessListUpdatePacket)Pack; // viewers do send estimated number of packets and sequenceID, but don't seem reliable. List entries = new(updatePacket.List.Length); foreach (ParcelAccessListUpdatePacket.ListBlock block in updatePacket.List) { LandAccessEntry entry = new() { AgentID = block.ID, Flags = (AccessList)block.Flags, Expires = block.Time }; entries.Add(entry); } c.OnParcelAccessListUpdateRequest?.Invoke(updatePacket.AgentData.AgentID, updatePacket.Data.Flags, updatePacket.Data.TransactionID, updatePacket.Data.LocalID, entries, c); } private static void HandleParcelPropertiesRequest(LLClientView c, Packet Pack) { ParcelPropertiesRequestPacket propertiesRequest = (ParcelPropertiesRequestPacket)Pack; ParcelPropertiesRequestPacket.ParcelDataBlock pdb = propertiesRequest.ParcelData; c.OnParcelPropertiesRequest?.Invoke((int)MathF.Round(pdb.West), (int)MathF.Round(pdb.South), (int)MathF.Round(pdb.East), (int)MathF.Round(pdb.North), pdb.SequenceID, pdb.SnapSelection, c); } private static void HandleParcelDivide(LLClientView c, Packet Pack) { ParcelDividePacket landDivide = (ParcelDividePacket)Pack; c.OnParcelDivideRequest?.Invoke((int)MathF.Round(landDivide.ParcelData.West), (int)MathF.Round(landDivide.ParcelData.South), (int)Math.Round(landDivide.ParcelData.East), (int)MathF.Round(landDivide.ParcelData.North), c); } private static void HandleParcelJoin(LLClientView c, Packet Pack) { ParcelJoinPacket landJoin = (ParcelJoinPacket)Pack; c.OnParcelJoinRequest?.Invoke((int)MathF.Round(landJoin.ParcelData.West), (int)MathF.Round(landJoin.ParcelData.South), (int)MathF.Round(landJoin.ParcelData.East), (int)MathF.Round(landJoin.ParcelData.North), c); } private static void HandleParcelPropertiesUpdate(LLClientView c, Packet Pack) { if (c.OnParcelPropertiesUpdateRequest is null) return; ParcelPropertiesUpdatePacket parcelPropertiesPacket = (ParcelPropertiesUpdatePacket)Pack; if (parcelPropertiesPacket.AgentData.SessionID.NotEqual(c.m_sessionId) || parcelPropertiesPacket.AgentData.AgentID.NotEqual(c.m_agentId)) return; LandUpdateArgs args = new() { AuthBuyerID = parcelPropertiesPacket.ParcelData.AuthBuyerID, Category = (ParcelCategory)parcelPropertiesPacket.ParcelData.Category, Desc = Utils.BytesToString(parcelPropertiesPacket.ParcelData.Desc), GroupID = parcelPropertiesPacket.ParcelData.GroupID, LandingType = parcelPropertiesPacket.ParcelData.LandingType, MediaAutoScale = parcelPropertiesPacket.ParcelData.MediaAutoScale, MediaID = parcelPropertiesPacket.ParcelData.MediaID, MediaURL = Utils.BytesToString(parcelPropertiesPacket.ParcelData.MediaURL), MusicURL = Utils.BytesToString(parcelPropertiesPacket.ParcelData.MusicURL), Name = Utils.BytesToString(parcelPropertiesPacket.ParcelData.Name), ParcelFlags = parcelPropertiesPacket.ParcelData.ParcelFlags, PassHours = parcelPropertiesPacket.ParcelData.PassHours, PassPrice = parcelPropertiesPacket.ParcelData.PassPrice, SalePrice = parcelPropertiesPacket.ParcelData.SalePrice, SnapshotID = parcelPropertiesPacket.ParcelData.SnapshotID, UserLocation = parcelPropertiesPacket.ParcelData.UserLocation, UserLookAt = parcelPropertiesPacket.ParcelData.UserLookAt }; c.OnParcelPropertiesUpdateRequest?.Invoke(args, parcelPropertiesPacket.ParcelData.LocalID, c); } private static void HandleParcelSelectObjects(LLClientView c, Packet Pack) { if(c.OnParcelSelectObjects is null) return; ParcelSelectObjectsPacket selectPacket = (ParcelSelectObjectsPacket)Pack; List returnIDs = new(selectPacket.ReturnIDs.Length); foreach (ParcelSelectObjectsPacket.ReturnIDsBlock rb in selectPacket.ReturnIDs) { returnIDs.Add(rb.ReturnID); } c.OnParcelSelectObjects?.Invoke(selectPacket.ParcelData.LocalID, Convert.ToInt32(selectPacket.ParcelData.ReturnType), returnIDs, c); } private static void HandleParcelObjectOwnersRequest(LLClientView c, Packet Pack) { ParcelObjectOwnersRequestPacket reqPacket = (ParcelObjectOwnersRequestPacket)Pack; c.OnParcelObjectOwnerRequest?.Invoke(reqPacket.ParcelData.LocalID, c); } private static void HandleParcelGodForceOwner(LLClientView c, Packet Pack) { ParcelGodForceOwnerPacket godForceOwnerPacket = (ParcelGodForceOwnerPacket)Pack; c.OnParcelGodForceOwner?.Invoke(godForceOwnerPacket.Data.LocalID, godForceOwnerPacket.Data.OwnerID, c); } private static void HandleParcelRelease(LLClientView c, Packet Pack) { ParcelReleasePacket releasePacket = (ParcelReleasePacket)Pack; c.OnParcelAbandonRequest?.Invoke(releasePacket.Data.LocalID, c); } private static void HandleParcelReclaim(LLClientView c, Packet Pack) { ParcelReclaimPacket reclaimPacket = (ParcelReclaimPacket)Pack; c.OnParcelReclaim?.Invoke(reclaimPacket.Data.LocalID, c); } private static void HandleParcelReturnObjects(LLClientView c, Packet Pack) { if(c.OnParcelReturnObjectsRequest is null) return; ParcelReturnObjectsPacket parcelReturnObjects = (ParcelReturnObjectsPacket)Pack; 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; c.OnParcelReturnObjectsRequest?.Invoke(parcelReturnObjects.ParcelData.LocalID, parcelReturnObjects.ParcelData.ReturnType, puserselectedOwnerIDs, puserselectedTaskIDs, c); } private static void HandleParcelSetOtherCleanTime(LLClientView c, Packet Pack) { ParcelSetOtherCleanTimePacket parcelSetOtherCleanTimePacket = (ParcelSetOtherCleanTimePacket)Pack; c.OnParcelSetOtherCleanTime?.Invoke(c, parcelSetOtherCleanTimePacket.ParcelData.LocalID, parcelSetOtherCleanTimePacket.ParcelData.OtherCleanTime); } private static void HandleLandStatRequest(LLClientView c, Packet Pack) { LandStatRequestPacket lsrp = (LandStatRequestPacket)Pack; c.OnLandStatRequest?.Invoke(lsrp.RequestData.ParcelLocalID, lsrp.RequestData.ReportType, lsrp.RequestData.RequestFlags, Utils.BytesToString(lsrp.RequestData.Filter), c); } private static void HandleParcelDwellRequest(LLClientView c, Packet Pack) { ParcelDwellRequestPacket dwellrq = (ParcelDwellRequestPacket)Pack; c.OnParcelDwellRequest?.Invoke(dwellrq.Data.LocalID, c); } #endregion Parcel related packets #region Estate Packets private static double m_lastMapRegenTime = Double.MinValue; private static void HandleEstateOwnerMessage(LLClientView c, Packet Pack) { EstateOwnerMessagePacket messagePacket = (EstateOwnerMessagePacket)Pack; string method = Utils.BytesToString(messagePacket.MethodData.Method); switch (method) { case "getinfo": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { c.OnDetailedEstateDataRequest?.Invoke(c, messagePacket.MethodData.Invoice); } return; case "setregioninfo": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { c.OnSetEstateFlagsRequest?.Invoke(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; // case "texturebase": // if (((Scene)m_scene).Permissions.CanIssueEstateCommand(m_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 (c.m_scene.Permissions.CanIssueEstateCommand(c.m_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(splitField[1]); c.OnSetEstateTerrainDetailTexture?.Invoke(c, corner, textureUUID); } } } return; case "textureheights": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_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 = Convert.ToSingle(splitField[1], Culture.NumberFormatInfo); float highValue = Convert.ToSingle(splitField[2], Culture.NumberFormatInfo); c.OnSetEstateTerrainTextureHeights?.Invoke(c, corner, lowValue, highValue); } } } return; case "texturecommit": c.OnCommitEstateTerrainTextureRequest?.Invoke(c); return; case "setregionterrain": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if (messagePacket.ParamList.Length != 9) { m_log.Error("EstateOwnerMessage: SetRegionTerrain method has a ParamList of invalid length"); } else { try { float WaterHeight = Convert.ToSingle(Utils.BytesToString(messagePacket.ParamList[0].Parameter), Culture.NumberFormatInfo); float TerrainRaiseLimit = Convert.ToSingle(Utils.BytesToString(messagePacket.ParamList[1].Parameter), Culture.NumberFormatInfo); float TerrainLowerLimit = Convert.ToSingle(Utils.BytesToString(messagePacket.ParamList[2].Parameter), Culture.NumberFormatInfo); bool UseEstateSun = convertParamStringToBool(messagePacket.ParamList[3].Parameter); bool UseFixedSun = convertParamStringToBool(messagePacket.ParamList[4].Parameter); float SunHour = Convert.ToSingle(Utils.BytesToString(messagePacket.ParamList[5].Parameter), Culture.NumberFormatInfo); bool UseGlobal = convertParamStringToBool(messagePacket.ParamList[6].Parameter); bool EstateFixedSun = convertParamStringToBool(messagePacket.ParamList[7].Parameter); float EstateSunHour = Convert.ToSingle(Utils.BytesToString(messagePacket.ParamList[8].Parameter), Culture.NumberFormatInfo); c.OnSetRegionTerrainSettings?.Invoke(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; case "restart": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_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) { if(Utils.TryParseSingle(Utils.BytesToString(block.Parameter), out float timeSeconds)) c.OnEstateRestartSimRequest?.Invoke(c, (int)timeSeconds); } } return; case "estatechangecovenantid": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { foreach (EstateOwnerMessagePacket.ParamListBlock block in messagePacket.ParamList) { UUID newCovenantID = new(Utils.BytesToString(block.Parameter)); c.OnEstateChangeCovenantRequest?.Invoke(c, newCovenantID); } } return; case "estateaccessdelta": // Estate access delta manages the banlist and allow list too. if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { int estateAccessType = Convert.ToInt16(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); c.OnUpdateEstateAccessDeltaRequest?.Invoke(c, messagePacket.MethodData.Invoice, estateAccessType, new UUID(Utils.BytesToString(messagePacket.ParamList[2].Parameter))); } return; case "simulatormessage": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { UUID SenderID = new(Utils.BytesToString(messagePacket.ParamList[2].Parameter)); string SenderName = Utils.BytesToString(messagePacket.ParamList[3].Parameter); string Message = Utils.BytesToString(messagePacket.ParamList[4].Parameter); c.OnSimulatorBlueBoxMessageRequest?.Invoke(c, messagePacket.MethodData.Invoice, SenderID, messagePacket.AgentData.SessionID, SenderName, Message); } return; case "instantmessage": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if (messagePacket.ParamList.Length < 2) return; UUID SenderID; string SenderName; string Message; if (messagePacket.ParamList.Length < 5) { SenderID = c.m_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); } c.OnEstateBlueBoxMessageRequest?.Invoke(c, messagePacket.MethodData.Invoice, SenderID, c.m_sessionId, SenderName, Message); } return; case "setregiondebug": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if(c.OnEstateDebugRegionRequest is not null) { bool scripted = convertParamStringToBool(messagePacket.ParamList[0].Parameter); bool collisionEvents = convertParamStringToBool(messagePacket.ParamList[1].Parameter); bool physics = convertParamStringToBool(messagePacket.ParamList[2].Parameter); c.OnEstateDebugRegionRequest?.Invoke(c, messagePacket.MethodData.Invoice, c.m_agentId, scripted, collisionEvents, physics); } } return; case "teleporthomeuser": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if(UUID.TryParse(Utils.BytesToString(messagePacket.ParamList[1].Parameter), out UUID Prey)) c.OnEstateTeleportOneUserHomeRequest?.Invoke(c, messagePacket.MethodData.Invoice, c.m_agentId, Prey, false); } return; case "teleporthomeallusers": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { c.OnEstateTeleportAllUsersHomeRequest?.Invoke(c, messagePacket.MethodData.Invoice, c.m_agentId); } return; case "colliders": c.OnLandStatRequest?.Invoke(0, 1, 0, "", c); return; case "scripts": c.OnLandStatRequest?.Invoke(0, 0, 0, "", c); return; case "terrain": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if (messagePacket.ParamList.Length > 0) { string p0 = Utils.BytesToString(messagePacket.ParamList[0].Parameter); switch(p0) { case "bake": c.OnBakeTerrain?.Invoke(c); break; case "download filename": if (messagePacket.ParamList.Length > 1) c.OnRequestTerrain?.Invoke(c, Utils.BytesToString(messagePacket.ParamList[1].Parameter)); break; case "upload filename": if (messagePacket.ParamList.Length > 1) c.OnUploadTerrain?.Invoke(c, Utils.BytesToString(messagePacket.ParamList[1].Parameter)); break; default: break; } } } return; case "estatechangeinfo": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { UInt32 param1 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); UInt32 param2 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[2].Parameter)); c.OnEstateChangeInfo?.Invoke(c, messagePacket.MethodData.Invoice, c.m_agentId, param1, param2); } return; case "telehub": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { UInt32 param1 = 0u; string command = Utils.BytesToString(messagePacket.ParamList[0].Parameter); if (command != "info ui") { try { param1 = Convert.ToUInt32(Utils.BytesToString(messagePacket.ParamList[1].Parameter)); } catch {} } c.OnEstateManageTelehub?.Invoke(c, messagePacket.MethodData.Invoice, messagePacket.AgentData.AgentID, command, param1); } return; case "refreshmapvisibility": if (c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { IWorldMapModule mapModule = c.m_scene.RequestModuleInterface(); if (mapModule is null) { c.SendAlertMessage("Terrain map generator not avaiable"); return; } if (m_lastMapRegenTime == Double.MaxValue) { c.SendAlertMessage("Terrain map generation still in progress"); return; } double now = Util.GetTimeStamp(); if (now - m_lastMapRegenTime < 120) // 2 minutes global cool down { c.SendAlertMessage("Please wait at least 2 minutes between map generation commands"); return; } m_lastMapRegenTime = Double.MaxValue; c.m_scene.RegenerateMaptileAndReregister(c, null); c.SendAlertMessage("Terrain map generated"); m_lastMapRegenTime = now; } return; case "kickestate": if(c.m_scene.Permissions.CanIssueEstateCommand(c.m_agentId, false)) { if(UUID.TryParse(Utils.BytesToString(messagePacket.ParamList[0].Parameter), out UUID Prey)) c.OnEstateTeleportOneUserHomeRequest?.Invoke(c, messagePacket.MethodData.Invoice, c.m_agentId, Prey, true); } return; default: m_log.WarnFormat( "[LLCLIENTVIEW]: EstateOwnerMessage: Unknown method {0} requested for {1} in {2}", method, c.Name, c.m_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; } //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 static void HandleRequestRegionInfo(LLClientView c, Packet Pack) { RequestRegionInfoPacket.AgentDataBlock mPacket = ((RequestRegionInfoPacket)Pack).AgentData; c.OnRegionInfoRequest?.Invoke(c); } private static void HandleEstateCovenantRequest(LLClientView c, Packet Pack) { EstateCovenantRequestPacket.AgentDataBlock epack = ((EstateCovenantRequestPacket)Pack).AgentData; c.OnEstateCovenantRequest?.Invoke(c); } #endregion Estate Packets #region GodPackets private static void HandleRequestGodlikePowers(LLClientView c, Packet Pack) { RequestGodlikePowersPacket rglpPack = (RequestGodlikePowersPacket)Pack; RequestGodlikePowersPacket.AgentDataBlock ablock = rglpPack.AgentData; if (ablock.SessionID.NotEqual(c.m_sessionId) || ablock.AgentID.NotEqual(c.m_agentId)) return; RequestGodlikePowersPacket.RequestBlockBlock rblock = rglpPack.RequestBlock; UUID token = rblock.Token; c.OnRequestGodlikePowers?.Invoke(ablock.AgentID, ablock.SessionID, token, rblock.Godlike); } private static void HandleGodUpdateRegionInfoUpdate(LLClientView c, Packet packet) { GodUpdateRegionInfoPacket GodUpdateRegionInfo = (GodUpdateRegionInfoPacket)packet; if (GodUpdateRegionInfo.AgentData.SessionID.NotEqual(c.m_sessionId) || GodUpdateRegionInfo.AgentData.AgentID.NotEqual(c.m_agentId)) return; c.OnGodUpdateRegionInfoUpdate?.Invoke(c, GodUpdateRegionInfo.RegionInfo.BillableFactor, GodUpdateRegionInfo.RegionInfo.EstateID, GodUpdateRegionInfo.RegionInfo.RegionFlags, GodUpdateRegionInfo.RegionInfo.SimName, GodUpdateRegionInfo.RegionInfo.RedirectGridX, GodUpdateRegionInfo.RegionInfo.RedirectGridY); } private static void HandleSimWideDeletes(LLClientView c, Packet packet) { SimWideDeletesPacket SimWideDeletesRequest = (SimWideDeletesPacket)packet; c.OnSimWideDeletes?.Invoke(c, SimWideDeletesRequest.AgentData.AgentID,(int)SimWideDeletesRequest.DataBlock.Flags, SimWideDeletesRequest.DataBlock.TargetID); } private static void HandleGodlikeMessage(LLClientView c, Packet packet) { GodlikeMessagePacket GodlikeMessage = (GodlikeMessagePacket)packet; c.onGodlikeMessage?.Invoke(c, GodlikeMessage.MethodData.Invoice, GodlikeMessage.MethodData.Method, GodlikeMessage.ParamList[0].Parameter); } private static void HandleSaveStatePacket(LLClientView c, Packet packet) { StateSavePacket SaveStateMessage = (StateSavePacket)packet; c.OnSaveState?.Invoke(c, SaveStateMessage.AgentData.AgentID); } private static void HandleGodKickUser(LLClientView c, Packet Pack) { GodKickUserPacket gkupack = (GodKickUserPacket)Pack; c.OnGodKickUser?.Invoke(gkupack.UserInfo.GodID, gkupack.UserInfo.AgentID, gkupack.UserInfo.KickFlags, gkupack.UserInfo.Reason); } #endregion GodPackets #region Economy/Transaction Packets private static void HandleMoneyBalanceRequest(LLClientView c, Packet Pack) { MoneyBalanceRequestPacket moneybalancerequestpacket = (MoneyBalanceRequestPacket)Pack; c.OnMoneyBalanceRequest?.Invoke(c, moneybalancerequestpacket.AgentData.AgentID, moneybalancerequestpacket.AgentData.SessionID, moneybalancerequestpacket.MoneyData.TransactionID); } private static void HandleEconomyDataRequest(LLClientView c, Packet _) { c.OnEconomyDataRequest?.Invoke(c); } private static void HandleRequestPayPrice(LLClientView c, Packet Pack) { RequestPayPricePacket requestPayPricePacket = (RequestPayPricePacket)Pack; c.OnRequestPayPrice?.Invoke(c, requestPayPricePacket.ObjectData.ObjectID); } private static void HandleObjectSaleInfo(LLClientView c, Packet Pack) { if(c.OnObjectSaleInfo is null) return; ObjectSaleInfoPacket objectSaleInfoPacket = (ObjectSaleInfoPacket)Pack; foreach (ObjectSaleInfoPacket.ObjectDataBlock d in objectSaleInfoPacket.ObjectData) { c.OnObjectSaleInfo?.Invoke(c, c.m_agentId, c.SessionId, d.LocalID, d.SaleType, d.SalePrice); } } private static void HandleObjectBuy(LLClientView c, Packet Pack) { if(c.OnObjectBuy is null) return; ObjectBuyPacket objectBuyPacket = (ObjectBuyPacket)Pack; foreach (ObjectBuyPacket.ObjectDataBlock d in objectBuyPacket.ObjectData) { c.OnObjectBuy?.Invoke(c, objectBuyPacket.AgentData.AgentID, objectBuyPacket.AgentData.SessionID, objectBuyPacket.AgentData.GroupID, objectBuyPacket.AgentData.CategoryID, d.ObjectLocalID, d.SaleType, d.SalePrice); } } #endregion Economy/Transaction Packets #region Script Packets private static void HandleGetScriptRunning(LLClientView c, Packet Pack) { GetScriptRunningPacket scriptRunning = (GetScriptRunningPacket)Pack; c.OnGetScriptRunning?.Invoke(c, scriptRunning.Script.ObjectID, scriptRunning.Script.ItemID); } private static void HandleSetScriptRunning(LLClientView c, Packet Pack) { SetScriptRunningPacket setScriptRunning = (SetScriptRunningPacket)Pack; c.OnSetScriptRunning?.Invoke(c, setScriptRunning.Script.ObjectID, setScriptRunning.Script.ItemID, setScriptRunning.Script.Running); } private static void HandleScriptReset(LLClientView c, Packet Pack) { ScriptResetPacket scriptResetPacket = (ScriptResetPacket)Pack; c.OnScriptReset?.Invoke(c, scriptResetPacket.Script.ObjectID, scriptResetPacket.Script.ItemID); } #endregion Script Packets #region Gesture Managment private static void HandleActivateGestures(LLClientView c, Packet Pack) { if(c.OnActivateGesture is null) return; ActivateGesturesPacket activateGesturePacket = (ActivateGesturesPacket)Pack; ActivateGesturesPacket.DataBlock[] data = activateGesturePacket.Data; for (int i= 0; i < data.Length; ++i) { c.OnActivateGesture?.Invoke(c, data[i].AssetID, data[i].ItemID); } } private static void HandleDeactivateGestures(LLClientView c, Packet Pack) { if(c.OnDeactivateGesture is null) return; DeactivateGesturesPacket deactivateGesturePacket = (DeactivateGesturesPacket)Pack; DeactivateGesturesPacket.DataBlock[] data = deactivateGesturePacket.Data; for (int i = 0; i < data.Length; ++i) { c.OnDeactivateGesture?.Invoke(c, data[i].ItemID); } } #endregion Gesture Managment private static void HandleObjectOwner(LLClientView c, Packet Pack) { if (c.OnObjectOwner is null) return; ObjectOwnerPacket objectOwnerPacket = (ObjectOwnerPacket)Pack; List localIDs = new(); foreach (ObjectOwnerPacket.ObjectDataBlock d in objectOwnerPacket.ObjectData) localIDs.Add(d.ObjectLocalID); c.OnObjectOwner?.Invoke(c, objectOwnerPacket.HeaderData.OwnerID, objectOwnerPacket.HeaderData.GroupID, localIDs); } private static void HandleAgentFOV(LLClientView c, Packet Pack) { AgentFOVPacket fovPacket = (AgentFOVPacket)Pack; uint genCounter = fovPacket.FOVBlock.GenCounter; if (genCounter == 0 || genCounter > c.m_agentFOVCounter) { c.m_agentFOVCounter = genCounter; c.FOV = fovPacket.FOVBlock.VerticalAngle; c.OnAgentFOV?.Invoke(c, c.FOV); } } #region unimplemented handlers private static void HandleViewerStats(LLClientView c, Packet Pack) { // TODO: handle this packet //m_log.Warn("[CLIENT]: unhandled ViewerStats packet"); } private static void HandleMapItemRequest(LLClientView c, Packet Pack) { MapItemRequestPacket mirpk = (MapItemRequestPacket)Pack; //m_log.Debug(mirpk.ToString()); try { c.OnMapItemRequest?.Invoke(c, 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} : {2}", LogHeader, e.Message, e.StackTrace); } } private static void HandleTransferAbort(LLClientView c, Packet Pack) { } private static void HandleMuteListRequest(LLClientView c, Packet Pack) { MuteListRequestPacket muteListRequest = (MuteListRequestPacket)Pack; if (c.OnMuteListRequest != null) { c.OnMuteListRequest?.Invoke(c, muteListRequest.MuteData.MuteCRC); } else { if(muteListRequest.MuteData.MuteCRC == 0) c.SendEmpytMuteList(); else c.SendUseCachedMuteList(); } } private static void HandleUpdateMuteListEntry(LLClientView c, Packet packet) { UpdateMuteListEntryPacket UpdateMuteListEntry = (UpdateMuteListEntryPacket)packet; c.OnUpdateMuteListEntry?.Invoke(c, UpdateMuteListEntry.MuteData.MuteID, Utils.BytesToString(UpdateMuteListEntry.MuteData.MuteName), UpdateMuteListEntry.MuteData.MuteType, UpdateMuteListEntry.MuteData.MuteFlags); } private static void HandleRemoveMuteListEntry(LLClientView c, Packet packet) { RemoveMuteListEntryPacket RemoveMuteListEntry = (RemoveMuteListEntryPacket)packet; c.OnRemoveMuteListEntry?.Invoke(c, RemoveMuteListEntry.MuteData.MuteID, Utils.BytesToString(RemoveMuteListEntry.MuteData.MuteName)); } private static void HandleUserReport(LLClientView c, Packet packet) { UserReportPacket UserReport = (UserReportPacket)packet; c.OnUserReport?.Invoke(c, 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); } private static void HandleSendPostcard(LLClientView c, Packet packet) { // SendPostcardPacket SendPostcard = // (SendPostcardPacket)packet; c.OnSendPostcard?.Invoke(c); } private static void HandleChangeInventoryItemFlags(LLClientView c, Packet packet) { if(c.OnChangeInventoryItemFlags is null) return; ChangeInventoryItemFlagsPacket ChangeInventoryItemFlags = (ChangeInventoryItemFlagsPacket)packet; foreach(ChangeInventoryItemFlagsPacket.InventoryDataBlock b in ChangeInventoryItemFlags.InventoryData) c.OnChangeInventoryItemFlags?.Invoke(c, b.ItemID, b.Flags); } private static void HandleUseCircuitCode(LLClientView c, Packet Pack) { /* UseCircuitCodePacket uccp = (UseCircuitCodePacket)Pack; if(uccp.CircuitCode.ID == m_agentId && uccp.CircuitCode.SessionID.Equals(m_sessionId) && uccp.CircuitCode.Code == m_circuitCode && SceneAgent != null && !((ScenePresence)SceneAgent).IsDeleted ) SendRegionHandshake(); // possible someone returning */ } private static void HandleCreateNewOutfitAttachments(LLClientView c, Packet Pack) { if(c.OnMoveItemsAndLeaveCopy is null) return; CreateNewOutfitAttachmentsPacket packet = (CreateNewOutfitAttachmentsPacket)Pack; List items = new(); foreach (CreateNewOutfitAttachmentsPacket.ObjectDataBlock n in packet.ObjectData) { InventoryItemBase b = new() { ID = n.OldItemID, Folder = n.OldFolderID }; items.Add(b); } c.OnMoveItemsAndLeaveCopy?.Invoke(c, items, packet.HeaderData.NewFolderID); } private static void HandleAgentHeightWidth(LLClientView c, Packet Pack) { AgentHeightWidthPacket hwPacket = (AgentHeightWidthPacket)Pack; c.viewHeight = hwPacket.HeightWidthBlock.Height; c.viewWidth = hwPacket.HeightWidthBlock.Width; } private static void HandleInventoryDescendents(LLClientView c, Packet Pack) { } #endregion unimplemented handlers #region Dir handlers private static void HandleDirPlacesQuery(LLClientView c, Packet Pack) { DirPlacesQueryPacket dirPlacesQueryPacket = (DirPlacesQueryPacket)Pack; c.OnDirPlacesQuery?.Invoke(c, dirPlacesQueryPacket.QueryData.QueryID, Utils.BytesToString( dirPlacesQueryPacket.QueryData.QueryText), (int)dirPlacesQueryPacket.QueryData.QueryFlags, (int)dirPlacesQueryPacket.QueryData.Category, Utils.BytesToString( dirPlacesQueryPacket.QueryData.SimName), dirPlacesQueryPacket.QueryData.QueryStart); } private static void HandleDirFindQuery(LLClientView c, Packet Pack) { DirFindQueryPacket dirFindQueryPacket = (DirFindQueryPacket)Pack; c.OnDirFindQuery?.Invoke(c, dirFindQueryPacket.QueryData.QueryID, Utils.BytesToString( dirFindQueryPacket.QueryData.QueryText).Trim(), dirFindQueryPacket.QueryData.QueryFlags, dirFindQueryPacket.QueryData.QueryStart); } private static void HandleDirLandQuery(LLClientView c, Packet Pack) { DirLandQueryPacket dirLandQueryPacket = (DirLandQueryPacket)Pack; c.OnDirLandQuery?.Invoke(c, dirLandQueryPacket.QueryData.QueryID, dirLandQueryPacket.QueryData.QueryFlags, dirLandQueryPacket.QueryData.SearchType, dirLandQueryPacket.QueryData.Price, dirLandQueryPacket.QueryData.Area, dirLandQueryPacket.QueryData.QueryStart); } private static void HandleDirPopularQuery(LLClientView c, Packet Pack) { DirPopularQueryPacket dirPopularQueryPacket = (DirPopularQueryPacket)Pack; c.OnDirPopularQuery?.Invoke(c, dirPopularQueryPacket.QueryData.QueryID, dirPopularQueryPacket.QueryData.QueryFlags); } private static void HandleDirClassifiedQuery(LLClientView c, Packet Pack) { DirClassifiedQueryPacket dirClassifiedQueryPacket = (DirClassifiedQueryPacket)Pack; c.OnDirClassifiedQuery?.Invoke(c, dirClassifiedQueryPacket.QueryData.QueryID, Utils.BytesToString( dirClassifiedQueryPacket.QueryData.QueryText), dirClassifiedQueryPacket.QueryData.QueryFlags, dirClassifiedQueryPacket.QueryData.Category, dirClassifiedQueryPacket.QueryData.QueryStart); } private static void HandleEventInfoRequest(LLClientView c, Packet Pack) { EventInfoRequestPacket eventInfoRequestPacket = (EventInfoRequestPacket)Pack; c.OnEventInfoRequest?.Invoke(c, eventInfoRequestPacket.EventData.EventID); } #endregion #region Calling Card private static void HandleOfferCallingCard(LLClientView c, Packet Pack) { OfferCallingCardPacket offerCallingCardPacket = (OfferCallingCardPacket)Pack; c.OnOfferCallingCard?.Invoke(c, offerCallingCardPacket.AgentBlock.DestID, offerCallingCardPacket.AgentBlock.TransactionID); } private static void HandleAcceptCallingCard(LLClientView c, Packet Pack) { AcceptCallingCardPacket acceptCallingCardPacket = (AcceptCallingCardPacket)Pack; // according to http://wiki.secondlife.com/wiki/AcceptCallingCard FolderData should // contain exactly one entry if (acceptCallingCardPacket.FolderData.Length > 0) { c.OnAcceptCallingCard?.Invoke(c, acceptCallingCardPacket.TransactionBlock.TransactionID, acceptCallingCardPacket.FolderData[0].FolderID); } } private static void HandleDeclineCallingCard(LLClientView c, Packet Pack) { DeclineCallingCardPacket declineCallingCardPacket = (DeclineCallingCardPacket)Pack; c.OnDeclineCallingCard?.Invoke(c, declineCallingCardPacket.TransactionBlock.TransactionID); } #endregion Calling Card #region Groups private static void HandleActivateGroup(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; ActivateGroupPacket activateGroupPacket = (ActivateGroupPacket)Pack; c.m_GroupsModule.ActivateGroup(c, activateGroupPacket.AgentData.GroupID); } private static void HandleGroupVoteHistoryRequest(LLClientView c, Packet packet) { GroupVoteHistoryRequestPacket GroupVoteHistoryRequest = (GroupVoteHistoryRequestPacket)packet; c.OnGroupVoteHistoryRequest?.Invoke(c, c.m_agentId, c.SessionId, GroupVoteHistoryRequest.GroupData.GroupID, GroupVoteHistoryRequest.TransactionData.TransactionID); } private static void HandleGroupActiveProposalsRequest(LLClientView c, Packet packet) { GroupActiveProposalsRequestPacket GroupActiveProposalsRequest = (GroupActiveProposalsRequestPacket)packet; c.OnGroupActiveProposalsRequest?.Invoke(c, c.m_agentId, c.SessionId, GroupActiveProposalsRequest.GroupData.GroupID, GroupActiveProposalsRequest.TransactionData.TransactionID); } private static void HandleGroupAccountDetailsRequest(LLClientView c, Packet packet) { GroupAccountDetailsRequestPacket GroupAccountDetailsRequest = (GroupAccountDetailsRequestPacket)packet; c.OnGroupAccountDetailsRequest?.Invoke(c, c.m_agentId, GroupAccountDetailsRequest.AgentData.GroupID, GroupAccountDetailsRequest.MoneyData.RequestID, c.SessionId); } private static void HandleGroupAccountSummaryRequest(LLClientView c, Packet packet) { GroupAccountSummaryRequestPacket GroupAccountSummaryRequest = (GroupAccountSummaryRequestPacket)packet; c.OnGroupAccountSummaryRequest?.Invoke(c, c.m_agentId, GroupAccountSummaryRequest.AgentData.GroupID); } private static void HandleGroupTransactionsDetailsRequest(LLClientView c, Packet packet) { GroupAccountTransactionsRequestPacket GroupAccountTransactionsRequest = (GroupAccountTransactionsRequestPacket)packet; c.OnGroupAccountTransactionsRequest?.Invoke(c, c.m_agentId, GroupAccountTransactionsRequest.AgentData.GroupID, GroupAccountTransactionsRequest.MoneyData.RequestID, c.SessionId); } private static void HandleGroupTitlesRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupTitlesRequestPacket groupTitlesRequest = (GroupTitlesRequestPacket)Pack; GroupTitlesReplyPacket groupTitlesReply = (GroupTitlesReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupTitlesReply); groupTitlesReply.AgentData = new GroupTitlesReplyPacket.AgentDataBlock { AgentID = c.m_agentId, GroupID = groupTitlesRequest.AgentData.GroupID, RequestID = groupTitlesRequest.AgentData.RequestID }; List titles = c.m_GroupsModule.GroupTitlesRequest(c, groupTitlesRequest.AgentData.GroupID); groupTitlesReply.GroupData = new GroupTitlesReplyPacket.GroupDataBlock[titles.Count]; int i = 0; foreach (GroupTitlesData d in titles) { groupTitlesReply.GroupData[i++] = new GroupTitlesReplyPacket.GroupDataBlock { Title = Util.StringToBytes256(d.Name), RoleID = d.UUID, Selected = d.Selected }; } c.OutPacket(groupTitlesReply, ThrottleOutPacketType.Task); } UUID lastGroupProfileRequestID = UUID.Zero; double lastGroupProfileRequestTS = Util.GetTimeStampMS(); private static void HandleGroupProfileRequest(LLClientView c, Packet Pack) { if(c.m_GroupsModule is null) return; GroupProfileRequestPacket groupProfileRequest = (GroupProfileRequestPacket)Pack; UUID grpID = groupProfileRequest.GroupData.GroupID; double ts = Util.GetTimeStampMS(); if(grpID == c.lastGroupProfileRequestID && ts - c.lastGroupProfileRequestTS < 10000) return; c.lastGroupProfileRequestID = grpID; c.lastGroupProfileRequestTS = ts; GroupProfileReplyPacket groupProfileReply = (GroupProfileReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupProfileReply); groupProfileReply.AgentData = new GroupProfileReplyPacket.AgentDataBlock(); groupProfileReply.GroupData = new GroupProfileReplyPacket.GroupDataBlock(); groupProfileReply.AgentData.AgentID = c.m_agentId; GroupProfileData d = c.m_GroupsModule.GroupProfileRequest(c, groupProfileRequest.GroupData.GroupID); if(d.GroupID.IsZero()) // don't send broken data return; 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; if (c.m_scene.Permissions.IsGod(c.m_agentId) && (!c.IsGroupMember(groupProfileRequest.GroupData.GroupID))) { if (c.m_scene.TryGetScenePresence(c.m_agentId, out ScenePresence p)) { if (p.IsViewerUIGod) { groupProfileReply.GroupData.OpenEnrollment = true; groupProfileReply.GroupData.MembershipFee = 0; } } } c.OutPacket(groupProfileReply, ThrottleOutPacketType.Task); if(grpID.Equals(c.lastGroupProfileRequestID)) c.lastGroupProfileRequestTS = Util.GetTimeStampMS() - 7000; } private static void HandleGroupMembersRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupMembersRequestPacket groupMembersRequestPacket = (GroupMembersRequestPacket)Pack; List members = c.m_GroupsModule.GroupMembersRequest(c, 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 = c.m_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 { AgentID = m.AgentID, Contribution = m.Contribution, OnlineStatus = Util.StringToBytes256(m.OnlineStatus), AgentPowers = m.AgentPowers, Title = Util.StringToBytes256(m.Title), IsOwner = m.IsOwner }; } c.OutPacket(groupMembersReply, ThrottleOutPacketType.Task); } } private static void HandleGroupRoleDataRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupRoleDataRequestPacket groupRolesRequest = (GroupRoleDataRequestPacket)Pack; GroupRoleDataReplyPacket groupRolesReply = (GroupRoleDataReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupRoleDataReply); groupRolesReply.AgentData = new GroupRoleDataReplyPacket.AgentDataBlock { AgentID = c.m_agentId }; groupRolesReply.GroupData = new GroupRoleDataReplyPacket.GroupDataBlock { GroupID = groupRolesRequest.GroupData.GroupID, RequestID = groupRolesRequest.GroupData.RequestID }; List titles = c.m_GroupsModule.GroupRoleDataRequest(c, 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 { RoleID = d.RoleID, Name = Util.StringToBytes256(d.Name), Title = Util.StringToBytes256(d.Title), Description = Util.StringToBytes1024(d.Description), Powers = d.Powers, Members = (uint)d.Members }; i++; } c.OutPacket(groupRolesReply, ThrottleOutPacketType.Task); } private static void HandleGroupRoleMembersRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupRoleMembersRequestPacket groupRoleMembersRequest = (GroupRoleMembersRequestPacket)Pack; List mappings = c.m_GroupsModule.GroupRoleMembersRequest(c, 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 { AgentID = c.m_agentId, GroupID = groupRoleMembersRequest.GroupData.GroupID, RequestID = groupRoleMembersRequest.GroupData.RequestID, 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 { RoleID = d.RoleID, MemberID = d.MemberID }; } c.OutPacket(groupRoleMembersReply, ThrottleOutPacketType.Task); } } private static void HandleCreateGroupRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; CreateGroupRequestPacket createGroupRequest = (CreateGroupRequestPacket)Pack; c.m_GroupsModule.CreateGroup(c, 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); } private static void HandleUpdateGroupInfo(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; UpdateGroupInfoPacket updateGroupInfo = (UpdateGroupInfoPacket)Pack; c.m_GroupsModule.UpdateGroupInfo(c, 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); } private static void HandleSetGroupAcceptNotices(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; SetGroupAcceptNoticesPacket setGroupAcceptNotices = (SetGroupAcceptNoticesPacket)Pack; c.m_GroupsModule.SetGroupAcceptNotices(c, setGroupAcceptNotices.Data.GroupID, setGroupAcceptNotices.Data.AcceptNotices, setGroupAcceptNotices.NewData.ListInProfile); } private static void HandleGroupTitleUpdate(LLClientView c, Packet Pack) { GroupTitleUpdatePacket groupTitleUpdate = (GroupTitleUpdatePacket)Pack; c.m_GroupsModule.GroupTitleUpdate(c, groupTitleUpdate.AgentData.GroupID, groupTitleUpdate.AgentData.TitleRoleID); } private static void HandleParcelDeedToGroup(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; ParcelDeedToGroupPacket parcelDeedToGroup = (ParcelDeedToGroupPacket)Pack; c.OnParcelDeedToGroup?.Invoke(parcelDeedToGroup.Data.LocalID, parcelDeedToGroup.Data.GroupID, c); } private static void HandleGroupNoticesListRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupNoticesListRequestPacket groupNoticesListRequest = (GroupNoticesListRequestPacket)Pack; GroupNoticeData[] gn = c.m_GroupsModule.GroupNoticesListRequest(c, groupNoticesListRequest.Data.GroupID); GroupNoticesListReplyPacket groupNoticesListReply = (GroupNoticesListReplyPacket)PacketPool.Instance.GetPacket(PacketType.GroupNoticesListReply); groupNoticesListReply.AgentData = new GroupNoticesListReplyPacket.AgentDataBlock { AgentID = c.m_agentId, 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 { NoticeID = g.NoticeID, Timestamp = g.Timestamp, FromName = Util.StringToBytes256(g.FromName), Subject = Util.StringToBytes256(g.Subject), HasAttachment = g.HasAttachment, AssetType = g.AssetType }; } c.OutPacket(groupNoticesListReply, ThrottleOutPacketType.Task); } private static void HandleGroupNoticeRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupNoticeRequestPacket groupNoticeRequest = (GroupNoticeRequestPacket)Pack; c.m_GroupsModule.GroupNoticeRequest(c, groupNoticeRequest.Data.GroupNoticeID); } private static void HandleGroupRoleUpdate(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupRoleUpdatePacket groupRoleUpdate = (GroupRoleUpdatePacket)Pack; foreach (GroupRoleUpdatePacket.RoleDataBlock d in groupRoleUpdate.RoleData) { c.m_GroupsModule.GroupRoleUpdate(c, groupRoleUpdate.AgentData.GroupID, d.RoleID, Utils.BytesToString(d.Name), Utils.BytesToString(d.Description), Utils.BytesToString(d.Title), d.Powers, d.UpdateType); } c.m_GroupsModule.NotifyChange(groupRoleUpdate.AgentData.GroupID); } private static void HandleGroupRoleChanges(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; GroupRoleChangesPacket groupRoleChanges = (GroupRoleChangesPacket)Pack; foreach (GroupRoleChangesPacket.RoleChangeBlock d in groupRoleChanges.RoleChange) { c.m_GroupsModule.GroupRoleChanges(c, groupRoleChanges.AgentData.GroupID, d.RoleID, d.MemberID, d.Change); } c.m_GroupsModule.NotifyChange(groupRoleChanges.AgentData.GroupID); } private static void HandleJoinGroupRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; JoinGroupRequestPacket joinGroupRequest = (JoinGroupRequestPacket)Pack; c.m_GroupsModule.JoinGroupRequest(c, joinGroupRequest.GroupData.GroupID); } private static void HandleLeaveGroupRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; LeaveGroupRequestPacket leaveGroupRequest = (LeaveGroupRequestPacket)Pack; c.m_GroupsModule.LeaveGroupRequest(c, leaveGroupRequest.GroupData.GroupID); } private static void HandleEjectGroupMemberRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; EjectGroupMemberRequestPacket ejectGroupMemberRequest = (EjectGroupMemberRequestPacket)Pack; foreach (EjectGroupMemberRequestPacket.EjectDataBlock e in ejectGroupMemberRequest.EjectData) { c.m_GroupsModule.EjectGroupMemberRequest(c, ejectGroupMemberRequest.GroupData.GroupID, e.EjecteeID); } } private static void HandleInviteGroupRequest(LLClientView c, Packet Pack) { if (c.m_GroupsModule is null) return; InviteGroupRequestPacket inviteGroupRequest = (InviteGroupRequestPacket)Pack; foreach (InviteGroupRequestPacket.InviteDataBlock b in inviteGroupRequest.InviteData) { c.m_GroupsModule.InviteGroupRequest(c, inviteGroupRequest.GroupData.GroupID, b.InviteeID, b.RoleID); } } #endregion Groups private static void HandleStartLure(LLClientView c, Packet Pack) { if(c.OnStartLure is null) return; StartLurePacket startLureRequest = (StartLurePacket)Pack; for (int i = 0 ; i < startLureRequest.TargetData.Length ; i++) { c.OnStartLure?.Invoke(startLureRequest.Info.LureType, Utils.BytesToString( startLureRequest.Info.Message), startLureRequest.TargetData[i].TargetID, c); } } private static void HandleTeleportLureRequest(LLClientView c, Packet Pack) { TeleportLureRequestPacket teleportLureRequest = (TeleportLureRequestPacket)Pack; c.OnTeleportLureRequest?.Invoke( teleportLureRequest.Info.LureID, teleportLureRequest.Info.TeleportFlags, c); } private static void HandleClassifiedInfoRequest(LLClientView c, Packet Pack) { ClassifiedInfoRequestPacket classifiedInfoRequest = (ClassifiedInfoRequestPacket)Pack; c.OnClassifiedInfoRequest?.Invoke( classifiedInfoRequest.Data.ClassifiedID, c); } private static void HandleClassifiedInfoUpdate(LLClientView c, Packet Pack) { ClassifiedInfoUpdatePacket classifiedInfoUpdate = (ClassifiedInfoUpdatePacket)Pack; // fix classifiedFlags maturity byte classifiedFlags = classifiedInfoUpdate.Data.ClassifiedFlags; if ((classifiedFlags & 0x4e) == 0) // if none classifiedFlags |= 0x4; // pg c.OnClassifiedInfoUpdate?.Invoke( 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), classifiedFlags, classifiedInfoUpdate.Data.PriceForListing, c); } private static void HandleClassifiedDelete(LLClientView c, Packet Pack) { ClassifiedDeletePacket classifiedDelete = (ClassifiedDeletePacket)Pack; c.OnClassifiedDelete?.Invoke( classifiedDelete.Data.ClassifiedID, c); } private static void HandleClassifiedGodDelete(LLClientView c, Packet Pack) { ClassifiedGodDeletePacket classifiedGodDelete = (ClassifiedGodDeletePacket)Pack; c.OnClassifiedGodDelete?.Invoke( classifiedGodDelete.Data.ClassifiedID, classifiedGodDelete.Data.QueryID, c); } private static void HandleEventGodDelete(LLClientView c, Packet Pack) { EventGodDeletePacket eventGodDelete = (EventGodDeletePacket)Pack; c.OnEventGodDelete?.Invoke( eventGodDelete.EventData.EventID, eventGodDelete.QueryData.QueryID, Utils.BytesToString( eventGodDelete.QueryData.QueryText), eventGodDelete.QueryData.QueryFlags, eventGodDelete.QueryData.QueryStart, c); } private static void HandleEventNotificationAddRequest(LLClientView c, Packet Pack) { EventNotificationAddRequestPacket eventNotificationAdd = (EventNotificationAddRequestPacket)Pack; c.OnEventNotificationAddRequest?.Invoke(eventNotificationAdd.EventData.EventID, c); } private static void HandleEventNotificationRemoveRequest(LLClientView c, Packet Pack) { EventNotificationRemoveRequestPacket eventNotificationRemove = (EventNotificationRemoveRequestPacket)Pack; c.OnEventNotificationRemoveRequest?.Invoke(eventNotificationRemove.EventData.EventID, c); } private static void HandleRetrieveInstantMessages(LLClientView c, Packet Pack) { RetrieveInstantMessagesPacket rimpInstantMessagePack = (RetrieveInstantMessagesPacket)Pack; c.OnRetrieveInstantMessages?.Invoke(c); } private static void HandlePickDelete(LLClientView c, Packet Pack) { PickDeletePacket pickDelete = (PickDeletePacket)Pack; c.OnPickDelete?.Invoke(c, pickDelete.Data.PickID); } private static void HandlePickGodDelete(LLClientView c, Packet Pack) { PickGodDeletePacket pickGodDelete = (PickGodDeletePacket)Pack; c.OnPickGodDelete?.Invoke(c, pickGodDelete.AgentData.AgentID, pickGodDelete.Data.PickID, pickGodDelete.Data.QueryID); } private static void HandlePickInfoUpdate(LLClientView c, Packet Pack) { PickInfoUpdatePacket pickInfoUpdate = (PickInfoUpdatePacket)Pack; c.OnPickInfoUpdate?.Invoke(c, 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); } private static void HandleAvatarNotesUpdate(LLClientView c, Packet Pack) { AvatarNotesUpdatePacket avatarNotesUpdate = (AvatarNotesUpdatePacket)Pack; c.OnAvatarNotesUpdate?.Invoke(c, avatarNotesUpdate.Data.TargetID, Utils.BytesToString(avatarNotesUpdate.Data.Notes)); } private static void HandleAvatarInterestsUpdate(LLClientView c, Packet Pack) { AvatarInterestsUpdatePacket avatarInterestUpdate = (AvatarInterestsUpdatePacket)Pack; c.OnAvatarInterestUpdate?.Invoke(c, avatarInterestUpdate.PropertiesData.WantToMask, Utils.BytesToString(avatarInterestUpdate.PropertiesData.WantToText), avatarInterestUpdate.PropertiesData.SkillsMask, Utils.BytesToString(avatarInterestUpdate.PropertiesData.SkillsText), Utils.BytesToString(avatarInterestUpdate.PropertiesData.LanguagesText)); } private static void HandleGrantUserRights(LLClientView c, Packet Pack) { GrantUserRightsPacket GrantUserRights = (GrantUserRightsPacket)Pack; c.OnGrantUserRights?.Invoke(c, GrantUserRights.Rights[0].AgentRelated, GrantUserRights.Rights[0].RelatedRights); } private double m_nextRevokePermissionsTime = Double.MinValue; private uint m_lastRevokePermissionsSeq = uint.MinValue; private static void HandleRevokePermissions(LLClientView c, Packet Pack) { RevokePermissionsPacket pkt = (RevokePermissionsPacket)Pack; uint thisSeq = pkt.Header.Sequence; if (thisSeq == c.m_lastRevokePermissionsSeq) return; c.m_lastRevokePermissionsSeq = thisSeq; ScenePresence sp = (ScenePresence)c.SceneAgent; if(sp != null && !sp.IsDeleted && !sp.IsInTransit) { UUID objectID = pkt.Data.ObjectID; double now = Util.GetTimeStampMS(); if (now < c.m_nextRevokePermissionsTime) return; if (objectID == c.m_scene.RegionInfo.RegionID) c.m_nextRevokePermissionsTime = now + 2000; else c.m_nextRevokePermissionsTime = now + 50; uint permissions = pkt.Data.ObjectPermissions; sp.HandleRevokePermissions(objectID , permissions); } } private static void HandlePlacesQuery(LLClientView c, Packet Pack) { PlacesQueryPacket placesQueryPacket = (PlacesQueryPacket)Pack; c.OnPlacesQuery?.Invoke(placesQueryPacket.AgentData.QueryID, placesQueryPacket.TransactionData.TransactionID, Utils.BytesToString( placesQueryPacket.QueryData.QueryText), placesQueryPacket.QueryData.QueryFlags, (byte)placesQueryPacket.QueryData.Category, Utils.BytesToString( placesQueryPacket.QueryData.SimName), c); } #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.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); } protected static void HandleLogout(LLClientView c, Packet packet) { var lrp = (LogoutRequestPacket)packet; m_log.Info($"[CLIENT]: Got a logout request for {c.Name} in {c.Scene.Name}"); c.OnLogout?.Invoke(c); } /// /// /// /// /// protected virtual void Logout(IClientAPI client) { m_log.Info($"[CLIENT]: Got a logout request for {Name} in {Scene.Name}"); OnLogout?.Invoke(client); } /// /// /// /// /// /// /// protected static void HandleAgentTextureCached(LLClientView c, Packet packet) { //m_log.Debug("texture cached: " + packet.ToString()); AgentCachedTexturePacket cachedtex = (AgentCachedTexturePacket)packet; AgentCachedTextureResponsePacket cachedresp = (AgentCachedTextureResponsePacket)PacketPool.Instance.GetPacket(PacketType.AgentCachedTextureResponse); ScenePresence p = c.m_scene.GetScenePresence(c.m_agentId); if (p is null) return; WearableCacheItem[] cacheItems = p.Appearance?.WearableCacheItems; cachedresp.AgentData.AgentID = c.m_agentId; cachedresp.AgentData.SessionID = c.m_sessionId; cachedresp.AgentData.SerialNum = cachedtex.AgentData.SerialNum; cachedresp.WearableData = new AgentCachedTextureResponsePacket.WearableDataBlock[cachedtex.WearableData.Length]; if (cacheItems is null) { for (int i = 0; i < cachedtex.WearableData.Length; i++) { cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock() { TextureIndex = cachedtex.WearableData[i].TextureIndex, TextureID = UUID.Zero, HostName = Array.Empty() }; } } else { for (int i = 0; i < cachedtex.WearableData.Length; i++) { var checkdWear = cachedtex.WearableData[i]; int idx = checkdWear.TextureIndex; if (idx < cacheItems.Length) { var cachedIt = cacheItems[idx]; cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock() { TextureIndex = (byte)idx, TextureID = checkdWear.ID.Equals(cachedIt.CacheId) ? cachedIt.TextureID : UUID.Zero, HostName = Array.Empty() }; } else { cachedresp.WearableData[i] = new AgentCachedTextureResponsePacket.WearableDataBlock() { TextureIndex = (byte)idx, TextureID = UUID.Zero, HostName = Array.Empty() }; } } } cachedresp.Header.Zerocoded = true; c.OutPacket(cachedresp, ThrottleOutPacketType.Task); } /// /// 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) { if (avatar is not ScenePresence) 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() { TextureIndex = (byte)cachedTextures[i].BakedTextureIndex, TextureID = cachedTextures[i].BakedTextureID, HostName = Array.Empty() }; } cachedresp.Header.Zerocoded = true; OutPacket(cachedresp, ThrottleOutPacketType.Task); } protected static void HandleMultipleObjUpdate(LLClientView c, Packet packet) { MultipleObjectUpdatePacket multipleupdate = (MultipleObjectUpdatePacket)packet; // m_log.DebugFormat( // "[CLIENT]: Incoming MultipleObjectUpdatePacket contained {0} blocks", multipleupdate.ObjectData.Length); Scene tScene = c.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 is null) { // It's a ghost! tell the client to delete it from view. c.SendKillObject([localId]); } else { ClientChangeObject updatehandler = c.onClientChangeObject; if (updatehandler != null) { ObjectChangeData udata = new(); /*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, c); break; case 2: // rotation sp udata.rotation = new Quaternion(block.Data, 0, true); udata.change = ObjectChangeType.primR; updatehandler(localId, udata, c); 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, c); break; case 4: // scale sp udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.primS; updatehandler(localId, udata, c); break; case 0x14: // uniform scale sp udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.primUS; updatehandler(localId, udata, c); 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, c); 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, c); 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, c); break; case 0x0A: // (8 + 2) group rotation udata.rotation = new Quaternion(block.Data, 0, true); udata.change = ObjectChangeType.groupR; updatehandler(localId, udata, c); 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, c); 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, c); 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, c); break; case 0x1C: // (0x10 + 8 + 4 ) group scale UNIFORM udata.scale = new Vector3(block.Data, 0); udata.change = ObjectChangeType.groupUS; updatehandler(localId, udata, c); 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, c); break; default: m_log.Debug("[CLIENT]: MultipleObjUpdate recieved an unknown packet type: " + (block.Type)); break; } } } } } } 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 = m_agentId; mapReply.AgentData.Flags = 0; mapReply.LayerData = new MapLayerReplyPacket.LayerDataBlock[1]; mapReply.LayerData[0] = new MapLayerReplyPacket.LayerDataBlock { Bottom = 0, Left = 0, Top = 30000, Right = 30000, ImageID = new UUID("00000000-0000-1111-9999-000000000006") }; mapReply.Header.Zerocoded = true; OutPacket(mapReply, ThrottleOutPacketType.Land); } public static 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 = m_agentId; int len; if (simMapProfiles is 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); OnUpdateThrottles?.Invoke(); } /// /// 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, 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 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 is not 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) { if(OnAutoPilotGo is null) return; Utils.LongToUInts(m_scene.RegionInfo.RegionHandle, out uint regionX, out uint regionY); float locx = (float)(Convert.ToDouble(args[0]) - (double)regionX); float locy = (float)(Convert.ToDouble(args[1]) - (double)regionY); float locz = Convert.ToSingle(args[2]); OnAutoPilotGo?.Invoke(new Vector3(locx, locy, locz), false, true); } /// /// 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.Debug( $"[CLIENT]: PACKET IN from {Name} ({(SceneAgent.IsChildAgent ? "child" : "root")}) in {Scene.Name} - {packet.Type}"); } if (!ProcessPacketMethod(packet)) m_log.Warn($"[CLIENT]: ignoring unhandled packet {packet.Type} from {Name} ({(SceneAgent.IsChildAgent ? "child" : "root")}) in {m_scene.Name}"); } private static PrimitiveBaseShape GetShapeFromAddPacket(ObjectAddPacket addPacket) { PrimitiveBaseShape shape = new() { PCode = addPacket.ObjectData.PCode, State = addPacket.ObjectData.State, LastAttachPoint = addPacket.ObjectData.State, PathBegin = addPacket.ObjectData.PathBegin, PathEnd = addPacket.ObjectData.PathEnd, PathScaleX = addPacket.ObjectData.PathScaleX, PathScaleY = addPacket.ObjectData.PathScaleY, PathShearX = addPacket.ObjectData.PathShearX, PathShearY = addPacket.ObjectData.PathShearY, PathSkew = addPacket.ObjectData.PathSkew, ProfileBegin = addPacket.ObjectData.ProfileBegin, ProfileEnd = addPacket.ObjectData.ProfileEnd, Scale = addPacket.ObjectData.Scale, PathCurve = addPacket.ObjectData.PathCurve, ProfileCurve = addPacket.ObjectData.ProfileCurve, ProfileHollow = addPacket.ObjectData.ProfileHollow, PathRadiusOffset = addPacket.ObjectData.PathRadiusOffset, PathRevolutions = addPacket.ObjectData.PathRevolutions, PathTaperX = addPacket.ObjectData.PathTaperX, PathTaperY = addPacket.ObjectData.PathTaperY, PathTwist = addPacket.ObjectData.PathTwist, PathTwistBegin = addPacket.ObjectData.PathTwistBegin }; Primitive.TextureEntry ntex = new(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; 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(); 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(); 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() { Type = pair.Key, 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(); /// /// Register an interface on this client, should only be called in the constructor. /// /// /// protected void RegisterInterface(T iface) { lock (m_clientInterfaces) { m_clientInterfaces.TryAdd(typeof(T), iface); } } public bool TryGet(out T iface) { if(m_clientInterfaces.TryGetValue(typeof(T), out object o)) { iface = (T)o; return true; } iface = default; 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(m_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(m_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 is 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) { int sourceType = transferRequest.TransferInfo.SourceType; var requestID = sourceType switch { (int)SourceType.Asset => new UUID(transferRequest.TransferInfo.Params, 0), (int)SourceType.SimInventoryItem => new UUID(transferRequest.TransferInfo.Params, 80), (int)SourceType.SimEstate => taskID, _ => UUID.Zero, }; //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; if (asset is null) { // Try the user's asset server IInventoryAccessModule inventoryAccessModule = Scene.RequestModuleInterface(); if (inventoryAccessModule.IsForeignUser(m_agentId, out string assetServerURL) && !string.IsNullOrEmpty(assetServerURL)) { if (!assetServerURL.EndsWith('/') && !assetServerURL.EndsWith('=')) 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 is null) { SendAssetNotFound( new AssetRequestToClient() { AssetInf = null, AssetRequestSource = source, IsTextureRequest = false, NumPackets = 0, Params = transferRequest.TransferInfo.Params, RequestAssetID = requestID, TransferRequestID = transferRequest.TransferInfo.TransferID }); 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 == (sbyte)AssetType.LSLText) return; // The asset is known to exist and is in our cache, so add it to the AssetRequests list SendAsset(new AssetRequestToClient() { AssetInf = asset, AssetRequestSource = source, IsTextureRequest = false, NumPackets = CalculateNumPackets(asset.Data), Params = transferRequest.TransferInfo.Params, RequestAssetID = requestID, TransferRequestID = transferRequest.TransferInfo.TransferID }); } /// /// Calculate the number of packets required to send the asset to the client. /// /// /// private static int CalculateNumPackets(byte[] data) { if (data is null || data.Length == 0) return 0; if (data.LongLength > MaxTransferBytesPerPacket) { // over max number of bytes so split up file long restData = data.LongLength - MaxTransferBytesPerPacket; int restPackets = (int)((restData + MaxTransferBytesPerPacket - 1) / MaxTransferBytesPerPacket); return restPackets + 1; } else return 1; } public void SendRebakeAvatarTextures(UUID textureID) { RebakeAvatarTexturesPacket pack = (RebakeAvatarTexturesPacket)PacketPool.Instance.GetPacket(PacketType.RebakeAvatarTextures); pack.TextureData = new RebakeAvatarTexturesPacket.TextureDataBlock { TextureID = textureID }; OutPacket(pack, ThrottleOutPacketType.Task); } public readonly struct PacketProcessor(PacketMethod m, bool doasync ) { public readonly PacketMethod method = m; public readonly bool Async = doasync; } private readonly struct StaticPacketProcessor(StaticPacketMethod m, bool doasync) { public readonly StaticPacketMethod method = m; public readonly bool Async = doasync; } 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.AgentID = m_agentId; packet.AgentData.AvatarID = avatarID; 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.AgentID = agentID; packet.Rights = new ChangeUserRightsPacket.RightsBlock[] { new ChangeUserRightsPacket.RightsBlock { AgentRelated = friendID, 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); dialog.Buttons = new ScriptDialogPacket.ButtonsBlock[] { new ScriptDialogPacket.ButtonsBlock { ButtonLabel = Util.StringToBytes256("!!llTextBox!!") } }; dialog.OwnerData = new ScriptDialogPacket.OwnerDataBlock[] { new ScriptDialogPacket.OwnerDataBlock { OwnerID = ownerID } }; OutPacket(dialog, ThrottleOutPacketType.Task); } public void SendAgentTerseUpdate(ISceneEntity p) { if (p is ScenePresence sp) { UDPPacketBuffer buf = OpenSimUDPBase.GetNewUDPBuffer(m_udpClient.RemoteEndPoint); //setup header and regioninfo block Buffer.BlockCopy(terseUpdateHeader, 0, buf.Data, 0, 7); unsafe { fixed(byte* bdata = &buf.Data[0]) { Utils.UInt64ToBytes(m_scene.RegionInfo.RegionHandle, bdata + 7); Utils.UInt16ToBytes(Utils.FloatZeroOneToushort(m_scene.TimeDilation), bdata + 15); bdata[17] = 1; byte* data = bdata + 18; CreateAvatartImprovedTerseBlock(sp, ref data); buf.DataLength = (int)(data - bdata); } } m_udpServer.SendUDPPacket(m_udpClient, buf, ThrottleOutPacketType.Task, null, true); } } public void SendPlacesReply(UUID queryID, UUID transactionID, PlacesReplyData[] data) { PlacesReplyPacket reply = null; PlacesReplyPacket.QueryDataBlock[] dataBlocks = Array.Empty(); for (int i = 0 ; i < data.Length ; i++) { PlacesReplyPacket.QueryDataBlock block = new() { OwnerID = data[i].OwnerID, Name = Util.StringToBytes256(data[i].Name), Desc = Util.StringToBytes1024(data[i].Desc), ActualArea = data[i].ActualArea, BillableArea = data[i].BillableArea, Flags = data[i].Flags, GlobalX = data[i].GlobalX, GlobalY = data[i].GlobalY, GlobalZ = data[i].GlobalZ, SimName = Util.StringToBytes256(data[i].SimName), SnapshotID = data[i].SnapshotID, Dwell = data[i].Dwell, Price = data[i].Price }; if (reply is not null && reply.Length + block.Length > 1400) { OutPacket(reply, ThrottleOutPacketType.Task); reply = null; dataBlocks = Array.Empty(); } if (reply is null) { reply = (PlacesReplyPacket)PacketPool.Instance.GetPacket(PacketType.PlacesReply); reply.AgentData.AgentID = m_agentId; reply.AgentData.QueryID = queryID; reply.TransactionData.TransactionID = transactionID; reply.QueryData = dataBlocks; } Array.Resize(ref dataBlocks, dataBlocks.Length + 1); dataBlocks[^1] = block; reply.QueryData = dataBlocks; } if (reply != null) OutPacket(reply, ThrottleOutPacketType.Task); } public void SendRemoveInventoryItems(UUID[] items) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } osUTF8 sb = eq.StartEvent("RemoveInventoryItem"); LLSDxmlEncode2.AddArrayAndMap("AgentData", sb); LLSDxmlEncode2.AddElem("AgentID", m_agentId, sb); LLSDxmlEncode2.AddElem("SessionID", SessionId, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); LLSDxmlEncode2.AddArray("InventoryData", sb); foreach (UUID item in items) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("ItemID",item, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } public void SendRemoveInventoryFolders(UUID[] folders) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } osUTF8 sb = eq.StartEvent("RemoveInventoryFolder"); LLSDxmlEncode2.AddArrayAndMap("AgentData", sb); LLSDxmlEncode2.AddElem("AgentID", m_agentId, sb); LLSDxmlEncode2.AddElem("SessionID", SessionId, sb); LLSDxmlEncode2.AddEndMapAndArray(sb); LLSDxmlEncode2.AddArray("FolderData", sb); foreach (UUID folder in folders) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("FolderID", folder, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } public void SendBulkUpdateInventory(InventoryFolderBase[] folders, InventoryItemBase[] items) { IEventQueue eq = Scene.RequestModuleInterface(); if (eq is null) { m_log.DebugFormat("[LLCLIENT]: Null event queue"); return; } osUTF8 sb = eq.StartEvent("BulkUpdateInventory"); LLSDxmlEncode2.AddArrayAndMap("AgentData", sb); LLSDxmlEncode2.AddElem("AgentID", m_agentId, sb); LLSDxmlEncode2.AddElem("TransactionID", UUID.Random(), sb); LLSDxmlEncode2.AddEndMapAndArray(sb); if(folders.Length == 0) { LLSDxmlEncode2.AddEmptyArray("FolderData", sb); } else { LLSDxmlEncode2.AddArray("FolderData", sb); foreach (InventoryFolderBase folder in folders) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("FolderID", folder.ID, sb); LLSDxmlEncode2.AddElem("ParentID", folder.ParentID, sb); LLSDxmlEncode2.AddElem("Type", (int)folder.Type, sb); LLSDxmlEncode2.AddElem("Name", folder.Name, sb); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); } if(items.Length == 0) { LLSDxmlEncode2.AddEmptyArray("ItemData", sb); } else { LLSDxmlEncode2.AddArray("ItemData", sb); foreach (InventoryItemBase item in items) { LLSDxmlEncode2.AddMap(sb); LLSDxmlEncode2.AddElem("ItemID", item.ID, sb); LLSDxmlEncode2.AddElem("CallbackID", (uint)0, sb); LLSDxmlEncode2.AddElem("FolderID", item.Folder, sb); LLSDxmlEncode2.AddElem("CreatorID", item.CreatorIdAsUuid, sb); LLSDxmlEncode2.AddElem("OwnerID", item.Owner, sb); LLSDxmlEncode2.AddElem("GroupID", item.GroupID, sb); LLSDxmlEncode2.AddElem("BaseMask", item.BasePermissions, sb); LLSDxmlEncode2.AddElem("OwnerMask", item.CurrentPermissions, sb); LLSDxmlEncode2.AddElem("GroupMask", item.GroupPermissions, sb); LLSDxmlEncode2.AddElem("EveryoneMask", item.EveryOnePermissions, sb); LLSDxmlEncode2.AddElem("NextOwnerMask", item.NextPermissions, sb); LLSDxmlEncode2.AddElem("GroupOwned", item.GroupOwned, sb); LLSDxmlEncode2.AddElem("AssetID", item.AssetID, sb); LLSDxmlEncode2.AddElem("Type", item.AssetType, sb); LLSDxmlEncode2.AddElem("InvType", item.InvType, sb); LLSDxmlEncode2.AddElem("Flags", item.Flags, sb); LLSDxmlEncode2.AddElem("SaleType", item.SaleType, sb); LLSDxmlEncode2.AddElem("SalePrice", item.SalePrice, sb); LLSDxmlEncode2.AddElem("Name", item.Name, sb); LLSDxmlEncode2.AddElem("Description", item.Description, sb); LLSDxmlEncode2.AddElem("CreationDate", item.CreationDate, sb); LLSDxmlEncode2.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); LLSDxmlEncode2.AddEndMap(sb); } LLSDxmlEncode2.AddEndArray(sb); } eq.Enqueue(eq.EndEventToBytes(sb), m_agentId); } private HashSet m_outPacketsToDrop; public bool AddOutPacketToDropSet(string packetName) { m_outPacketsToDrop ??= new HashSet(); return m_outPacketsToDrop.Add(packetName); } public bool RemoveOutPacketFromDropSet(string packetName) { if (m_outPacketsToDrop is 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) { m_inPacketsToDrop ??= new HashSet(); return m_inPacketsToDrop.Add(packetName); } public bool RemoveInPacketFromDropSet(string packetName) { if (m_inPacketsToDrop is null) return false; return m_inPacketsToDrop.Remove(packetName); } public HashSet GetInPacketDropSet() { return new HashSet(m_inPacketsToDrop); } public uint GetViewerCaps() { m_SupportObjectAnimations = false; uint ret; if(m_supportViewerCache) ret = m_viewerHandShakeFlags; else ret = (m_viewerHandShakeFlags & 4) | 2; // disable probes if (m_scene.CapsModule != null) { Caps cap = m_scene.CapsModule.GetCapsForUser(CircuitCode); if(cap != null) { if((cap.Flags & Caps.CapsFlags.SentSeeds) != 0) ret |= (uint)ViewerFlags.SentSeeds; if ((cap.Flags & Caps.CapsFlags.ObjectAnim) != 0) { m_SupportObjectAnimations = true; ret |= (uint)ViewerFlags.ObjectAnim; } if ((cap.Flags & Caps.CapsFlags.WLEnv) != 0) ret |= (uint)ViewerFlags.WLEnv; if ((cap.Flags & Caps.CapsFlags.AdvEnv) != 0) ret |= (uint)ViewerFlags.AdvEnv; if ((cap.Flags & Caps.CapsFlags.PBR) != 0) { ret |= (uint)ViewerFlags.PBR; m_SupportPBR = true; } if ((cap.Flags & Caps.CapsFlags.TPBR) != 0) { ret |= (uint)ViewerFlags.TPBR; SupportTerrainPBR = true; } } } ret |= (uint)(ViewerFlags & (ViewerFlags.SentTPBR)); ViewerFlags = (ViewerFlags)ret; return ret; } } }