From c6cfb683f5e0b7246b6a87e889a4ac68a85007b9 Mon Sep 17 00:00:00 2001 From: nmittler Date: Fri, 10 Apr 2015 09:34:39 -0700 Subject: [PATCH] Removing Http2StreamRemovalPolicy Motivation: Due to a recent flurry of cleanup and fixes, we no longer need the stream removal policy to protect against recently removed streams. We should get rid of it. Modifications: Removed Http2StreamRemovalPolicy and everywhere it's used. Result: Fixes #3448 --- .../codec/http2/DefaultHttp2Connection.java | 60 ++++------ .../DefaultHttp2StreamRemovalPolicy.java | 106 ------------------ .../handler/codec/http2/Http2CodecUtil.java | 37 +----- .../codec/http2/Http2StreamRemovalPolicy.java | 42 ------- 4 files changed, 20 insertions(+), 225 deletions(-) delete mode 100644 codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2StreamRemovalPolicy.java delete mode 100644 codec-http2/src/main/java/io/netty/handler/codec/http2/Http2StreamRemovalPolicy.java diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Connection.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Connection.java index a83436b649..7c2fe82864 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Connection.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Connection.java @@ -19,7 +19,6 @@ import static io.netty.handler.codec.http2.Http2CodecUtil.CONNECTION_STREAM_ID; import static io.netty.handler.codec.http2.Http2CodecUtil.DEFAULT_PRIORITY_WEIGHT; import static io.netty.handler.codec.http2.Http2CodecUtil.MAX_WEIGHT; import static io.netty.handler.codec.http2.Http2CodecUtil.MIN_WEIGHT; -import static io.netty.handler.codec.http2.Http2CodecUtil.immediateRemovalPolicy; import static io.netty.handler.codec.http2.Http2Error.PROTOCOL_ERROR; import static io.netty.handler.codec.http2.Http2Error.REFUSED_STREAM; import static io.netty.handler.codec.http2.Http2Exception.closedStreamError; @@ -35,7 +34,6 @@ import static io.netty.handler.codec.http2.Http2Stream.State.RESERVED_REMOTE; import static io.netty.util.internal.ObjectUtil.checkNotNull; import io.netty.buffer.ByteBuf; -import io.netty.handler.codec.http2.Http2StreamRemovalPolicy.Action; import io.netty.util.collection.IntObjectHashMap; import io.netty.util.collection.IntObjectMap; import io.netty.util.collection.PrimitiveCollections; @@ -71,36 +69,16 @@ public class DefaultHttp2Connection implements Http2Connection { final ActiveStreams activeStreams; /** - * Creates a connection with an immediate stream removal policy. + * Creates a new connection with the given settings. * * @param server * whether or not this end-point is the server-side of the HTTP/2 connection. */ public DefaultHttp2Connection(boolean server) { - this(server, immediateRemovalPolicy()); - } - - /** - * Creates a new connection with the given settings. - * - * @param server - * whether or not this end-point is the server-side of the HTTP/2 connection. - * @param removalPolicy - * the policy to be used for removal of closed stream. - */ - public DefaultHttp2Connection(boolean server, Http2StreamRemovalPolicy removalPolicy) { - activeStreams = new ActiveStreams(listeners, checkNotNull(removalPolicy, "removalPolicy")); + activeStreams = new ActiveStreams(listeners); localEndpoint = new DefaultEndpoint(server); remoteEndpoint = new DefaultEndpoint(!server); - // Tell the removal policy how to remove a stream from this connection. - removalPolicy.setAction(new Action() { - @Override - public void removeStream(Http2Stream stream) { - DefaultHttp2Connection.this.removeStream((DefaultStream) stream); - } - }); - // Add the connection stream to the map. streamMap.put(connectionStream.id(), connectionStream); } @@ -997,31 +975,32 @@ public class DefaultHttp2Connection implements Http2Connection { } /** - * Default implementation of the {@link ActiveStreams} class. + * Allows events which would modify the collection of active streams to be queued while iterating via {@link + * #forEachActiveStream(Http2StreamVisitor)}. */ - private static final class ActiveStreams { + interface Event { /** - * Allows events which would modify {@link #streams} to be queued while iterating over {@link #streams}. + * Trigger the original intention of this event. Expect to modify the active streams list. + *

+ * If a {@link RuntimeException} object is thrown it will be logged and not propagated. + * Throwing from this method is not supported and is considered a programming error. */ - interface Event { - /** - * Trigger the original intention of this event. Expect to modify {@link #streams}. - *

- * If a {@link RuntimeException} object is thrown it will be logged and not propagated. - * Throwing from this method is not supported and is considered a programming error. - */ - void process(); - } + void process(); + } + + /** + * Manages the list of currently active streams. Queues any {@link Event}s that would modify the list of + * active streams in order to prevent modification while iterating. + */ + private final class ActiveStreams { private final List listeners; - private final Http2StreamRemovalPolicy removalPolicy; private final Queue pendingEvents = new ArrayDeque(4); private final Set streams = new LinkedHashSet(); private int pendingIterations; - public ActiveStreams(List listeners, Http2StreamRemovalPolicy removalPolicy) { + public ActiveStreams(List listeners) { this.listeners = listeners; - this.removalPolicy = removalPolicy; } public int size() { @@ -1110,8 +1089,7 @@ public class DefaultHttp2Connection implements Http2Connection { } } } finally { - // Mark this stream for removal. - removalPolicy.markForRemoval(stream); + removeStream(stream); } } } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2StreamRemovalPolicy.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2StreamRemovalPolicy.java deleted file mode 100644 index 550bf4c27a..0000000000 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2StreamRemovalPolicy.java +++ /dev/null @@ -1,106 +0,0 @@ -/* - * Copyright 2014 The Netty Project - * - * The Netty Project licenses this file to you under the Apache License, version 2.0 (the - * "License"); you may not use this file except in compliance with the License. You may obtain a - * copy of the License at: - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software distributed under the License - * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express - * or implied. See the License for the specific language governing permissions and limitations under - * the License. - */ -package io.netty.handler.codec.http2; - -import static java.util.concurrent.TimeUnit.NANOSECONDS; -import static java.util.concurrent.TimeUnit.SECONDS; -import io.netty.channel.ChannelHandlerAdapter; -import io.netty.channel.ChannelHandlerContext; - -import java.util.ArrayDeque; -import java.util.Queue; -import java.util.concurrent.ScheduledFuture; - -/** - * A {@link Http2StreamRemovalPolicy} that periodically runs garbage collection on streams that have - * been marked for removal. - */ -public class DefaultHttp2StreamRemovalPolicy extends ChannelHandlerAdapter implements - Http2StreamRemovalPolicy, Runnable { - - /** - * The interval (in ns) at which the removed priority garbage collector runs. - */ - private static final long GARBAGE_COLLECTION_INTERVAL = SECONDS.toNanos(5); - - private final Queue garbage = new ArrayDeque(); - private ScheduledFuture timerFuture; - private Action action; - - @Override - public void handlerAdded(ChannelHandlerContext ctx) throws Exception { - // Schedule the periodic timer for performing the policy check. - timerFuture = ctx.channel().eventLoop().scheduleWithFixedDelay(this, - GARBAGE_COLLECTION_INTERVAL, - GARBAGE_COLLECTION_INTERVAL, - NANOSECONDS); - } - - @Override - public void handlerRemoved(ChannelHandlerContext ctx) throws Exception { - // Cancel the periodic timer. - if (timerFuture != null) { - timerFuture.cancel(false); - timerFuture = null; - } - } - - @Override - public void setAction(Action action) { - this.action = action; - } - - @Override - public void markForRemoval(Http2Stream stream) { - garbage.add(new Garbage(stream)); - } - - /** - * Runs garbage collection of any streams marked for removal > - * {@link #GARBAGE_COLLECTION_INTERVAL} nanoseconds ago. - */ - @Override - public void run() { - if (garbage.isEmpty() || action == null) { - return; - } - - long time = System.nanoTime(); - for (;;) { - Garbage next = garbage.peek(); - if (next == null) { - break; - } - if (time - next.removalTime > GARBAGE_COLLECTION_INTERVAL) { - garbage.remove(); - action.removeStream(next.stream); - } else { - break; - } - } - } - - /** - * Wrapper around a stream and its removal time. - */ - private static final class Garbage { - private final long removalTime = System.nanoTime(); - private final Http2Stream stream; - - Garbage(Http2Stream stream) { - this.stream = stream; - } - } -} diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2CodecUtil.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2CodecUtil.java index 19b27f5938..87735a5326 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2CodecUtil.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2CodecUtil.java @@ -16,14 +16,13 @@ package io.netty.handler.codec.http2; import static io.netty.util.CharsetUtil.UTF_8; -import static io.netty.util.internal.ObjectUtil.checkNotNull; + import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import io.netty.channel.Channel; import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelPromise; import io.netty.channel.DefaultChannelPromise; -import io.netty.handler.codec.http2.Http2StreamRemovalPolicy.Action; import io.netty.util.concurrent.EventExecutor; /** @@ -113,30 +112,6 @@ public final class Http2CodecUtil { return Unpooled.wrappedBuffer(EMPTY_PING); } - /** - * Returns a simple {@link Http2StreamRemovalPolicy} that immediately calls back the - * {@link Action} when a stream is marked for removal. - */ - public static Http2StreamRemovalPolicy immediateRemovalPolicy() { - return new Http2StreamRemovalPolicy() { - private Action action; - - @Override - public void setAction(Action action) { - this.action = checkNotNull(action, "action"); - } - - @Override - public void markForRemoval(Http2Stream stream) { - if (action == null) { - throw new IllegalStateException( - "Action must be called before removing streams."); - } - action.removeStream(stream); - } - }; - } - /** * Iteratively looks through the causaility chain for the given exception and returns the first * {@link Http2Exception} or {@code null} if none. @@ -321,15 +296,5 @@ public final class Http2CodecUtil { } } - /** - * Fails the given promise with the cause and then re-throws the cause. - */ - public static T failAndThrow(ChannelPromise promise, T cause) throws T { - if (!promise.isDone()) { - promise.setFailure(cause); - } - throw cause; - } - private Http2CodecUtil() { } } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2StreamRemovalPolicy.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2StreamRemovalPolicy.java deleted file mode 100644 index d8e57dd7f7..0000000000 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2StreamRemovalPolicy.java +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright 2014 The Netty Project - * - * The Netty Project licenses this file to you under the Apache License, version 2.0 (the - * "License"); you may not use this file except in compliance with the License. You may obtain a - * copy of the License at: - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software distributed under the License - * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express - * or implied. See the License for the specific language governing permissions and limitations under - * the License. - */ -package io.netty.handler.codec.http2; - -/** - * A policy for determining when it is appropriate to remove streams from an HTTP/2 stream registry. - */ -public interface Http2StreamRemovalPolicy { - - /** - * Performs the action of removing the stream. - */ - interface Action { - /** - * Removes the stream from the registry. - */ - void removeStream(Http2Stream stream); - } - - /** - * Sets the removal action. - */ - void setAction(Action action); - - /** - * Marks the given stream for removal. When this policy has determined that the given stream - * should be removed, it will call back the {@link Action}. - */ - void markForRemoval(Http2Stream stream); -}