From b5875ca07bf40e4b47b080b465e844f1a2c6998f Mon Sep 17 00:00:00 2001
From: jrandom <jrandom>
Date: Tue, 22 Mar 2005 02:00:10 +0000
Subject: [PATCH] 2005-03-21  jrandom     * Fixed the tunnel fragmentation
 handler to deal with multiple fragments       in a single message properly
 (rather than release the buffer into the       cache after processing the
 first one) (duh!)     * Added the batching preprocessor which will bundle
 together multiple       small messages inside a single tunnel message by
 delaying their delivery       up to .5s, or whenever the pending data will
 fill a full message,       whichever comes first.  This is disabled at the
 moment, since without the       above bugfix widely deployed, lots and lots
 of messages would fail.     * Within each tunnel pool, stick with a randomly
 selected peer for up to       .5s before randomizing and selecting again,
 instead of randomizing the       pool each time a tunnel is needed.

---
 .../i2p/router/tunnel/pool/TunnelPool.java    | 32 +++++++++++++++++++
 1 file changed, 32 insertions(+)

diff --git a/router/java/src/net/i2p/router/tunnel/pool/TunnelPool.java b/router/java/src/net/i2p/router/tunnel/pool/TunnelPool.java
index d217e69948..afa92ac6e1 100644
--- a/router/java/src/net/i2p/router/tunnel/pool/TunnelPool.java
+++ b/router/java/src/net/i2p/router/tunnel/pool/TunnelPool.java
@@ -33,6 +33,8 @@ public class TunnelPool {
     private int _buildsThisMinute;
     private long _currentMinute;
     private RefreshJob _refreshJob;
+    private TunnelInfo _lastSelected;
+    private long _lastSelectionPeriod;
     
     /**
      * Only 3 builds per minute per pool, even if we have failing tunnels,
@@ -50,6 +52,8 @@ public class TunnelPool {
         _peerSelector = sel;
         _builder = builder;
         _alive = false;
+        _lastSelectionPeriod = 0;
+        _lastSelected = null;
         _lifetimeProcessed = 0;
         _buildsThisMinute = 0;
         _currentMinute = ctx.clock().now();
@@ -77,6 +81,8 @@ public class TunnelPool {
     }
     public void shutdown() {
         _alive = false;
+        _lastSelectionPeriod = 0;
+        _lastSelected = null;
     }
 
     private int countUsableTunnels() {
@@ -153,6 +159,12 @@ public class TunnelPool {
         }
     }
     
+    /** 
+     * when selecting tunnels, stick with the same one for a brief 
+     * period to allow batching if we can.
+     */
+    private static final long SELECTION_PERIOD = 500;
+    
     /**
      * Pull a random tunnel out of the pool.  If there are none available but
      * the pool is configured to allow 0hop tunnels, this builds a fake one
@@ -161,7 +173,18 @@ public class TunnelPool {
      */
     public TunnelInfo selectTunnel() { return selectTunnel(true); }
     private TunnelInfo selectTunnel(boolean allowRecurseOnFail) {
+        long period = _context.clock().now();
+        period -= period % SELECTION_PERIOD;
         synchronized (_tunnels) {
+            if (_lastSelectionPeriod == period) {
+                if ( (_lastSelected != null) && 
+                     (_lastSelected.getExpiration() > period) &&
+                     (_tunnels.contains(_lastSelected)) )
+                    return _lastSelected;
+            }
+            _lastSelectionPeriod = period;
+            _lastSelected = null;
+
             if (_tunnels.size() <= 0) {
                 if (_log.shouldLog(Log.WARN))
                     _log.warn(toString() + ": No tunnels to select from");
@@ -172,6 +195,7 @@ public class TunnelPool {
                     TunnelInfo info = (TunnelInfo)_tunnels.get(i);
                     if (info.getExpiration() > _context.clock().now()) {
                         //_log.debug("Selecting tunnel: " + info + " - " + _tunnels);
+                        _lastSelected = info;
                         return info;
                     }
                 }
@@ -261,6 +285,10 @@ public class TunnelPool {
             if (_settings.isInbound() && (_settings.getDestination() != null) )
                 ls = locked_buildNewLeaseSet();
             remaining = _tunnels.size();
+            if (_lastSelected == info) {
+                _lastSelected = null;
+                _lastSelectionPeriod = 0;
+            }
         }
         
         _lifetimeProcessed += info.getProcessedMessagesCount();
@@ -297,6 +325,10 @@ public class TunnelPool {
             if (_settings.isInbound() && (_settings.getDestination() != null) )
                 ls = locked_buildNewLeaseSet();
             remaining = _tunnels.size();
+            if (_lastSelected == cfg) {
+                _lastSelected = null;
+                _lastSelectionPeriod = 0;
+            }
         }
         
         _lifetimeProcessed += cfg.getProcessedMessagesCount();
-- 
GitLab