Merge branch 'master' of opensimulator.org:/var/git/opensim

LSLKeyTest
Melanie Thielker 2016-07-11 22:32:27 +01:00
commit f87d5d0780
4 changed files with 302 additions and 140 deletions

View File

@ -1303,7 +1303,7 @@ namespace OpenSim.Framework
} }
catch (Exception e) catch (Exception e)
{ {
m_log.WarnFormat("[UTILS]: Exception copying configuration file {0} to {1}: {2}", configFile, exampleConfigFile, e.Message); m_log.WarnFormat("[UTILS]: Exception copying configuration file {0} to {1}: {2}", exampleConfigFile, configFile, e.Message);
return false; return false;
} }
} }

View File

@ -4796,7 +4796,7 @@ namespace OpenSim.Region.ClientStack.LindenUDP
block.SaleType = sop.ObjectSaleType; block.SaleType = sop.ObjectSaleType;
block.SalePrice = sop.SalePrice; block.SalePrice = sop.SalePrice;
block.Category = sop.Category; block.Category = sop.Category;
block.LastOwnerID = sop.CreatorID; // copied from old SOG call... is this right? block.LastOwnerID = sop.LastOwnerID;
block.Name = Util.StringToBytes256(sop.Name); block.Name = Util.StringToBytes256(sop.Name);
block.Description = Util.StringToBytes256(sop.Description); block.Description = Util.StringToBytes256(sop.Description);

View File

@ -28,10 +28,12 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Reflection; using System.Reflection;
using System.Threading;
using Nini.Config; using Nini.Config;
using log4net; using log4net;
using OpenMetaverse; using OpenMetaverse;
using OpenSim.Framework; using OpenSim.Framework;
using OpenSim.Framework.Monitoring;
using OpenSim.Region.Framework.Interfaces; using OpenSim.Region.Framework.Interfaces;
using OpenSim.Region.Framework.Scenes; using OpenSim.Region.Framework.Scenes;
@ -45,9 +47,13 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
private Scene m_scene; private Scene m_scene;
private Dictionary<string, FileData> NewFiles = new Dictionary<string, FileData>(); private Dictionary<string, FileData> NewFiles = new Dictionary<string, FileData>();
private Dictionary<ulong, XferDownLoad> Transfers = new Dictionary<ulong, XferDownLoad>(); private Dictionary<ulong, XferDownLoad> Transfers = new Dictionary<ulong, XferDownLoad>();
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
private object timeTickLock = new object();
private double lastTimeTick = 0.0;
private double lastFilesExpire = 0.0;
private bool inTimeTick = false;
public struct XferRequest public struct XferRequest
{ {
public IClientAPI remoteClient; public IClientAPI remoteClient;
@ -59,26 +65,30 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
private class FileData private class FileData
{ {
public byte[] Data; public byte[] Data;
public int Count; public int refsCount;
public double timeStampMS;
} }
#region INonSharedRegionModule Members #region INonSharedRegionModule Members
public void Initialise(IConfigSource config) public void Initialise(IConfigSource config)
{ {
lastTimeTick = Util.GetTimeStampMS() + 30000.0;
lastFilesExpire = lastTimeTick + 180000.0;
} }
public void AddRegion(Scene scene) public void AddRegion(Scene scene)
{ {
m_scene = scene; m_scene = scene;
m_scene.EventManager.OnNewClient += NewClient;
m_scene.RegisterModuleInterface<IXfer>(this); m_scene.RegisterModuleInterface<IXfer>(this);
m_scene.EventManager.OnNewClient += NewClient;
m_scene.EventManager.OnRegionHeartbeatEnd += OnTimeTick;
} }
public void RemoveRegion(Scene scene) public void RemoveRegion(Scene scene)
{ {
m_scene.EventManager.OnNewClient -= NewClient; m_scene.EventManager.OnNewClient -= NewClient;
m_scene.EventManager.OnRegionHeartbeatEnd -= OnTimeTick;
m_scene.UnregisterModuleInterface<IXfer>(this); m_scene.UnregisterModuleInterface<IXfer>(this);
m_scene = null; m_scene = null;
@ -104,6 +114,35 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
#endregion #endregion
public void OnTimeTick(Scene scene)
{
// we are on a heartbeat thread we there can be several
if(Monitor.TryEnter(timeTickLock))
{
if(!inTimeTick)
{
double now = Util.GetTimeStampMS();
if(now - lastTimeTick > 1750.0)
{
inTimeTick = true;
//don't overload busy heartbeat
WorkManager.RunInThread(
delegate
{
transfersTimeTick(now);
expireFiles(now);
lastTimeTick = now;
inTimeTick = false;
},
null,
"XferTimeTick");
}
}
Monitor.Exit(timeTickLock);
}
}
#region IXfer Members #region IXfer Members
/// <summary> /// <summary>
@ -118,24 +157,45 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
{ {
lock (NewFiles) lock (NewFiles)
{ {
double now = Util.GetTimeStampMS();
if (NewFiles.ContainsKey(fileName)) if (NewFiles.ContainsKey(fileName))
{ {
NewFiles[fileName].Count++; NewFiles[fileName].refsCount++;
NewFiles[fileName].Data = data; NewFiles[fileName].Data = data;
NewFiles[fileName].timeStampMS = now;
} }
else else
{ {
FileData fd = new FileData(); FileData fd = new FileData();
fd.Count = 1; fd.refsCount = 1;
fd.Data = data; fd.Data = data;
fd.timeStampMS = now;
NewFiles.Add(fileName, fd); NewFiles.Add(fileName, fd);
} }
} }
return true; return true;
} }
#endregion #endregion
public void expireFiles(double now)
{
lock (NewFiles)
{
// hopefully we will not have many files so nasty code will do it
if(now - lastFilesExpire > 120000.0)
{
lastFilesExpire = now;
List<string> expires = new List<string>();
foreach(string fname in NewFiles.Keys)
{
if(NewFiles[fname].refsCount == 0 && now - NewFiles[fname].timeStampMS > 120000.0)
expires.Add(fname);
}
foreach(string fname in expires)
NewFiles.Remove(fname);
}
}
}
public void NewClient(IClientAPI client) public void NewClient(IClientAPI client)
{ {
@ -144,6 +204,51 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
client.OnAbortXfer += AbortXfer; client.OnAbortXfer += AbortXfer;
} }
public void OnClientClosed(IClientAPI client)
{
client.OnRequestXfer -= RequestXfer;
client.OnConfirmXfer -= AckPacket;
client.OnAbortXfer -= AbortXfer;
}
private void RemoveOrDecrementFile(string fileName)
{
// NewFiles must be locked
if (NewFiles.ContainsKey(fileName))
{
if (NewFiles[fileName].refsCount == 1)
NewFiles.Remove(fileName);
else
NewFiles[fileName].refsCount--;
}
}
public void transfersTimeTick(double now)
{
XferDownLoad[] xfrs;
lock(Transfers)
{
if(Transfers.Count == 0)
return;
xfrs = new XferDownLoad[Transfers.Count];
Transfers.Values.CopyTo(xfrs,0);
}
foreach(XferDownLoad xfr in xfrs)
{
if(xfr.checkTime(now))
{
ulong xfrID = xfr.XferID;
lock(Transfers)
{
if(Transfers.ContainsKey(xfrID))
Transfers.Remove(xfrID);
}
}
}
}
/// <summary> /// <summary>
/// ///
/// </summary> /// </summary>
@ -156,80 +261,52 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
{ {
if (NewFiles.ContainsKey(fileName)) if (NewFiles.ContainsKey(fileName))
{ {
if (!Transfers.ContainsKey(xferID)) lock(Transfers)
{ {
byte[] fileData = NewFiles[fileName].Data; if (!Transfers.ContainsKey(xferID))
XferDownLoad transaction = new XferDownLoad(fileName, fileData, xferID, remoteClient); {
if (fileName.StartsWith("inventory_")) byte[] fileData = NewFiles[fileName].Data;
transaction.isTaskInventory = true; int burstSize = remoteClient.GetAgentThrottleSilent((int)ThrottleOutPacketType.Asset) >> 11;
if(Transfers.Count > 1)
burstSize /= Transfers.Count;
XferDownLoad transaction =
new XferDownLoad(fileName, fileData, xferID, remoteClient, burstSize);
Transfers.Add(xferID, transaction); Transfers.Add(xferID, transaction);
if (transaction.StartSend()) transaction.StartSend();
RemoveXferData(xferID);
// The transaction for this file is either complete or on its way
RemoveOrDecrement(fileName);
// The transaction for this file is on its way
RemoveOrDecrementFile(fileName);
}
} }
} }
else else
m_log.WarnFormat("[Xfer]: {0} not found", fileName); m_log.WarnFormat("[Xfer]: {0} not found", fileName);
} }
} }
public void AckPacket(IClientAPI remoteClient, ulong xferID, uint packet) public void AckPacket(IClientAPI remoteClient, ulong xferID, uint packet)
{ {
lock (NewFiles) // This is actually to lock Transfers lock (Transfers)
{ {
if (Transfers.ContainsKey(xferID)) if (Transfers.ContainsKey(xferID))
{ {
XferDownLoad dl = Transfers[xferID];
if (Transfers[xferID].AckPacket(packet)) if (Transfers[xferID].AckPacket(packet))
{ Transfers.Remove(xferID);
RemoveXferData(xferID);
RemoveOrDecrement(dl.FileName);
}
} }
} }
} }
private void RemoveXferData(ulong xferID)
{
// NewFiles must be locked!
if (Transfers.ContainsKey(xferID))
{
XferModule.XferDownLoad xferItem = Transfers[xferID];
//string filename = xferItem.FileName;
Transfers.Remove(xferID);
xferItem.Data = new byte[0]; // Clear the data
xferItem.DataPointer = 0;
}
}
public void AbortXfer(IClientAPI remoteClient, ulong xferID) public void AbortXfer(IClientAPI remoteClient, ulong xferID)
{ {
lock (NewFiles) lock (Transfers)
{ {
if (Transfers.ContainsKey(xferID)) if (Transfers.ContainsKey(xferID))
RemoveOrDecrement(Transfers[xferID].FileName); {
Transfers[xferID].done();
RemoveXferData(xferID); Transfers.Remove(xferID);
} }
}
private void RemoveOrDecrement(string fileName)
{
// NewFiles must be locked
if (NewFiles.ContainsKey(fileName))
{
if (NewFiles[fileName].Count == 1)
NewFiles.Remove(fileName);
else
NewFiles[fileName].Count--;
} }
} }
@ -238,53 +315,124 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
public class XferDownLoad public class XferDownLoad
{ {
public IClientAPI Client; public IClientAPI Client;
private bool complete;
public byte[] Data = new byte[0]; public byte[] Data = new byte[0];
public int DataPointer = 0;
public string FileName = String.Empty; public string FileName = String.Empty;
public uint Packet = 0;
public uint Serial = 1;
public ulong XferID = 0; public ulong XferID = 0;
public bool isTaskInventory = false; public bool isDeleted = false;
public XferDownLoad(string fileName, byte[] data, ulong xferID, IClientAPI client) private object myLock = new object();
private double lastsendTimeMS;
private int LastPacket;
private int lastBytes;
private int lastSentPacket;
private int lastAckPacket;
private int burstSize;
private int retries = 0;
public XferDownLoad(string fileName, byte[] data, ulong xferID, IClientAPI client, int burstsz)
{ {
FileName = fileName; FileName = fileName;
Data = data; Data = data;
XferID = xferID; XferID = xferID;
Client = client; Client = client;
burstSize = burstsz;
} }
public XferDownLoad() public XferDownLoad()
{ {
} }
public void done()
{
if(!isDeleted)
{
Data = new byte[0];
isDeleted = true;
}
}
/// <summary> /// <summary>
/// Start a transfer /// Start a transfer
/// </summary> /// </summary>
/// <returns>True if the transfer is complete, false if not</returns> /// <returns>True if the transfer is complete, false if not</returns>
public bool StartSend() public void StartSend()
{ {
if (Data.Length < 1000) lock(myLock)
{ {
// for now (testing) we only support files under 1000 bytes if(Data.Length == 0) //??
byte[] transferData = new byte[Data.Length + 4]; {
Array.Copy(Utils.IntToBytes(Data.Length), 0, transferData, 0, 4); LastPacket = 0;
Array.Copy(Data, 0, transferData, 4, Data.Length); lastBytes = 0;
Client.SendXferPacket(XferID, 0 + 0x80000000, transferData, isTaskInventory); burstSize = 0;
complete = true; }
else
{
// payload of 1024bytes
LastPacket = Data.Length >> 10;
lastBytes = Data.Length & 0x3ff;
if(lastBytes == 0)
{
lastBytes = 1024;
LastPacket--;
}
}
lastAckPacket = -1;
lastSentPacket = -1;
double now = Util.GetTimeStampMS();
SendBurst(now);
return;
}
}
private void SendBurst(double now)
{
int start = lastAckPacket + 1;
int end = start + burstSize;
if (end > LastPacket)
end = LastPacket;
while(start <= end)
SendPacket(start++ , now);
}
private void SendPacket(int pkt, double now)
{
if(pkt > LastPacket)
return;
int pktsize;
uint pktid;
if (pkt == LastPacket)
{
pktsize = lastBytes;
pktid = (uint)pkt | 0x80000000u;
} }
else else
{ {
byte[] transferData = new byte[1000 + 4]; pktsize = 1024;
Array.Copy(Utils.IntToBytes(Data.Length), 0, transferData, 0, 4); pktid = (uint)pkt;
Array.Copy(Data, 0, transferData, 4, 1000);
Client.SendXferPacket(XferID, 0, transferData, isTaskInventory);
Packet++;
DataPointer = 1000;
} }
return complete; byte[] transferData;
if(pkt == 0)
{
transferData = new byte[pktsize + 4];
Array.Copy(Utils.IntToBytes(Data.Length), 0, transferData, 0, 4);
Array.Copy(Data, 0, transferData, 4, pktsize);
}
else
{
transferData = new byte[pktsize];
Array.Copy(Data, pkt << 10, transferData, 0, pktsize);
}
Client.SendXferPacket(XferID, pktid, transferData, false);
lastSentPacket = pkt;
lastsendTimeMS = now;
} }
/// <summary> /// <summary>
@ -294,30 +442,46 @@ namespace OpenSim.Region.CoreModules.Agent.Xfer
/// <returns>True if the transfer is complete, false otherwise</returns> /// <returns>True if the transfer is complete, false otherwise</returns>
public bool AckPacket(uint packet) public bool AckPacket(uint packet)
{ {
if (!complete) lock(myLock)
{ {
if ((Data.Length - DataPointer) > 1000) if(isDeleted)
{ return true;
byte[] transferData = new byte[1000];
Array.Copy(Data, DataPointer, transferData, 0, 1000);
Client.SendXferPacket(XferID, Packet, transferData, isTaskInventory);
Packet++;
DataPointer += 1000;
}
else
{
byte[] transferData = new byte[Data.Length - DataPointer];
Array.Copy(Data, DataPointer, transferData, 0, Data.Length - DataPointer);
uint endPacket = Packet |= (uint) 0x80000000;
Client.SendXferPacket(XferID, endPacket, transferData, isTaskInventory);
Packet++;
DataPointer += (Data.Length - DataPointer);
complete = true; packet &= 0x7fffffff;
if(lastAckPacket < packet)
lastAckPacket = (int)packet;
if(lastAckPacket == LastPacket)
{
done();
return true;
} }
double now = Util.GetTimeStampMS();
SendPacket(lastSentPacket + 1, now);
return false;
} }
}
return complete; public bool checkTime(double now)
{
if(Monitor.TryEnter(myLock))
{
if(!isDeleted)
{
double timeMS = now - lastsendTimeMS;
if(timeMS > 60000.0)
done();
else if(timeMS > 3500.0 && retries++ < 3)
{
burstSize >>= 1;
SendBurst(now);
}
}
Monitor.Exit(myLock);
return isDeleted;
}
return false;
} }
} }

View File

@ -48,6 +48,8 @@ namespace OpenSim.Region.Framework.Scenes
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
private byte[] m_inventoryFileData = new byte[0]; private byte[] m_inventoryFileData = new byte[0];
private byte[] m_inventoryFileNameBytes = new byte[0];
private string m_inventoryFileName = "";
private uint m_inventoryFileNameSerial = 0; private uint m_inventoryFileNameSerial = 0;
private bool m_inventoryPrivileged = false; private bool m_inventoryPrivileged = false;
private object m_inventoryFileLock = new object(); private object m_inventoryFileLock = new object();
@ -1112,18 +1114,34 @@ namespace OpenSim.Region.Framework.Scenes
/// <param name="xferManager"></param> /// <param name="xferManager"></param>
public void RequestInventoryFile(IClientAPI client, IXfer xferManager) public void RequestInventoryFile(IClientAPI client, IXfer xferManager)
{ {
lock (m_inventoryFileLock) lock (m_inventoryFileLock)
{ {
string filename = "inventory_" + UUID.Random().ToString() + ".tmp";
bool changed = false; bool changed = false;
if (m_inventoryFileNameSerial < m_inventorySerial)
Items.LockItemsForRead(true);
if (m_inventorySerial == 0) // No inventory
{
Items.LockItemsForRead(false);
client.SendTaskInventory(m_part.UUID, 0, new byte[0]);
return;
}
if (m_items.Count == 0) // No inventory
{
Items.LockItemsForRead(false);
client.SendTaskInventory(m_part.UUID, 0, new byte[0]);
return;
}
if (m_inventoryFileNameSerial != m_inventorySerial)
{ {
m_inventoryFileNameSerial = m_inventorySerial; m_inventoryFileNameSerial = m_inventorySerial;
changed = true; changed = true;
} }
Items.LockItemsForRead(false);
if (m_inventoryFileData.Length < 2) if (m_inventoryFileData.Length < 2)
changed = true; changed = true;
@ -1134,32 +1152,11 @@ namespace OpenSim.Region.Framework.Scenes
if (m_inventoryPrivileged != includeAssets) if (m_inventoryPrivileged != includeAssets)
changed = true; changed = true;
Items.LockItemsForRead(true);
if (m_inventorySerial == 0) // No inventory
{
Items.LockItemsForRead(false);
client.SendTaskInventory(m_part.UUID, 0, new byte[0]);
return;
}
if (m_items.Count == 0) // No inventory
{
Items.LockItemsForRead(false);
client.SendTaskInventory(m_part.UUID, 0, new byte[0]);
return;
}
if (!changed) if (!changed)
{ {
Items.LockItemsForRead(false); xferManager.AddNewFile(m_inventoryFileName, m_inventoryFileData);
xferManager.AddNewFile(filename,
m_inventoryFileData);
client.SendTaskInventory(m_part.UUID, (short)m_inventoryFileNameSerial, client.SendTaskInventory(m_part.UUID, (short)m_inventoryFileNameSerial,
Util.StringToBytes256(filename)); m_inventoryFileNameBytes);
return; return;
} }
@ -1168,6 +1165,8 @@ namespace OpenSim.Region.Framework.Scenes
InventoryStringBuilder invString = new InventoryStringBuilder(m_part.UUID, UUID.Zero); InventoryStringBuilder invString = new InventoryStringBuilder(m_part.UUID, UUID.Zero);
Items.LockItemsForRead(true);
foreach (TaskInventoryItem item in m_items.Values) foreach (TaskInventoryItem item in m_items.Values)
{ {
UUID ownerID = item.OwnerID; UUID ownerID = item.OwnerID;
@ -1222,9 +1221,10 @@ namespace OpenSim.Region.Framework.Scenes
if (m_inventoryFileData.Length > 2) if (m_inventoryFileData.Length > 2)
{ {
xferManager.AddNewFile(filename, m_inventoryFileData); m_inventoryFileName = "inventory_" + UUID.Random().ToString() + ".tmp";
client.SendTaskInventory(m_part.UUID, (short)m_inventoryFileNameSerial, m_inventoryFileNameBytes = Util.StringToBytes256(m_inventoryFileName);
Util.StringToBytes256(filename)); xferManager.AddNewFile(m_inventoryFileName, m_inventoryFileData);
client.SendTaskInventory(m_part.UUID, (short)m_inventoryFileNameSerial,m_inventoryFileNameBytes);
return; return;
} }
@ -1267,26 +1267,22 @@ namespace OpenSim.Region.Framework.Scenes
AddNameValueLine("obj_id", folderID.ToString()); AddNameValueLine("obj_id", folderID.ToString());
AddNameValueLine("parent_id", parentID.ToString()); AddNameValueLine("parent_id", parentID.ToString());
AddNameValueLine("type", "category"); AddNameValueLine("type", "category");
AddNameValueLine("name", "Contents|"); AddNameValueLine("name", "Contents|\n\t}");
AddSectionEnd();
} }
public void AddItemStart() public void AddItemStart()
{ {
BuildString.Append("\tinv_item\t0\n"); BuildString.Append("\tinv_item\t0\n\t{\n");
AddSectionStart();
} }
public void AddPermissionsStart() public void AddPermissionsStart()
{ {
BuildString.Append("\tpermissions 0\n"); BuildString.Append("\tpermissions 0\n\t{\n");
AddSectionStart();
} }
public void AddSaleStart() public void AddSaleStart()
{ {
BuildString.Append("\tsale_info\t0\n"); BuildString.Append("\tsale_info\t0\n\t{\n");
AddSectionStart();
} }
protected void AddSectionStart() protected void AddSectionStart()
@ -1307,8 +1303,10 @@ namespace OpenSim.Region.Framework.Scenes
public void AddNameValueLine(string name, string value) public void AddNameValueLine(string name, string value)
{ {
BuildString.Append("\t\t"); BuildString.Append("\t\t");
BuildString.Append(name + "\t"); BuildString.Append(name);
BuildString.Append(value + "\n"); BuildString.Append("\t");
BuildString.Append(value);
BuildString.Append("\n");
} }
public String GetString() public String GetString()