-
Notifications
You must be signed in to change notification settings - Fork 476
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Added TcpConnectionNonBlocking (https://issues.redhat.com/browse/JG…
…RP-2759) - Added non_blocking_sends and max_send_queue to TCP/TcpConnection/TcpBaseServer
- Loading branch information
Showing
7 changed files
with
210 additions
and
28 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
164 changes: 164 additions & 0 deletions
164
src/org/jgroups/blocks/cs/TcpConnectionNonBlocking.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,164 @@ | ||
package org.jgroups.blocks.cs; | ||
|
||
import org.jgroups.Address; | ||
import org.jgroups.util.ByteArray; | ||
import org.jgroups.util.ThreadFactory; | ||
import org.jgroups.util.Util; | ||
|
||
import javax.net.ssl.SSLException; | ||
import javax.net.ssl.SSLHandshakeException; | ||
import java.io.EOFException; | ||
import java.io.IOException; | ||
import java.net.InetAddress; | ||
import java.net.Socket; | ||
import java.net.SocketException; | ||
import java.util.concurrent.ArrayBlockingQueue; | ||
import java.util.concurrent.BlockingQueue; | ||
import java.util.concurrent.atomic.LongAdder; | ||
|
||
/** | ||
* TCP connection which (despite the fancy name) blocks only a single thread at most. Uses a bounded queue, to which | ||
* senders add their messages, and a single consumer sending the messages. When the queue is full, messages will | ||
* get dropped. Therefore, at most one thread is blocked on TCP write when the send-window is full.<br/> | ||
* Link: https://issues.redhat.com/browse/JGRP-2759 | ||
* @author Bela Ban | ||
* @since 5.3.3 | ||
*/ | ||
public class TcpConnectionNonBlocking extends TcpConnection { | ||
protected BlockingQueue<ByteArray> queue; | ||
protected int max_size=128; | ||
protected volatile Sender sender; | ||
protected final LongAdder dropped_msgs=new LongAdder(); | ||
|
||
|
||
public TcpConnectionNonBlocking(Address peer_addr, TcpBaseServer server) throws Exception { | ||
super(peer_addr, server); | ||
} | ||
|
||
public TcpConnectionNonBlocking(Socket s, TcpServer server) throws Exception { | ||
super(s, server); | ||
} | ||
|
||
|
||
public int maxSize() {return max_size;} | ||
public TcpConnectionNonBlocking maxSize(int s) {max_size=s; return this;} | ||
public long droppedMessages() {return dropped_msgs.sum();} | ||
public int queueSize() {return queue != null? queue.size() : 0;} | ||
|
||
|
||
@Override public void start() { | ||
super.start(); | ||
queue=new ArrayBlockingQueue<>(max_size); | ||
if(sender != null) | ||
sender.stop(); | ||
sender=new Sender(server.factory).start(); | ||
} | ||
|
||
@Override public void close() throws IOException { | ||
super.close(); | ||
if(sender != null) { | ||
sender.stop(); | ||
sender=null; | ||
} | ||
} | ||
|
||
@Override | ||
public void send(byte[] data, int offset, int length) throws Exception { | ||
|
||
// to be on the safe side, we copy the data: some bundlers (e.g. TransferQueueBundler) reuse a buffer to | ||
// serialize messages to and - before the data is sent by the sender thread - the buffer might be changed! | ||
// This is similar to what NioConnection does on a partial write. If the send was synchronous (like in | ||
// TcpConnection), we would not have to copy the data | ||
ByteArray buf=new ByteArray(data, offset, length).copy(); | ||
boolean added=queue.offer(buf); | ||
if(!added) | ||
dropped_msgs.increment(); | ||
} | ||
|
||
@Override | ||
public String toString() { | ||
return String.format("%s [%d/%d, %d drops, sender: %b]", super.toString(), queueSize(), maxSize(), | ||
droppedMessages(), senderRunning()); | ||
} | ||
|
||
protected String name() { | ||
InetAddress local=sock.getLocalAddress(), remote=sock.getInetAddress(); | ||
String l=local != null? Util.shortName(local) : "<null>"; | ||
String r=remote != null? Util.shortName(remote) : "<null>"; | ||
return String.format("Connection.Sender [%s:%s-%s:%s]", l, sock.getLocalPort(), r, sock.getPort()); | ||
} | ||
|
||
|
||
|
||
protected boolean senderRunning() { | ||
final Sender tmp=sender; | ||
return tmp != null && tmp.running(); | ||
} | ||
|
||
|
||
protected class Sender implements Runnable { | ||
protected final Thread thread; | ||
protected volatile boolean running=true; | ||
|
||
public Sender(ThreadFactory f) { | ||
String name=name(); | ||
thread=f != null? f.newThread(this, name) : new Thread(this, name); | ||
} | ||
|
||
public Sender start() { | ||
running=true; | ||
thread.start(); | ||
return this; | ||
} | ||
|
||
public Sender stop() { | ||
running=false; | ||
Thread t=thread; | ||
if(t != null && t.isAlive()) | ||
t.interrupt(); | ||
return this; | ||
} | ||
|
||
public boolean running() { | ||
return running && isConnected(); | ||
} | ||
|
||
@Override public void run() { | ||
try { | ||
while(running()) { | ||
ByteArray data; | ||
try { | ||
data=queue.take(); | ||
} | ||
catch(InterruptedException iex) { | ||
continue; | ||
} | ||
|
||
// no synchronization needed as this thread is the only sender | ||
doSend(data.getArray(), data.getOffset(), data.getLength(), true); // flush | ||
} | ||
} | ||
catch(EOFException | SocketException ex) { | ||
; // regular use case when a peer closes its connection - we don't want to log this as exception | ||
} | ||
catch(Exception e) { | ||
//noinspection StatementWithEmptyBody | ||
if (e instanceof SSLException && e.getMessage().contains("Socket closed")) { | ||
; // regular use case when a peer closes its connection - we don't want to log this as exception | ||
} | ||
else if (e instanceof SSLHandshakeException && e.getCause() instanceof EOFException) { | ||
; // Ignore SSL handshakes closed early (usually liveness probes) | ||
} | ||
else { | ||
if(server.logDetails()) | ||
server.log.warn("failed sending message", e); | ||
else | ||
server.log.warn("failed sending message: " + e); | ||
} | ||
} | ||
finally { | ||
server.notifyConnectionClosed(TcpConnectionNonBlocking.this); | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters