123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635 |
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Net;
- using System.Net.Sockets;
- using System.Reflection;
- using log4net;
- using Nini.Config;
- using OpenMetaverse.Packets;
- using OpenSim.Framework;
- using OpenSim.Region.Framework.Scenes;
- namespace OpenSim.Region.ClientStack.LindenUDP
- {
-
-
-
- public class LLUDPServer : ILLClientStackNetworkHandler, IClientNetworkServer
- {
- private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
-
-
-
-
- protected Dictionary<EndPoint, uint> clientCircuits = new Dictionary<EndPoint, uint>();
- public Hashtable clientCircuits_reverse = Hashtable.Synchronized(new Hashtable());
- protected Dictionary<uint, EndPoint> proxyCircuits = new Dictionary<uint, EndPoint>();
-
- private Socket m_socket;
- protected IPEndPoint ServerIncoming;
- protected byte[] RecvBuffer = new byte[4096];
- protected byte[] ZeroBuffer = new byte[8192];
-
-
-
-
- protected EndPoint reusedEpSender = new IPEndPoint(IPAddress.Any, 0);
-
- protected int proxyPortOffset;
-
- protected AsyncCallback ReceivedData;
- protected LLPacketServer m_packetServer;
- protected Location m_location;
- protected uint listenPort;
- protected bool Allow_Alternate_Port;
- protected IPAddress listenIP = IPAddress.Parse("0.0.0.0");
- protected IScene m_localScene;
- protected IAssetCache m_assetCache;
- protected int m_clientSocketReceiveBuffer = 0;
-
-
-
- protected AgentCircuitManager m_circuitManager;
- public IScene LocalScene
- {
- set
- {
- m_localScene = value;
- m_packetServer.LocalScene = m_localScene;
- m_location = new Location(m_localScene.RegionInfo.RegionHandle);
- }
- }
- public ulong RegionHandle
- {
- get { return m_location.RegionHandle; }
- }
- Socket IClientNetworkServer.Server
- {
- get { return m_socket; }
- }
- public bool HandlesRegion(Location x)
- {
-
- return x == m_location;
- }
- public void AddScene(IScene x)
- {
- LocalScene = x;
- }
- public void Start()
- {
- ServerListener();
- }
- public void Stop()
- {
- m_socket.Close();
- }
- public LLUDPServer()
- {
- }
- public LLUDPServer(
- IPAddress _listenIP, ref uint port, int proxyPortOffset, bool allow_alternate_port, IConfigSource configSource,
- IAssetCache assetCache, AgentCircuitManager authenticateClass)
- {
- Initialise(_listenIP, ref port, proxyPortOffset, allow_alternate_port, configSource, authenticateClass);
- }
-
-
-
-
-
-
-
-
-
-
- public void Initialise(
- IPAddress _listenIP, ref uint port, int proxyPortOffsetParm, bool allow_alternate_port, IConfigSource configSource,
- AgentCircuitManager circuitManager)
- {
- ClientStackUserSettings userSettings = new ClientStackUserSettings();
-
- IConfig config = configSource.Configs["ClientStack.LindenUDP"];
-
- if (config != null)
- {
- if (config.Contains("client_throttle_multiplier"))
- userSettings.ClientThrottleMultipler = config.GetFloat("client_throttle_multiplier");
- if (config.Contains("client_socket_rcvbuf_size"))
- m_clientSocketReceiveBuffer = config.GetInt("client_socket_rcvbuf_size");
- }
-
- m_log.DebugFormat("[CLIENT]: client_throttle_multiplier = {0}", userSettings.ClientThrottleMultipler);
- m_log.DebugFormat("[CLIENT]: client_socket_rcvbuf_size = {0}", (m_clientSocketReceiveBuffer != 0 ?
- m_clientSocketReceiveBuffer.ToString() : "OS default"));
-
- proxyPortOffset = proxyPortOffsetParm;
- listenPort = (uint) (port + proxyPortOffsetParm);
- listenIP = _listenIP;
- Allow_Alternate_Port = allow_alternate_port;
- m_circuitManager = circuitManager;
- CreatePacketServer(userSettings);
-
-
-
- port = (uint)(listenPort - proxyPortOffsetParm);
- }
- protected virtual void CreatePacketServer(ClientStackUserSettings userSettings)
- {
- new LLPacketServer(this, userSettings);
- }
-
-
-
-
- protected virtual void OnReceivedData(IAsyncResult result)
- {
- Packet packet = null;
- int numBytes = 1;
- EndPoint epSender = new IPEndPoint(IPAddress.Any, 0);
- EndPoint epProxy = null;
- try
- {
- if (EndReceive(out numBytes, result, ref epSender))
- {
-
-
- Array.Clear(RecvBuffer, numBytes, RecvBuffer.Length - numBytes);
-
- int packetEnd = numBytes - 1;
- if (proxyPortOffset != 0) packetEnd -= 6;
-
- try
- {
- packet = PacketPool.Instance.GetPacket(RecvBuffer, ref packetEnd, ZeroBuffer);
- }
- catch (MalformedDataException e)
- {
- m_log.DebugFormat("[CLIENT]: Dropped Malformed Packet due to MalformedDataException: {0}", e.StackTrace);
- }
- catch (IndexOutOfRangeException e)
- {
- m_log.DebugFormat("[CLIENT]: Dropped Malformed Packet due to IndexOutOfRangeException: {0}", e.StackTrace);
- }
- catch (Exception e)
- {
- m_log.Debug("[CLIENT]: " + e);
- }
- }
-
-
- if (proxyPortOffset != 0)
- {
-
-
- if (packet != null && packet.Type == PacketType.UseCircuitCode)
- {
- epProxy = epSender;
- }
-
-
- epSender = ProxyCodec.DecodeProxyMessage(RecvBuffer, ref numBytes);
- }
- }
- catch (Exception ex)
- {
- m_log.ErrorFormat("[CLIENT]: Exception thrown during EndReceive(): {0}", ex);
- }
- BeginRobustReceive();
- if (packet != null)
- {
- if (packet.Type == PacketType.UseCircuitCode)
- AddNewClient((UseCircuitCodePacket)packet, epSender, epProxy);
- else
- ProcessInPacket(packet, epSender);
- }
- }
-
-
-
-
-
-
- protected virtual void ProcessInPacket(Packet packet, EndPoint epSender)
- {
- try
- {
-
- uint circuit;
- bool ret;
-
- lock (clientCircuits)
- {
- ret = clientCircuits.TryGetValue(epSender, out circuit);
- }
- if (ret)
- {
-
-
-
- m_packetServer.InPacket(circuit, packet);
- }
- }
- catch (Exception e)
- {
- m_log.Error("[CLIENT]: Exception in processing packet - ignoring: ", e);
- }
- }
-
-
-
-
- protected virtual void BeginReceive()
- {
- m_socket.BeginReceiveFrom(
- RecvBuffer, 0, RecvBuffer.Length, SocketFlags.None, ref reusedEpSender, ReceivedData, null);
- }
-
-
-
-
- private void BeginRobustReceive()
- {
- bool done = false;
- while (!done)
- {
- try
- {
- BeginReceive();
- done = true;
- }
- catch (SocketException e)
- {
-
-
-
-
- try
- {
- CloseCircuit(e);
- }
- catch (Exception e2)
- {
- m_log.ErrorFormat(
- "[CLIENT]: Exception thrown when trying to close the circuit for {0} - {1}", reusedEpSender,
- e2);
- }
- }
- catch (ObjectDisposedException)
- {
- m_log.Info(
- "[UDPSERVER]: UDP Object disposed. No need to worry about this if you're restarting the simulator.");
- done = true;
- }
- catch (Exception ex)
- {
- m_log.ErrorFormat("[CLIENT]: Exception thrown during BeginReceive(): {0}", ex);
- }
- }
- }
-
-
-
-
-
- private void CloseCircuit(Exception e)
- {
- uint circuit;
- lock (clientCircuits)
- {
- if (clientCircuits.TryGetValue(reusedEpSender, out circuit))
- {
- m_packetServer.CloseCircuit(circuit);
-
- if (e != null)
- m_log.ErrorFormat(
- "[CLIENT]: Closed circuit {0} {1} due to exception {2}", circuit, reusedEpSender, e);
- }
- }
- }
-
-
-
-
-
-
-
-
- protected virtual bool EndReceive(out int numBytes, IAsyncResult result, ref EndPoint epSender)
- {
- bool hasReceivedOkay = false;
- numBytes = 0;
-
- try
- {
- numBytes = m_socket.EndReceiveFrom(result, ref epSender);
- hasReceivedOkay = true;
- }
- catch (SocketException e)
- {
-
-
-
- switch (e.SocketErrorCode)
- {
- case SocketError.AlreadyInProgress:
- return hasReceivedOkay;
- case SocketError.NetworkReset:
- case SocketError.ConnectionReset:
- case SocketError.OperationAborted:
- break;
- default:
- throw;
- }
- }
- catch (ObjectDisposedException e)
- {
- m_log.DebugFormat("[CLIENT]: ObjectDisposedException: Object {0} disposed.", e.ObjectName);
-
- }
-
- return hasReceivedOkay;
- }
-
-
-
-
-
-
- protected virtual void AddNewClient(UseCircuitCodePacket useCircuit, EndPoint epSender, EndPoint epProxy)
- {
-
- if (m_localScene.RegionStatus != RegionStatus.SlaveScene)
- {
- AuthenticateResponse sessionInfo;
- bool isNewCircuit = false;
-
- if (!m_packetServer.IsClientAuthorized(useCircuit, m_circuitManager, out sessionInfo))
- {
- m_log.WarnFormat(
- "[CONNECTION FAILURE]: Connection request for client {0} connecting with unnotified circuit code {1} from {2}",
- useCircuit.CircuitCode.ID, useCircuit.CircuitCode.Code, epSender);
-
- return;
- }
-
- lock (clientCircuits)
- {
- if (!clientCircuits.ContainsKey(epSender))
- {
- clientCircuits.Add(epSender, useCircuit.CircuitCode.Code);
- isNewCircuit = true;
- }
- }
- if (isNewCircuit)
- {
-
- clientCircuits_reverse[useCircuit.CircuitCode.Code] = epSender;
- lock (proxyCircuits)
- {
- proxyCircuits[useCircuit.CircuitCode.Code] = epProxy;
- }
-
- m_packetServer.AddNewClient(epSender, useCircuit, m_assetCache, sessionInfo, epProxy);
-
-
-
-
- }
- }
-
-
- PacketAckPacket ack_it = (PacketAckPacket)PacketPool.Instance.GetPacket(PacketType.PacketAck);
-
- ack_it.Packets = new PacketAckPacket.PacketsBlock[1];
- ack_it.Packets[0] = new PacketAckPacket.PacketsBlock();
- ack_it.Packets[0].ID = useCircuit.Header.Sequence;
-
- ack_it.Header.Sequence = ((useCircuit.Header.Sequence < uint.MaxValue) ? useCircuit.Header.Sequence : 0) + 1;
- ack_it.Header.Reliable = false;
- byte[] ackmsg = ack_it.ToBytes();
-
-
- byte[] msg = new byte[4096];
- Buffer.BlockCopy(ackmsg, 0, msg, 0, ackmsg.Length);
- SendPacketTo(msg, ackmsg.Length, SocketFlags.None, useCircuit.CircuitCode.Code);
- PacketPool.Instance.ReturnPacket(useCircuit);
- }
- public void ServerListener()
- {
- uint newPort = listenPort;
- m_log.Info("[UDPSERVER]: Opening UDP socket on " + listenIP + " " + newPort + ".");
- ServerIncoming = new IPEndPoint(listenIP, (int)newPort);
- m_socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
- if (0 != m_clientSocketReceiveBuffer)
- m_socket.ReceiveBufferSize = m_clientSocketReceiveBuffer;
- m_socket.Bind(ServerIncoming);
-
-
-
-
-
-
- listenPort = newPort;
- m_log.Info("[UDPSERVER]: UDP socket bound, getting ready to listen");
- ReceivedData = OnReceivedData;
- BeginReceive();
- m_log.Info("[UDPSERVER]: Listening on port " + newPort);
- }
- public virtual void RegisterPacketServer(LLPacketServer server)
- {
- m_packetServer = server;
- }
- public virtual void SendPacketTo(byte[] buffer, int size, SocketFlags flags, uint circuitcode)
-
- {
-
- EndPoint sendto;
- try
- {
- sendto = (EndPoint)clientCircuits_reverse[circuitcode];
- }
- catch
- {
-
- m_log.Warn("[CLIENT]: Circuit not found, not sending packet");
- return;
- }
- if (sendto != null)
- {
-
- if (proxyPortOffset != 0)
- {
-
- ProxyCodec.EncodeProxyMessage(buffer, ref size, sendto);
- m_socket.SendTo(buffer, size, flags, proxyCircuits[circuitcode]);
- }
- else
- {
-
- try
- {
- m_socket.SendTo(buffer, size, flags, sendto);
- }
- catch (SocketException SockE)
- {
- m_log.ErrorFormat("[UDPSERVER]: Caught Socket Error in the send buffer!. {0}",SockE.ToString());
- }
- }
- }
- }
- public virtual void RemoveClientCircuit(uint circuitcode)
- {
- EndPoint sendto;
- if (clientCircuits_reverse.Contains(circuitcode))
- {
- sendto = (EndPoint)clientCircuits_reverse[circuitcode];
- clientCircuits_reverse.Remove(circuitcode);
- lock (clientCircuits)
- {
- if (sendto != null)
- {
- clientCircuits.Remove(sendto);
- }
- else
- {
- m_log.DebugFormat(
- "[CLIENT]: endpoint for circuit code {0} in RemoveClientCircuit() was unexpectedly null!", circuitcode);
- }
- }
- lock (proxyCircuits)
- {
- proxyCircuits.Remove(circuitcode);
- }
- }
- }
- public void RestoreClient(AgentCircuitData circuit, EndPoint userEP, EndPoint proxyEP)
- {
-
- UseCircuitCodePacket useCircuit = new UseCircuitCodePacket();
- useCircuit.CircuitCode.Code = circuit.circuitcode;
- useCircuit.CircuitCode.ID = circuit.AgentID;
- useCircuit.CircuitCode.SessionID = circuit.SessionID;
-
- AuthenticateResponse sessionInfo;
-
- if (!m_packetServer.IsClientAuthorized(useCircuit, m_circuitManager, out sessionInfo))
- {
- m_log.WarnFormat(
- "[CLIENT]: Restore request denied to avatar {0} connecting with unauthorized circuit code {1}",
- useCircuit.CircuitCode.ID, useCircuit.CircuitCode.Code);
-
- return;
- }
- lock (clientCircuits)
- {
- if (!clientCircuits.ContainsKey(userEP))
- clientCircuits.Add(userEP, useCircuit.CircuitCode.Code);
- else
- m_log.Error("[CLIENT]: clientCircuits already contains entry for user " + useCircuit.CircuitCode.Code + ". NOT adding.");
- }
-
- if (!clientCircuits_reverse.ContainsKey(useCircuit.CircuitCode.Code))
- clientCircuits_reverse.Add(useCircuit.CircuitCode.Code, userEP);
- else
- m_log.Error("[CLIENT]: clientCurcuits_reverse already contains entry for user " + useCircuit.CircuitCode.Code + ". NOT adding.");
- lock (proxyCircuits)
- {
- if (!proxyCircuits.ContainsKey(useCircuit.CircuitCode.Code))
- {
- proxyCircuits.Add(useCircuit.CircuitCode.Code, proxyEP);
- }
- else
- {
-
- proxyCircuits.Remove(useCircuit.CircuitCode.Code);
- proxyCircuits.Add(useCircuit.CircuitCode.Code, proxyEP);
- }
- }
- m_packetServer.AddNewClient(userEP, useCircuit, m_assetCache, sessionInfo, proxyEP);
- }
- }
- }
|