mirror of https://github.com/apache/activemq.git
Better selector synchonization to help resolve AMQ-2440
git-svn-id: https://svn.apache.org/repos/asf/activemq/trunk@916762 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
eac31e8417
commit
662324caac
|
@ -61,15 +61,25 @@ public final class SelectorManager {
|
||||||
public synchronized SelectorSelection register(SocketChannel socketChannel, Listener listener)
|
public synchronized SelectorSelection register(SocketChannel socketChannel, Listener listener)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
|
||||||
SelectorWorker worker = null;
|
SelectorSelection selection = null;
|
||||||
|
while( selection == null ) {
|
||||||
if (freeWorkers.size() > 0) {
|
if (freeWorkers.size() > 0) {
|
||||||
worker = freeWorkers.getFirst();
|
SelectorWorker worker = freeWorkers.getFirst();
|
||||||
|
if( worker.isReleased() ) {
|
||||||
|
freeWorkers.remove(worker);
|
||||||
} else {
|
} else {
|
||||||
worker = new SelectorWorker(this);
|
worker.retain();
|
||||||
freeWorkers.addFirst(worker);
|
selection = new SelectorSelection(worker, socketChannel, listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// Worker starts /w retain count of 1
|
||||||
|
SelectorWorker worker = new SelectorWorker(this);
|
||||||
|
freeWorkers.addFirst(worker);
|
||||||
|
selection = new SelectorSelection(worker, socketChannel, listener);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SelectorSelection selection = new SelectorSelection(worker, socketChannel, listener);
|
|
||||||
return selection;
|
return selection;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,10 +16,11 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.transport.nio;
|
package org.apache.activemq.transport.nio;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.nio.channels.CancelledKeyException;
|
||||||
import java.nio.channels.ClosedChannelException;
|
import java.nio.channels.ClosedChannelException;
|
||||||
import java.nio.channels.SelectionKey;
|
import java.nio.channels.SelectionKey;
|
||||||
import java.nio.channels.SocketChannel;
|
import java.nio.channels.SocketChannel;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
import org.apache.activemq.transport.nio.SelectorManager.Listener;
|
import org.apache.activemq.transport.nio.SelectorManager.Listener;
|
||||||
|
|
||||||
|
@ -29,53 +30,63 @@ import org.apache.activemq.transport.nio.SelectorManager.Listener;
|
||||||
public final class SelectorSelection {
|
public final class SelectorSelection {
|
||||||
|
|
||||||
private final SelectorWorker worker;
|
private final SelectorWorker worker;
|
||||||
private final SelectionKey key;
|
|
||||||
private final Listener listener;
|
private final Listener listener;
|
||||||
private int interest;
|
private int interest;
|
||||||
|
private SelectionKey key;
|
||||||
|
private AtomicBoolean closed = new AtomicBoolean();
|
||||||
|
|
||||||
public SelectorSelection(SelectorWorker worker, SocketChannel socketChannel, Listener listener) throws ClosedChannelException {
|
public SelectorSelection(final SelectorWorker worker, final SocketChannel socketChannel, Listener listener) throws ClosedChannelException {
|
||||||
this.worker = worker;
|
this.worker = worker;
|
||||||
this.listener = listener;
|
this.listener = listener;
|
||||||
|
worker.addIoTask(new Runnable() {
|
||||||
// Lock when mutating state of the selector
|
public void run() {
|
||||||
worker.lock();
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
this.key = socketChannel.register(worker.selector, 0, this);
|
SelectorSelection.this.key = socketChannel.register(worker.selector, 0, SelectorSelection.this);
|
||||||
worker.incrementUseCounter();
|
} catch (Exception e) {
|
||||||
} finally {
|
e.printStackTrace();
|
||||||
worker.unlock();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
public void setInterestOps(int ops) {
|
public void setInterestOps(int ops) {
|
||||||
interest = ops;
|
interest = ops;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void enable() {
|
public void enable() {
|
||||||
|
worker.addIoTask(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
key.interestOps(interest);
|
key.interestOps(interest);
|
||||||
worker.selector.wakeup();
|
} catch (CancelledKeyException e) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public void disable() {
|
public void disable() {
|
||||||
if (key.isValid()) {
|
worker.addIoTask(new Runnable() {
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
key.interestOps(0);
|
key.interestOps(0);
|
||||||
|
} catch (CancelledKeyException e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
public void close() {
|
public void close() {
|
||||||
worker.decrementUseCounter();
|
// guard against multiple closes.
|
||||||
|
if( closed.compareAndSet(false, true) ) {
|
||||||
// Lock when mutating state of the selector
|
worker.addIoTask(new Runnable() {
|
||||||
worker.lock();
|
public void run() {
|
||||||
try {
|
try {
|
||||||
key.cancel();
|
key.cancel();
|
||||||
if (!worker.isRunning()) {
|
} catch (CancelledKeyException e) {
|
||||||
worker.close();
|
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
worker.release();
|
||||||
} finally {
|
}
|
||||||
worker.unlock();
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,14 +17,12 @@
|
||||||
package org.apache.activemq.transport.nio;
|
package org.apache.activemq.transport.nio;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.channels.ClosedSelectorException;
|
|
||||||
import java.nio.channels.SelectionKey;
|
import java.nio.channels.SelectionKey;
|
||||||
import java.nio.channels.Selector;
|
import java.nio.channels.Selector;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.concurrent.locks.ReadWriteLock;
|
|
||||||
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
|
||||||
|
|
||||||
public class SelectorWorker implements Runnable {
|
public class SelectorWorker implements Runnable {
|
||||||
|
|
||||||
|
@ -33,56 +31,69 @@ public class SelectorWorker implements Runnable {
|
||||||
final SelectorManager manager;
|
final SelectorManager manager;
|
||||||
final Selector selector;
|
final Selector selector;
|
||||||
final int id = NEXT_ID.getAndIncrement();
|
final int id = NEXT_ID.getAndIncrement();
|
||||||
final AtomicInteger useCounter = new AtomicInteger();
|
|
||||||
private final int maxChannelsPerWorker;
|
private final int maxChannelsPerWorker;
|
||||||
private final ReadWriteLock selectorLock = new ReentrantReadWriteLock();
|
|
||||||
|
final AtomicInteger retainCounter = new AtomicInteger(1);
|
||||||
|
private final ConcurrentLinkedQueue<Runnable> ioTasks = new ConcurrentLinkedQueue<Runnable>();
|
||||||
|
|
||||||
public SelectorWorker(SelectorManager manager) throws IOException {
|
public SelectorWorker(SelectorManager manager) throws IOException {
|
||||||
this.manager = manager;
|
this.manager = manager;
|
||||||
selector = Selector.open();
|
selector = Selector.open();
|
||||||
maxChannelsPerWorker = manager.getMaxChannelsPerWorker();
|
maxChannelsPerWorker = manager.getMaxChannelsPerWorker();
|
||||||
|
manager.getSelectorExecutor().execute(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void incrementUseCounter() {
|
void retain() {
|
||||||
int use = useCounter.getAndIncrement();
|
if (retainCounter.incrementAndGet() == maxChannelsPerWorker) {
|
||||||
if (use == 0) {
|
|
||||||
manager.getSelectorExecutor().execute(this);
|
|
||||||
} else if (use + 1 == maxChannelsPerWorker) {
|
|
||||||
manager.onWorkerFullEvent(this);
|
manager.onWorkerFullEvent(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void decrementUseCounter() {
|
void release() {
|
||||||
int use = useCounter.getAndDecrement();
|
int use = retainCounter.decrementAndGet();
|
||||||
if (use == 1) {
|
if (use == 0) {
|
||||||
manager.onWorkerEmptyEvent(this);
|
manager.onWorkerEmptyEvent(this);
|
||||||
} else if (use == maxChannelsPerWorker) {
|
} else if (use < maxChannelsPerWorker) {
|
||||||
manager.onWorkerNotFullEvent(this);
|
manager.onWorkerNotFullEvent(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean isRunning() {
|
boolean isReleased() {
|
||||||
return useCounter.get() != 0;
|
return retainCounter.get()==0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void addIoTask(Runnable work) {
|
||||||
|
ioTasks.add(work);
|
||||||
|
selector.wakeup();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void processIoTasks() {
|
||||||
|
Runnable task;
|
||||||
|
while( (task= ioTasks.poll()) !=null ) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
} catch (Throwable e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public void run() {
|
public void run() {
|
||||||
|
|
||||||
String origName = Thread.currentThread().getName();
|
String origName = Thread.currentThread().getName();
|
||||||
try {
|
try {
|
||||||
Thread.currentThread().setName("Selector Worker: " + id);
|
Thread.currentThread().setName("Selector Worker: " + id);
|
||||||
while (isRunning()) {
|
while (!isReleased()) {
|
||||||
|
|
||||||
lockBarrier();
|
|
||||||
|
|
||||||
|
processIoTasks();
|
||||||
int count = selector.select(10);
|
int count = selector.select(10);
|
||||||
if (count == 0) {
|
if (count == 0) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isRunning()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get a java.util.Set containing the SelectionKey objects
|
// Get a java.util.Set containing the SelectionKey objects
|
||||||
// for all channels that are ready for I/O.
|
// for all channels that are ready for I/O.
|
||||||
Set keys = selector.selectedKeys();
|
Set keys = selector.selectedKeys();
|
||||||
|
@ -93,7 +104,9 @@ public class SelectorWorker implements Runnable {
|
||||||
|
|
||||||
final SelectorSelection s = (SelectorSelection)key.attachment();
|
final SelectorSelection s = (SelectorSelection)key.attachment();
|
||||||
try {
|
try {
|
||||||
s.disable();
|
if( key.isValid() ) {
|
||||||
|
key.interestOps(0);
|
||||||
|
}
|
||||||
|
|
||||||
// Kick off another thread to find newly selected keys
|
// Kick off another thread to find newly selected keys
|
||||||
// while we process the
|
// while we process the
|
||||||
|
@ -116,13 +129,8 @@ public class SelectorWorker implements Runnable {
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
} catch (ClosedSelectorException cse) {
|
|
||||||
// Don't accept any more selections
|
|
||||||
manager.onWorkerEmptyEvent(this);
|
|
||||||
} catch (IOException e) {
|
|
||||||
// Don't accept any more selections
|
|
||||||
manager.onWorkerEmptyEvent(this);
|
|
||||||
|
|
||||||
|
} catch (Throwable e) {
|
||||||
// Notify all the selections that the error occurred.
|
// Notify all the selections that the error occurred.
|
||||||
Set keys = selector.keys();
|
Set keys = selector.keys();
|
||||||
for (Iterator i = keys.iterator(); i.hasNext();) {
|
for (Iterator i = keys.iterator(); i.hasNext();) {
|
||||||
|
@ -132,25 +140,13 @@ public class SelectorWorker implements Runnable {
|
||||||
}
|
}
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
|
try {
|
||||||
|
manager.onWorkerEmptyEvent(this);
|
||||||
|
selector.close();
|
||||||
|
} catch (IOException ignore) {
|
||||||
|
}
|
||||||
Thread.currentThread().setName(origName);
|
Thread.currentThread().setName(origName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void lockBarrier() {
|
|
||||||
selectorLock.writeLock().lock();
|
|
||||||
selectorLock.writeLock().unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void lock() {
|
|
||||||
selectorLock.readLock().lock();
|
|
||||||
selector.wakeup();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void unlock() {
|
|
||||||
selectorLock.readLock().unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close() throws IOException {
|
|
||||||
selector.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,10 +25,6 @@ import java.io.OutputStream;
|
||||||
import java.net.Socket;
|
import java.net.Socket;
|
||||||
import java.net.UnknownHostException;
|
import java.net.UnknownHostException;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.regex.Matcher;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
import org.apache.activemq.transport.stomp.Stomp.Headers.Subscribe;
|
|
||||||
|
|
||||||
public class StompConnection {
|
public class StompConnection {
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue