
com.hazelcast.nio.tcp.spinning.SpinningOutputThread Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2016, Hazelcast, Inc. All Rights Reserved.
*
* 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 com.hazelcast.nio.tcp.spinning;
import com.hazelcast.instance.HazelcastThreadGroup;
import com.hazelcast.logging.ILogger;
import com.hazelcast.nio.tcp.TcpIpConnection;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import static java.lang.System.arraycopy;
import static java.util.concurrent.atomic.AtomicReferenceFieldUpdater.newUpdater;
public class SpinningOutputThread extends Thread {
private static final SocketWriters SHUTDOWN = new SocketWriters();
private static final AtomicReferenceFieldUpdater CONNECTION_HANDLERS
= newUpdater(SpinningOutputThread.class, SocketWriters.class, "socketWriters");
private final ILogger logger;
private volatile SocketWriters socketWriters;
public SpinningOutputThread(HazelcastThreadGroup threadGroup, ILogger logger) {
super(threadGroup.getInternalThreadGroup(), "out-thread");
this.logger = logger;
this.socketWriters = new SocketWriters();
}
public void addConnection(TcpIpConnection connection) {
SpinningSocketWriter writer = (SpinningSocketWriter) connection.getSocketWriter();
for (; ; ) {
SocketWriters current = socketWriters;
if (current == SHUTDOWN) {
return;
}
int length = current.writers.length;
SpinningSocketWriter[] newWriters = new SpinningSocketWriter[length + 1];
arraycopy(current.writers, 0, newWriters, 0, length);
newWriters[length] = writer;
SocketWriters update = new SocketWriters(newWriters);
if (CONNECTION_HANDLERS.compareAndSet(this, current, update)) {
return;
}
}
}
public void removeConnection(TcpIpConnection connection) {
SpinningSocketWriter writeHandlers = (SpinningSocketWriter) connection.getSocketWriter();
for (; ; ) {
SocketWriters current = socketWriters;
if (current == SHUTDOWN) {
return;
}
int indexOf = current.indexOf(writeHandlers);
if (indexOf == -1) {
return;
}
int length = current.writers.length;
SpinningSocketWriter[] newWriters = new SpinningSocketWriter[length - 1];
int destIndex = 0;
for (int sourceIndex = 0; sourceIndex < length; sourceIndex++) {
if (sourceIndex != indexOf) {
newWriters[destIndex] = current.writers[sourceIndex];
destIndex++;
}
}
SocketWriters update = new SocketWriters(newWriters);
if (CONNECTION_HANDLERS.compareAndSet(this, current, update)) {
return;
}
}
}
public void shutdown() {
socketWriters = SHUTDOWN;
interrupt();
}
@Override
public void run() {
for (; ; ) {
SocketWriters handlers = socketWriters;
if (handlers == SHUTDOWN) {
return;
}
for (SpinningSocketWriter writer : handlers.writers) {
try {
writer.write();
} catch (Throwable t) {
writer.onFailure(t);
}
}
}
}
static class SocketWriters {
final SpinningSocketWriter[] writers;
public SocketWriters() {
this(new SpinningSocketWriter[0]);
}
public SocketWriters(SpinningSocketWriter[] writers) {
this.writers = writers;
}
public int indexOf(SpinningSocketWriter handler) {
for (int k = 0; k < writers.length; k++) {
if (writers[k] == handler) {
return k;
}
}
return -1;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy