All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.signalfx.shaded.jetty.io.ChannelEndPoint Maven / Gradle / Ivy

//
//  ========================================================================
//  Copyright (c) 1995-2022 Mort Bay Consulting Pty Ltd and others.
//  ------------------------------------------------------------------------
//  All rights reserved. This program and the accompanying materials
//  are made available under the terms of the Eclipse Public License v1.0
//  and Apache License v2.0 which accompanies this distribution.
//
//      The Eclipse Public License is available at
//      http://www.eclipse.org/legal/epl-v10.html
//
//      The Apache License v2.0 is available at
//      http://www.opensource.org/licenses/apache2.0.php
//
//  You may elect to redistribute this code under either of these licenses.
//  ========================================================================
//

package com.signalfx.shaded.jetty.io;

import java.io.Closeable;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.nio.ByteBuffer;
import java.nio.channels.CancelledKeyException;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;

import com.signalfx.shaded.jetty.util.BufferUtil;
import com.signalfx.shaded.jetty.util.log.Log;
import com.signalfx.shaded.jetty.util.log.Logger;
import com.signalfx.shaded.jetty.util.thread.Invocable;
import com.signalfx.shaded.jetty.util.thread.Scheduler;

/**
 * Channel End Point.
 * 

Holds the channel and socket for an NIO endpoint. */ public abstract class ChannelEndPoint extends AbstractEndPoint implements ManagedSelector.Selectable { private static final Logger LOG = Log.getLogger(ChannelEndPoint.class); private final SocketChannel _channel; private final ManagedSelector _selector; private SelectionKey _key; private boolean _updatePending; // The current value for interestOps. private int _currentInterestOps; // The desired value for interestOps. private int _desiredInterestOps; private abstract class RunnableTask implements Runnable, Invocable { final String _operation; protected RunnableTask(String op) { _operation = op; } @Override public String toString() { return String.format("CEP:%s:%s:%s", ChannelEndPoint.this, _operation, getInvocationType()); } } private abstract class RunnableCloseable extends RunnableTask implements Closeable { protected RunnableCloseable(String op) { super(op); } @Override public void close() { try { ChannelEndPoint.this.close(); } catch (Throwable x) { LOG.warn(x); } } } private final ManagedSelector.SelectorUpdate _updateKeyAction = this::updateKeyAction; private final Runnable _runFillable = new RunnableCloseable("runFillable") { @Override public InvocationType getInvocationType() { return getFillInterest().getCallbackInvocationType(); } @Override public void run() { getFillInterest().fillable(); } }; private final Runnable _runCompleteWrite = new RunnableCloseable("runCompleteWrite") { @Override public InvocationType getInvocationType() { return getWriteFlusher().getCallbackInvocationType(); } @Override public void run() { getWriteFlusher().completeWrite(); } @Override public String toString() { return String.format("CEP:%s:%s:%s->%s", ChannelEndPoint.this, _operation, getInvocationType(), getWriteFlusher()); } }; private final Runnable _runCompleteWriteFillable = new RunnableCloseable("runCompleteWriteFillable") { @Override public InvocationType getInvocationType() { InvocationType fillT = getFillInterest().getCallbackInvocationType(); InvocationType flushT = getWriteFlusher().getCallbackInvocationType(); if (fillT == flushT) return fillT; if (fillT == InvocationType.EITHER && flushT == InvocationType.NON_BLOCKING) return InvocationType.EITHER; if (fillT == InvocationType.NON_BLOCKING && flushT == InvocationType.EITHER) return InvocationType.EITHER; return InvocationType.BLOCKING; } @Override public void run() { getWriteFlusher().completeWrite(); getFillInterest().fillable(); } }; public ChannelEndPoint(SocketChannel channel, ManagedSelector selector, SelectionKey key, Scheduler scheduler) { super(scheduler); _channel = channel; _selector = selector; _key = key; } @Override public InetSocketAddress getLocalAddress() { return (InetSocketAddress)_channel.socket().getLocalSocketAddress(); } @Override public InetSocketAddress getRemoteAddress() { return (InetSocketAddress)_channel.socket().getRemoteSocketAddress(); } @Override public boolean isOptimizedForDirectBuffers() { return true; } @Override public boolean isOpen() { return _channel.isOpen(); } @Override protected void doShutdownOutput() { try { Socket socket = _channel.socket(); if (!socket.isOutputShutdown()) socket.shutdownOutput(); } catch (IOException e) { LOG.debug(e); } } @Override public void doClose() { if (LOG.isDebugEnabled()) LOG.debug("doClose {}", this); try { _key.cancel(); _channel.close(); } catch (IOException e) { LOG.debug(e); } finally { super.doClose(); } } @Override public void onClose() { try { super.onClose(); } finally { if (_selector != null) _selector.destroyEndPoint(this); } } @Override public int fill(ByteBuffer buffer) throws IOException { if (isInputShutdown()) return -1; int pos = BufferUtil.flipToFill(buffer); int filled; try { filled = _channel.read(buffer); if (filled > 0) notIdle(); else if (filled == -1) shutdownInput(); } catch (IOException e) { LOG.debug(e); shutdownInput(); filled = -1; } finally { BufferUtil.flipToFlush(buffer, pos); } if (LOG.isDebugEnabled()) LOG.debug("filled {} {}", filled, BufferUtil.toDetailString(buffer)); return filled; } @Override public boolean flush(ByteBuffer... buffers) throws IOException { long flushed; try { flushed = _channel.write(buffers); if (LOG.isDebugEnabled()) LOG.debug("flushed {} {}", flushed, this); } catch (IOException e) { throw new EofException(e); } if (flushed > 0) notIdle(); for (ByteBuffer b : buffers) { if (!BufferUtil.isEmpty(b)) return false; } return true; } public SocketChannel getChannel() { return _channel; } @Override public Object getTransport() { return _channel; } @Override protected void needsFillInterest() { changeInterests(SelectionKey.OP_READ); } @Override protected void onIncompleteFlush() { changeInterests(SelectionKey.OP_WRITE); } @Override public Runnable onSelected() { // This method runs from the selector thread, // possibly concurrently with changeInterests(int). int readyOps = _key.readyOps(); int oldInterestOps; int newInterestOps; synchronized (this) { _updatePending = true; // Remove the readyOps, that here can only be OP_READ or OP_WRITE (or both). oldInterestOps = _desiredInterestOps; newInterestOps = oldInterestOps & ~readyOps; _desiredInterestOps = newInterestOps; } boolean fillable = (readyOps & SelectionKey.OP_READ) != 0; boolean flushable = (readyOps & SelectionKey.OP_WRITE) != 0; if (LOG.isDebugEnabled()) LOG.debug("onSelected {}->{} r={} w={} for {}", oldInterestOps, newInterestOps, fillable, flushable, this); // return task to complete the job Runnable task = fillable ? (flushable ? _runCompleteWriteFillable : _runFillable) : (flushable ? _runCompleteWrite : null); if (LOG.isDebugEnabled()) LOG.debug("task {}", task); return task; } private void updateKeyAction(Selector selector) { updateKey(); } @Override public void updateKey() { // This method runs from the selector thread, // possibly concurrently with changeInterests(int). try { int oldInterestOps; int newInterestOps; synchronized (this) { _updatePending = false; oldInterestOps = _currentInterestOps; newInterestOps = _desiredInterestOps; if (oldInterestOps != newInterestOps) { _currentInterestOps = newInterestOps; _key.interestOps(newInterestOps); } } if (LOG.isDebugEnabled()) LOG.debug("Key interests updated {} -> {} on {}", oldInterestOps, newInterestOps, this); } catch (CancelledKeyException x) { if (LOG.isDebugEnabled()) LOG.debug("Ignoring key update for cancelled key {}", this, x); close(); } catch (Throwable x) { LOG.warn("Ignoring key update for {}", this, x); close(); } } @Override public void replaceKey(SelectionKey newKey) { _key = newKey; } private void changeInterests(int operation) { // This method runs from any thread, possibly // concurrently with updateKey() and onSelected(). int oldInterestOps; int newInterestOps; boolean pending; synchronized (this) { pending = _updatePending; oldInterestOps = _desiredInterestOps; newInterestOps = oldInterestOps | operation; if (newInterestOps != oldInterestOps) _desiredInterestOps = newInterestOps; } if (LOG.isDebugEnabled()) LOG.debug("changeInterests p={} {}->{} for {}", pending, oldInterestOps, newInterestOps, this); if (!pending && _selector != null) _selector.submit(_updateKeyAction); } @Override public String toEndPointString() { // We do a best effort to print the right toString() and that's it. return String.format("%s{io=%d/%d,kio=%d,kro=%d}", super.toEndPointString(), _currentInterestOps, _desiredInterestOps, ManagedSelector.safeInterestOps(_key), ManagedSelector.safeReadyOps(_key)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy