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

org.xnio.StreamConnection Maven / Gradle / Ivy

There is a newer version: 3.8.16.Final
Show newest version
/*
 * JBoss, Home of Professional Open Source
 *
 * Copyright 2013 Red Hat, Inc. and/or its affiliates.
 *
 * Licensed 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 org.xnio;

import java.io.IOException;
import java.util.concurrent.atomic.AtomicReference;

import org.jboss.logging.Logger;
import org.xnio.channels.CloseListenerSettable;
import org.xnio.conduits.ConduitStreamSinkChannel;
import org.xnio.conduits.ConduitStreamSourceChannel;
import org.xnio.conduits.StreamSinkConduit;
import org.xnio.conduits.StreamSourceConduit;

import static org.xnio._private.Messages.msg;

/**
 * A connection between peers.
 *
 * @author David M. Lloyd
 * @author Flavia Rainone
 */
public abstract class StreamConnection extends Connection implements CloseListenerSettable {

    /**
     * An empty listener used as a flag, to indicate that close listener has been invoked.
     */
    private static final ChannelListener INVOKED_CLOSE_LISTENER_FLAG = (StreamConnection connection)->{};
    private static final Logger log = Logger.getLogger("org.xnio.StreamConnection");

    private ConduitStreamSourceChannel sourceChannel;
    private ConduitStreamSinkChannel sinkChannel;
    private AtomicReference> closeListener;

    /**
     * Construct a new instance.
     *
     * @param thread the I/O thread
     */
    protected StreamConnection(final XnioIoThread thread) {
        super(thread);
        closeListener = new AtomicReference<>();
    }

    public void setCloseListener(final ChannelListener listener) {
        ChannelListener currentListener;
        ChannelListener newListener;
        do {
            newListener = listener;
            currentListener = closeListener.get();
            if (currentListener != null) {
                // channel is closed, just invoke the new listener and do not update closeListener
                if (currentListener == INVOKED_CLOSE_LISTENER_FLAG) {
                    ChannelListeners.invokeChannelListener(this, listener);
                    return;
                } else {
                    newListener = mergeListeners(currentListener, listener);
                }
            }
        } while (!closeListener.compareAndSet(currentListener, newListener));
    }

    private final ChannelListener mergeListeners(final ChannelListener listener1, final ChannelListener listener2) {
        return (StreamConnection channel) -> {
            listener1.handleEvent(channel);
            listener2.handleEvent(channel);
        };
    }

    @Override protected void notifyReadClosed() {

        try {
            this.getSourceChannel().shutdownReads();
        } catch (IOException e) {
            msg.connectionNotifyReadClosedFailed(e, this);
        }
    }

    @Override protected void notifyWriteClosed() {
        try {
            this.getSinkChannel().shutdownWrites();
        } catch (IOException e) {
            msg.connectionNotifyWriteClosedFailed(e, this);
        }
    }

    public ChannelListener getCloseListener() {
        return closeListener.get();
    }

    public ChannelListener.Setter getCloseSetter() {
        return new Setter<>(this);
    }

    /**
     * Set the source conduit for this channel.  The source channel will automatically be updated.
     *
     * @param conduit the source conduit for this channel
     */
    protected void setSourceConduit(StreamSourceConduit conduit) {
        this.sourceChannel = conduit == null ? null : new ConduitStreamSourceChannel(this, conduit);
    }

    /**
     * Set the sink conduit for this channel.  The sink channel will automatically be updated.
     *
     * @param conduit the sink conduit for this channel
     */
    protected void setSinkConduit(StreamSinkConduit conduit) {
        this.sinkChannel = conduit == null ? null : new ConduitStreamSinkChannel(this, conduit);
    }

    void invokeCloseListener() {
        // use a flag to indicate that closeListener has been invoked
        final ChannelListener listener = closeListener.getAndSet(INVOKED_CLOSE_LISTENER_FLAG);
        ChannelListeners.invokeChannelListener(this, listener);
    }

    private static  T notNull(T orig) throws IllegalStateException {
        if (orig == null) {
            throw msg.channelNotAvailable();
        }
        return orig;
    }

    /**
     * Get the source channel.
     *
     * @return the source channel
     */
    public ConduitStreamSourceChannel getSourceChannel() {
        return notNull(sourceChannel);
    }

    /**
     * Get the sink channel.
     *
     * @return the sink channel
     */
    public ConduitStreamSinkChannel getSinkChannel() {
        return notNull(sinkChannel);
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy