200 lines
6.5 KiB
C#
200 lines
6.5 KiB
C#
/*
|
|
* 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 OpenSim 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.Runtime.Remoting;
|
|
using System.Runtime.Remoting.Channels;
|
|
using System.Runtime.Remoting.Channels.Tcp;
|
|
using Nini.Config;
|
|
using OpenSim.Framework;
|
|
using OpenSim.Region.Framework.Interfaces;
|
|
using OpenSim.Region.Framework.Scenes;
|
|
|
|
namespace OpenSim.Region.CoreModules.Grid.Interregion
|
|
{
|
|
public class InterregionModule : IInterregionModule, IRegionModule
|
|
{
|
|
#region Direction enum
|
|
|
|
public enum Direction
|
|
{
|
|
North,
|
|
NorthEast,
|
|
East,
|
|
SouthEast,
|
|
South,
|
|
SouthWest,
|
|
West,
|
|
NorthWest
|
|
}
|
|
|
|
#endregion
|
|
|
|
private readonly Dictionary<Type, Object> m_interfaces = new Dictionary<Type, object>();
|
|
private readonly Object m_lockObject = new object();
|
|
private readonly List<Location> m_myLocations = new List<Location>();
|
|
|
|
private readonly Dictionary<Location, string[]> m_neighbourInterfaces = new Dictionary<Location, string[]>();
|
|
private readonly Dictionary<Location, RemotingObject> m_neighbourRemote = new Dictionary<Location, RemotingObject>();
|
|
// private IConfigSource m_config;
|
|
private const bool m_enabled = false;
|
|
|
|
private RemotingObject m_myRemote;
|
|
private TcpChannel m_tcpChannel;
|
|
private int m_tcpPort = 10101;
|
|
|
|
#region IInterregionModule Members
|
|
|
|
public void internal_CreateRemotingObjects()
|
|
{
|
|
lock (m_lockObject)
|
|
{
|
|
if (m_tcpChannel == null)
|
|
{
|
|
m_myRemote = new RemotingObject(m_interfaces, m_myLocations.ToArray());
|
|
m_tcpChannel = new TcpChannel(m_tcpPort);
|
|
|
|
ChannelServices.RegisterChannel(m_tcpChannel, false);
|
|
RemotingServices.Marshal(m_myRemote, "OpenSimRemote2", typeof (RemotingObject));
|
|
}
|
|
}
|
|
}
|
|
|
|
public void RegisterMethod<T>(T e)
|
|
{
|
|
m_interfaces[typeof (T)] = e;
|
|
}
|
|
|
|
public bool HasInterface<T>(Location loc)
|
|
{
|
|
foreach (string val in m_neighbourInterfaces[loc])
|
|
{
|
|
if (val == typeof (T).FullName)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public T RequestInterface<T>(Location loc)
|
|
{
|
|
if (m_neighbourRemote.ContainsKey(loc))
|
|
{
|
|
return m_neighbourRemote[loc].RequestInterface<T>();
|
|
}
|
|
throw new IndexOutOfRangeException("No neighbour availible at that location");
|
|
}
|
|
|
|
public T[] RequestInterface<T>()
|
|
{
|
|
List<T> m_t = new List<T>();
|
|
foreach (RemotingObject remote in m_neighbourRemote.Values)
|
|
{
|
|
try
|
|
{
|
|
m_t.Add(remote.RequestInterface<T>());
|
|
}
|
|
catch (NotSupportedException)
|
|
{
|
|
}
|
|
}
|
|
return m_t.ToArray();
|
|
}
|
|
|
|
public Location GetLocationByDirection(Scene scene, Direction dir)
|
|
{
|
|
return new Location(0, 0);
|
|
}
|
|
|
|
public void RegisterRemoteRegion(string uri)
|
|
{
|
|
RegisterRemotingInterface((RemotingObject) Activator.GetObject(typeof (RemotingObject), uri));
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IRegionModule Members
|
|
|
|
public void Initialise(Scene scene, IConfigSource source)
|
|
{
|
|
m_myLocations.Add(new Location((int) scene.RegionInfo.RegionLocX,
|
|
(int) scene.RegionInfo.RegionLocY));
|
|
// m_config = source;
|
|
|
|
scene.RegisterModuleInterface<IInterregionModule>(this);
|
|
}
|
|
|
|
public void PostInitialise()
|
|
{
|
|
// Commenting out to remove 'unreachable code' warning since m_enabled is never true
|
|
// if (m_enabled)
|
|
// {
|
|
// try
|
|
// {
|
|
// m_tcpPort = m_config.Configs["Comms"].GetInt("remoting_port", m_tcpPort);
|
|
// }
|
|
// catch
|
|
// {
|
|
// }
|
|
//
|
|
// internal_CreateRemotingObjects();
|
|
// }
|
|
}
|
|
|
|
public void Close()
|
|
{
|
|
if (null != m_tcpChannel)
|
|
ChannelServices.UnregisterChannel(m_tcpChannel);
|
|
}
|
|
|
|
public string Name
|
|
{
|
|
get { return "InterregionModule"; }
|
|
}
|
|
|
|
public bool IsSharedModule
|
|
{
|
|
get { return true; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
private void RegisterRemotingInterface(RemotingObject remote)
|
|
{
|
|
Location[] locs = remote.GetLocations();
|
|
string[] interfaces = remote.GetInterfaces();
|
|
foreach (Location loc in locs)
|
|
{
|
|
m_neighbourInterfaces[loc] = interfaces;
|
|
m_neighbourRemote[loc] = remote;
|
|
}
|
|
}
|
|
}
|
|
}
|