diff --git a/common/src/main/java/io/netty/util/Recycler.java b/common/src/main/java/io/netty/util/Recycler.java index c074f6754c..d6b8941660 100644 --- a/common/src/main/java/io/netty/util/Recycler.java +++ b/common/src/main/java/io/netty/util/Recycler.java @@ -396,12 +396,14 @@ public abstract class Recycler { // While we also enforce the recycling ratio when we transfer objects from the WeakOrderQueue to the Stack // we better should enforce it as well early. Missing to do so may let the WeakOrderQueue grow very fast // without control - if (handleRecycleCount < interval) { - handleRecycleCount++; - // Drop the item to prevent recycling to aggressive. - return; + if (!handle.hasBeenRecycled) { + if (handleRecycleCount < interval) { + handleRecycleCount++; + // Drop the item to prevent from recycling too aggressively. + return; + } + handleRecycleCount = 0; } - handleRecycleCount = 0; Link tail = this.tail; int writeIndex; diff --git a/common/src/test/java/io/netty/util/RecyclerTest.java b/common/src/test/java/io/netty/util/RecyclerTest.java index f99ceae1d3..87e5b3d20b 100644 --- a/common/src/test/java/io/netty/util/RecyclerTest.java +++ b/common/src/test/java/io/netty/util/RecyclerTest.java @@ -20,6 +20,8 @@ import org.junit.jupiter.api.Timeout; import java.util.Random; import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicInteger; @@ -328,6 +330,43 @@ public class RecyclerTest { assertNotSame(recycler.get(), o2); } + @Test + public void testRecycleAtTwoThreadsMulti() throws Exception { + final Recycler recycler = newRecycler(256); + final HandledObject o = recycler.get(); + + ExecutorService single = Executors.newSingleThreadExecutor(); + + final CountDownLatch latch1 = new CountDownLatch(1); + single.execute(new Runnable() { + @Override + public void run() { + o.recycle(); + latch1.countDown(); + } + }); + assertTrue(latch1.await(100, TimeUnit.MILLISECONDS)); + final HandledObject o2 = recycler.get(); + // Always recycler the first object, that is Ok + assertSame(o2, o); + + final CountDownLatch latch2 = new CountDownLatch(1); + single.execute(new Runnable() { + @Override + public void run() { + //The object should be recycled + o2.recycle(); + latch2.countDown(); + } + }); + assertTrue(latch2.await(100, TimeUnit.MILLISECONDS)); + + // It should be the same object, right? + final HandledObject o3 = recycler.get(); + assertSame(o3, o); + single.shutdown(); + } + @Test public void testMaxCapacityWithRecycleAtDifferentThread() throws Exception { final int maxCapacity = 4; // Choose the number smaller than WeakOrderQueue.LINK_CAPACITY