aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/OpenSim/Region/ClientStack/Linden/Caps
diff options
context:
space:
mode:
Diffstat (limited to 'OpenSim/Region/ClientStack/Linden/Caps')
-rw-r--r--OpenSim/Region/ClientStack/Linden/Caps/GetTextureModule.cs186
1 files changed, 172 insertions, 14 deletions
diff --git a/OpenSim/Region/ClientStack/Linden/Caps/GetTextureModule.cs b/OpenSim/Region/ClientStack/Linden/Caps/GetTextureModule.cs
index 24a0190..d1ec4aa 100644
--- a/OpenSim/Region/ClientStack/Linden/Caps/GetTextureModule.cs
+++ b/OpenSim/Region/ClientStack/Linden/Caps/GetTextureModule.cs
@@ -58,6 +58,13 @@ namespace OpenSim.Region.ClientStack.Linden
58 public Hashtable request; 58 public Hashtable request;
59 } 59 }
60 60
61 public class aPollResponse
62 {
63 public Hashtable response;
64 public int bytes;
65 }
66
67
61 private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); 68 private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
62 69
63 private Scene m_scene; 70 private Scene m_scene;
@@ -72,6 +79,8 @@ namespace OpenSim.Region.ClientStack.Linden
72 private static OpenMetaverse.BlockingQueue<aPollRequest> m_queue = 79 private static OpenMetaverse.BlockingQueue<aPollRequest> m_queue =
73 new OpenMetaverse.BlockingQueue<aPollRequest>(); 80 new OpenMetaverse.BlockingQueue<aPollRequest>();
74 81
82 private Dictionary<UUID,PollServiceTextureEventArgs> m_pollservices = new Dictionary<UUID,PollServiceTextureEventArgs>();
83
75 #region ISharedRegionModule Members 84 #region ISharedRegionModule Members
76 85
77 public void Initialise(IConfigSource source) 86 public void Initialise(IConfigSource source)
@@ -88,6 +97,7 @@ namespace OpenSim.Region.ClientStack.Linden
88 { 97 {
89 m_scene.EventManager.OnRegisterCaps -= RegisterCaps; 98 m_scene.EventManager.OnRegisterCaps -= RegisterCaps;
90 m_scene.EventManager.OnDeregisterCaps -= DeregisterCaps; 99 m_scene.EventManager.OnDeregisterCaps -= DeregisterCaps;
100 m_scene.EventManager.OnThrottleUpdate -= ThrottleUpdate;
91 m_scene = null; 101 m_scene = null;
92 } 102 }
93 103
@@ -98,6 +108,7 @@ namespace OpenSim.Region.ClientStack.Linden
98 108
99 m_scene.EventManager.OnRegisterCaps += RegisterCaps; 109 m_scene.EventManager.OnRegisterCaps += RegisterCaps;
100 m_scene.EventManager.OnDeregisterCaps += DeregisterCaps; 110 m_scene.EventManager.OnDeregisterCaps += DeregisterCaps;
111 m_scene.EventManager.OnThrottleUpdate += ThrottleUpdate;
101 112
102 if (m_workerThreads == null) 113 if (m_workerThreads == null)
103 { 114 {
@@ -115,6 +126,56 @@ namespace OpenSim.Region.ClientStack.Linden
115 } 126 }
116 } 127 }
117 } 128 }
129 private int ExtractImageThrottle(byte[] pthrottles)
130 {
131
132 byte[] adjData;
133 int pos = 0;
134
135 if (!BitConverter.IsLittleEndian)
136 {
137 byte[] newData = new byte[7 * 4];
138 Buffer.BlockCopy(pthrottles, 0, newData, 0, 7 * 4);
139
140 for (int i = 0; i < 7; i++)
141 Array.Reverse(newData, i * 4, 4);
142
143 adjData = newData;
144 }
145 else
146 {
147 adjData = pthrottles;
148 }
149
150 // 0.125f converts from bits to bytes
151 //int resend = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
152 //pos += 4;
153 // int land = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
154 //pos += 4;
155 // int wind = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
156 // pos += 4;
157 // int cloud = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
158 // pos += 4;
159 // int task = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
160 // pos += 4;
161 pos = pos + 20;
162 int texture = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f); //pos += 4;
163 //int asset = (int)(BitConverter.ToSingle(adjData, pos) * 0.125f);
164 return texture;
165 }
166
167 // Now we know when the throttle is changed by the client in the case of a root agent or by a neighbor region in the case of a child agent.
168 public void ThrottleUpdate(ScenePresence p)
169 {
170 byte[] throttles = p.ControllingClient.GetThrottlesPacked(1);
171 UUID user = p.UUID;
172 int imagethrottle = ExtractImageThrottle(throttles);
173 PollServiceTextureEventArgs args;
174 if (m_pollservices.TryGetValue(user,out args))
175 {
176 args.UpdateThrottle(imagethrottle);
177 }
178 }
118 179
119 public void PostInitialise() 180 public void PostInitialise()
120 { 181 {
@@ -142,20 +203,25 @@ namespace OpenSim.Region.ClientStack.Linden
142 { 203 {
143 private List<Hashtable> requests = 204 private List<Hashtable> requests =
144 new List<Hashtable>(); 205 new List<Hashtable>();
145 private Dictionary<UUID, Hashtable> responses = 206 private Dictionary<UUID, aPollResponse> responses =
146 new Dictionary<UUID, Hashtable>(); 207 new Dictionary<UUID, aPollResponse>();
147 208
148 private Scene m_scene; 209 private Scene m_scene;
149 210 private CapsDataThrottler m_throttler = new CapsDataThrottler(100000, 1400000,10000);
150 public PollServiceTextureEventArgs(UUID pId, Scene scene) : 211 public PollServiceTextureEventArgs(UUID pId, Scene scene) :
151 base(null, null, null, null, pId, int.MaxValue) 212 base(null, null, null, null, pId, int.MaxValue)
152 { 213 {
153 m_scene = scene; 214 m_scene = scene;
154 215 // x is request id, y is userid
155 HasEvents = (x, y) => 216 HasEvents = (x, y) =>
156 { 217 {
157 lock (responses) 218 lock (responses)
158 return responses.ContainsKey(x); 219 {
220 bool ret = m_throttler.hasEvents(x, responses);
221 m_throttler.ProcessTime();
222 return ret;
223
224 }
159 }; 225 };
160 GetEvents = (x, y) => 226 GetEvents = (x, y) =>
161 { 227 {
@@ -163,7 +229,7 @@ namespace OpenSim.Region.ClientStack.Linden
163 { 229 {
164 try 230 try
165 { 231 {
166 return responses[x]; 232 return responses[x].response;
167 } 233 }
168 finally 234 finally
169 { 235 {
@@ -171,14 +237,14 @@ namespace OpenSim.Region.ClientStack.Linden
171 } 237 }
172 } 238 }
173 }; 239 };
174 240 // x is request id, y is request data hashtable
175 Request = (x, y) => 241 Request = (x, y) =>
176 { 242 {
177 aPollRequest reqinfo = new aPollRequest(); 243 aPollRequest reqinfo = new aPollRequest();
178 reqinfo.thepoll = this; 244 reqinfo.thepoll = this;
179 reqinfo.reqID = x; 245 reqinfo.reqID = x;
180 reqinfo.request = y; 246 reqinfo.request = y;
181 247
182 m_queue.Enqueue(reqinfo); 248 m_queue.Enqueue(reqinfo);
183 }; 249 };
184 250
@@ -220,16 +286,29 @@ namespace OpenSim.Region.ClientStack.Linden
220 response["content_type"] = "text/plain"; 286 response["content_type"] = "text/plain";
221 response["keepalive"] = false; 287 response["keepalive"] = false;
222 response["reusecontext"] = false; 288 response["reusecontext"] = false;
223 289
224 lock (responses) 290 lock (responses)
225 responses[requestID] = response; 291 responses[requestID] = new aPollResponse() {bytes = 0, response = response};
226 292
227 return; 293 return;
228 } 294 }
229 295
230 response = m_getTextureHandler.Handle(requestinfo.request); 296 response = m_getTextureHandler.Handle(requestinfo.request);
231 lock (responses) 297 lock (responses)
232 responses[requestID] = response; 298 {
299 responses[requestID] = new aPollResponse()
300 {
301 bytes = (int) response["int_bytes"],
302 response = response
303 };
304
305 }
306 m_throttler.ProcessTime();
307 }
308
309 internal void UpdateThrottle(int pimagethrottle)
310 {
311 m_throttler.ThrottleBytes = pimagethrottle;
233 } 312 }
234 } 313 }
235 314
@@ -254,19 +333,23 @@ namespace OpenSim.Region.ClientStack.Linden
254 protocol = "https"; 333 protocol = "https";
255 } 334 }
256 caps.RegisterHandler("GetTexture", String.Format("{0}://{1}:{2}{3}", protocol, hostName, port, capUrl)); 335 caps.RegisterHandler("GetTexture", String.Format("{0}://{1}:{2}{3}", protocol, hostName, port, capUrl));
257 336 m_pollservices.Add(agentID, args);
258 m_capsDict[agentID] = capUrl; 337 m_capsDict[agentID] = capUrl;
259 } 338 }
260 339
261 private void DeregisterCaps(UUID agentID, Caps caps) 340 private void DeregisterCaps(UUID agentID, Caps caps)
262 { 341 {
263 string capUrl; 342 string capUrl;
264 343 PollServiceTextureEventArgs args;
265 if (m_capsDict.TryGetValue(agentID, out capUrl)) 344 if (m_capsDict.TryGetValue(agentID, out capUrl))
266 { 345 {
267 MainServer.Instance.RemoveHTTPHandler("", capUrl); 346 MainServer.Instance.RemoveHTTPHandler("", capUrl);
268 m_capsDict.Remove(agentID); 347 m_capsDict.Remove(agentID);
269 } 348 }
349 if (m_pollservices.TryGetValue(agentID, out args))
350 {
351 m_pollservices.Remove(agentID);
352 }
270 } 353 }
271 354
272 private void DoTextureRequests() 355 private void DoTextureRequests()
@@ -279,4 +362,79 @@ namespace OpenSim.Region.ClientStack.Linden
279 } 362 }
280 } 363 }
281 } 364 }
365
366 internal sealed class CapsDataThrottler
367 {
368
369 private volatile int currenttime = 0;
370 private volatile int lastTimeElapsed = 0;
371 private volatile int BytesSent = 0;
372 private int oversizedImages = 0;
373 public CapsDataThrottler(int pBytes, int max, int min)
374 {
375 ThrottleBytes = pBytes;
376 lastTimeElapsed = Util.EnvironmentTickCount();
377 }
378 public bool hasEvents(UUID key, Dictionary<UUID, GetTextureModule.aPollResponse> responses)
379 {
380 PassTime();
381 // Note, this is called IN LOCK
382 bool haskey = responses.ContainsKey(key);
383 if (!haskey)
384 {
385 return false;
386 }
387 GetTextureModule.aPollResponse response;
388 if (responses.TryGetValue(key,out response))
389 {
390
391 // Normal
392 if (BytesSent + response.bytes <= ThrottleBytes)
393 {
394 BytesSent += response.bytes;
395 //TimeBasedAction timeBasedAction = new TimeBasedAction { byteRemoval = response.bytes, requestId = key, timeMS = currenttime + 1000, unlockyn = false };
396 //m_actions.Add(timeBasedAction);
397 return true;
398 }
399 // Big textures
400 else if (response.bytes > ThrottleBytes && oversizedImages <= ((ThrottleBytes%50000) + 1))
401 {
402 Interlocked.Increment(ref oversizedImages);
403 BytesSent += response.bytes;
404 //TimeBasedAction timeBasedAction = new TimeBasedAction { byteRemoval = response.bytes, requestId = key, timeMS = currenttime + (((response.bytes % ThrottleBytes)+1)*1000) , unlockyn = false };
405 //m_actions.Add(timeBasedAction);
406 return true;
407 }
408 else
409 {
410 return false;
411 }
412 }
413
414 return haskey;
415 }
416 public void ProcessTime()
417 {
418 PassTime();
419 }
420
421
422 private void PassTime()
423 {
424 currenttime = Util.EnvironmentTickCount();
425 int timeElapsed = Util.EnvironmentTickCountSubtract(currenttime, lastTimeElapsed);
426 //processTimeBasedActions(responses);
427 if (Util.EnvironmentTickCountSubtract(currenttime, timeElapsed) >= 1000)
428 {
429 lastTimeElapsed = Util.EnvironmentTickCount();
430 BytesSent -= ThrottleBytes;
431 if (BytesSent < 0) BytesSent = 0;
432 if (BytesSent < ThrottleBytes)
433 {
434 oversizedImages = 0;
435 }
436 }
437 }
438 public int ThrottleBytes;
439 }
282} 440}