From f26f5f6efffd1cd9cb8dd35372ad9a0b191e09e7 Mon Sep 17 00:00:00 2001
From: Justin Clarke Casey
Date: Mon, 17 Nov 2008 21:10:08 +0000
Subject: * Apply http://opensimulator.org/mantis/view.php?id=2611 *
Progressive texture delivery (ported from jhurliman's Simian code) * Thanks
jhurliman!
---
.../Modules/Agent/TextureSender/TextureSender.cs | 268 +++++++++++++--------
1 file changed, 169 insertions(+), 99 deletions(-)
diff --git a/OpenSim/Region/Environment/Modules/Agent/TextureSender/TextureSender.cs b/OpenSim/Region/Environment/Modules/Agent/TextureSender/TextureSender.cs
index cd61798..265247c 100644
--- a/OpenSim/Region/Environment/Modules/Agent/TextureSender/TextureSender.cs
+++ b/OpenSim/Region/Environment/Modules/Agent/TextureSender/TextureSender.cs
@@ -34,99 +34,177 @@ using OpenSim.Region.Environment.Interfaces;
namespace OpenSim.Region.Environment.Modules.Agent.TextureSender
{
- ///
- /// A TextureSender handles the process of receiving a texture requested by the client from the
- /// AssetCache, and then sending that texture back to the client.
- ///
- public class TextureSender : ITextureSender
+ public class ImageDownload
{
- private static readonly ILog m_log
- = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
+ public const int FIRST_IMAGE_PACKET_SIZE = 600;
+ public const int IMAGE_PACKET_SIZE = 1000;
- ///
- /// Records the number of times texture send has been called.
- ///
- public int counter = 0;
+ public OpenMetaverse.AssetTexture Texture;
+ public int DiscardLevel;
+ public float Priority;
+ public int CurrentPacket;
+ public int StopPacket;
- public bool ImageLoaded = false;
+ public ImageDownload(OpenMetaverse.AssetTexture texture, int discardLevel, float priority, int packet)
+ {
+ Texture = texture;
+ Update(discardLevel, priority, packet);
+ }
///
- /// Holds the texture asset to send.
+ /// Updates an image transfer with new information and recalculates
+ /// offsets
///
- private AssetBase m_asset;
+ /// New requested discard level
+ /// New requested priority
+ /// New requested packet offset
+ public void Update(int discardLevel, float priority, int packet)
+ {
+ Priority = priority;
+ DiscardLevel = Clamp(discardLevel, 0, Texture.LayerInfo.Length - 1);
+ StopPacket = GetPacketForBytePosition(Texture.LayerInfo[(Texture.LayerInfo.Length - 1) - DiscardLevel].End);
+ CurrentPacket = Clamp(packet, 1, TexturePacketCount());
+ }
- //public UUID assetID { get { return m_asset.FullID; } }
+ ///
+ /// Returns the total number of packets needed to transfer this texture,
+ /// including the first packet of size FIRST_IMAGE_PACKET_SIZE
+ ///
+ /// Total number of packets needed to transfer this texture
+ public int TexturePacketCount()
+ {
+ return ((Texture.AssetData.Length - FIRST_IMAGE_PACKET_SIZE + IMAGE_PACKET_SIZE - 1) / IMAGE_PACKET_SIZE) + 1;
+ }
- // private bool m_cancel = false;
+ ///
+ /// Returns the current byte offset for this transfer, calculated from
+ /// the CurrentPacket
+ ///
+ /// Current byte offset for this transfer
+ public int CurrentBytePosition()
+ {
+ return FIRST_IMAGE_PACKET_SIZE + (CurrentPacket - 1) * IMAGE_PACKET_SIZE;
+ }
- // See ITextureSender
+ ///
+ /// Returns the size, in bytes, of the last packet. This will be somewhere
+ /// between 1 and IMAGE_PACKET_SIZE bytes
+ ///
+ /// Size of the last packet in the transfer
+ public int LastPacketSize()
+ {
+ return Texture.AssetData.Length - (FIRST_IMAGE_PACKET_SIZE + ((TexturePacketCount() - 2) * IMAGE_PACKET_SIZE));
+ }
- // private bool m_sending = false;
+ ///
+ /// Find the packet number that contains a given byte position
+ ///
+ /// Byte position
+ /// Packet number that contains the given byte position
+ int GetPacketForBytePosition(int bytePosition)
+ {
+ return ((bytePosition - FIRST_IMAGE_PACKET_SIZE + IMAGE_PACKET_SIZE - 1) / IMAGE_PACKET_SIZE);
+ }
///
- /// This is actually the number of extra packets required to send the texture data! We always assume
- /// at least one is required.
+ /// Clamp a given value between a range
///
- private int NumPackets = 0;
+ /// Value to clamp
+ /// Minimum allowable value
+ /// Maximum allowable value
+ /// A value inclusively between lower and upper
+ static int Clamp(int value, int min, int max)
+ {
+ // First we check to see if we're greater than the max
+ value = (value > max) ? max : value;
+
+ // Then we check to see if we're less than the min.
+ value = (value < min) ? min : value;
+
+ // There's no check to see if min > max.
+ return value;
+ }
+ }
+
+ ///
+ /// A TextureSender handles the process of receiving a texture requested by the client from the
+ /// AssetCache, and then sending that texture back to the client.
+ ///
+ public class TextureSender : ITextureSender
+ {
+ private static readonly ILog m_log
+ = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
+
+ public bool ImageLoaded = false;
///
- /// Holds the packet number to send next. In this case, each packet is 1000 bytes long and starts
- /// at the 600th byte (0th indexed).
+ /// Holds the texture asset to send.
///
- private int PacketCounter = 0;
+ private AssetBase m_asset;
+ private bool m_cancel = false;
+ private bool m_sending = false;
+ private bool sendFirstPacket = false;
+ private int initialDiscardLevel = 0;
+ private int initialPacketNum = 0;
- private int RequestedDiscardLevel = -1;
+ private ImageDownload download;
private IClientAPI RequestUser;
- private uint StartPacketNumber = 0;
public TextureSender(IClientAPI client, int discardLevel, uint packetNumber)
{
RequestUser = client;
- RequestedDiscardLevel = discardLevel;
- StartPacketNumber = packetNumber;
+ initialDiscardLevel = discardLevel;
+ initialPacketNum = (int)packetNumber;
}
#region ITextureSender Members
public bool Cancel
{
- get { return false; }
- set
- {
- // m_cancel = value;
- }
+ get { return m_cancel; }
+ set { m_cancel = value; }
}
public bool Sending
{
- get { return false; }
- set
- {
- // m_sending = value;
- }
+ get { return m_sending; }
+ set { m_sending = value; }
}
// See ITextureSender
public void UpdateRequest(int discardLevel, uint packetNumber)
{
- RequestedDiscardLevel = discardLevel;
- StartPacketNumber = packetNumber;
- PacketCounter = (int) StartPacketNumber;
+ lock (download)
+ {
+ if (discardLevel < download.DiscardLevel)
+ m_log.DebugFormat("Image download {0} is changing from DiscardLevel {1} to {2}",
+ m_asset.FullID, download.DiscardLevel, discardLevel);
+
+ if (packetNumber != download.CurrentPacket)
+ m_log.DebugFormat("Image download {0} is changing from Packet {1} to {2}",
+ m_asset.FullID, download.CurrentPacket, packetNumber);
+
+ download.Update(discardLevel, download.Priority, (int)packetNumber);
+
+ sendFirstPacket = true;
+ }
}
// See ITextureSender
public bool SendTexturePacket()
{
- //m_log.DebugFormat("[TEXTURE SENDER]: Sending packet for {0}", m_asset.FullID);
-
- SendPacket();
- counter++;
- if ((NumPackets == 0) || (RequestedDiscardLevel == -1) || (PacketCounter > NumPackets) ||
- ((RequestedDiscardLevel > 0) && (counter > 50 + (NumPackets / (RequestedDiscardLevel + 1)))))
+ if (!m_cancel && download.CurrentPacket <= download.StopPacket)
+ {
+ SendPacket();
+ return false;
+ }
+ else
{
+ m_sending = false;
+ m_cancel = true;
+ sendFirstPacket = false;
return true;
}
- return false;
}
#endregion
@@ -140,9 +218,33 @@ namespace OpenSim.Region.Environment.Modules.Agent.TextureSender
public void TextureReceived(AssetBase asset)
{
m_asset = asset;
- NumPackets = CalculateNumPackets(asset.Data.Length);
- PacketCounter = (int) StartPacketNumber;
- ImageLoaded = true;
+
+ try
+ {
+ OpenMetaverse.AssetTexture texture = new OpenMetaverse.AssetTexture(m_asset.FullID, m_asset.Data);
+ if (texture.DecodeLayerBoundaries())
+ {
+ download = new ImageDownload(texture, initialDiscardLevel, 0.0f, initialPacketNum);
+ ImageLoaded = true;
+ m_sending = true;
+ m_cancel = false;
+ sendFirstPacket = true;
+
+ return;
+ }
+ else
+ {
+ m_log.Error("JPEG2000 texture decoding failed");
+ }
+ }
+ catch (Exception ex)
+ {
+ m_log.Error("JPEG2000 texture decoding threw an exception", ex);
+ }
+
+ ImageLoaded = false;
+ m_sending = false;
+ m_cancel = true;
}
///
@@ -150,66 +252,34 @@ namespace OpenSim.Region.Environment.Modules.Agent.TextureSender
///
private void SendPacket()
{
- if (PacketCounter <= NumPackets)
+ lock (download)
{
- if (PacketCounter == 0)
+ if (sendFirstPacket)
{
- if (NumPackets == 0)
+ sendFirstPacket = false;
+
+ if (m_asset.Data.Length <= 600)
{
RequestUser.SendImageFirstPart(1, m_asset.FullID, (uint)m_asset.Data.Length, m_asset.Data, 2);
- PacketCounter++;
+ return;
}
else
{
- byte[] ImageData1 = new byte[600];
- Array.Copy(m_asset.Data, 0, ImageData1, 0, 600);
-
- RequestUser.SendImageFirstPart(
- (ushort)(NumPackets), m_asset.FullID, (uint)m_asset.Data.Length, ImageData1, 2);
- PacketCounter++;
+ byte[] firstImageData = new byte[600];
+ Buffer.BlockCopy(m_asset.Data, 0, firstImageData, 0, 600);
+ RequestUser.SendImageFirstPart((ushort)download.TexturePacketCount(), m_asset.FullID, (uint)m_asset.Data.Length, firstImageData, 2);
}
}
- else
- {
- int size = m_asset.Data.Length - 600 - (1000 * (PacketCounter - 1));
- if (size > 1000) size = 1000;
- byte[] imageData = new byte[size];
- try
- {
- Array.Copy(m_asset.Data, 600 + (1000 * (PacketCounter - 1)), imageData, 0, size);
- }
- catch (ArgumentOutOfRangeException)
- {
- m_log.Error("[TEXTURE SENDER]: Unable to separate texture into multiple packets: Array bounds failure on asset:" +
- m_asset.FullID.ToString());
- return;
- }
-
- RequestUser.SendImageNextPart((ushort)PacketCounter, m_asset.FullID, imageData);
- PacketCounter++;
- }
- }
- }
- ///
- /// Calculate the number of packets that will be required to send the texture loaded into this sender
- /// This is actually the number of 1000 byte packets not including an initial 600 byte packet...
- ///
- ///
- ///
- private int CalculateNumPackets(int length)
- {
- int numPackets = 0;
+ int imagePacketSize = (download.CurrentPacket == download.TexturePacketCount() - 1) ?
+ download.LastPacketSize() : ImageDownload.IMAGE_PACKET_SIZE;
- if (length > 600)
- {
- //over 600 bytes so split up file
- int restData = (length - 600);
- int restPackets = ((restData + 999) / 1000);
- numPackets = restPackets;
- }
+ byte[] imageData = new byte[imagePacketSize];
+ Buffer.BlockCopy(m_asset.Data, download.CurrentBytePosition(), imageData, 0, imagePacketSize);
- return numPackets;
+ RequestUser.SendImageNextPart((ushort)download.CurrentPacket, m_asset.FullID, imageData);
+ ++download.CurrentPacket;
+ }
}
}
}
--
cgit v1.1