zmq.Mailbox Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jeromq Show documentation
Show all versions of jeromq Show documentation
Pure Java implementation of libzmq
/*
Copyright (c) 2007-2014 Contributors as noted in the AUTHORS file
This file is part of 0MQ.
0MQ is free software; you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
0MQ is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see .
*/
package zmq;
import java.io.Closeable;
import java.io.IOException;
import java.nio.channels.SelectableChannel;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class Mailbox
implements Closeable
{
// The pipe to store actual commands.
private final YPipe cpipe;
// Signaler to pass signals from writer thread to reader thread.
private final Signaler signaler;
// There's only one thread receiving from the mailbox, but there
// is arbitrary number of threads sending. Given that ypipe requires
// synchronised access on both of its endpoints, we have to synchronise
// the sending side.
private final Lock sync;
// True if the underlying pipe is active, ie. when we are allowed to
// read commands from it.
private boolean active;
// mailbox name, for better debugging
private final String name;
public Mailbox(String name)
{
cpipe = new YPipe(Config.COMMAND_PIPE_GRANULARITY.getValue());
sync = new ReentrantLock();
signaler = new Signaler();
// Get the pipe into passive state. That way, if the users starts by
// polling on the associated file descriptor it will get woken up when
// new command is posted.
Command cmd = cpipe.read();
assert (cmd == null);
active = false;
this.name = name;
}
public SelectableChannel getFd()
{
return signaler.getFd();
}
public void send(final Command cmd)
{
boolean ok = false;
sync.lock();
try {
cpipe.write(cmd, false);
ok = cpipe.flush();
}
finally {
sync.unlock();
}
if (!ok) {
signaler.send();
}
}
public Command recv(long timeout)
{
Command cmd = null;
// Try to get the command straight away.
if (active) {
cmd = cpipe.read();
if (cmd != null) {
return cmd;
}
// If there are no more commands available, switch into passive state.
active = false;
signaler.recv();
}
// Wait for signal from the command sender.
boolean rc = signaler.waitEvent(timeout);
if (!rc) {
return null;
}
// We've got the signal. Now we can switch into active state.
active = true;
// Get a command.
cmd = cpipe.read();
assert (cmd != null);
return cmd;
}
@Override
public void close() throws IOException
{
// TODO: Retrieve and deallocate commands inside the cpipe.
// Work around problem that other threads might still be in our
// send() method, by waiting on the mutex before disappearing.
sync.lock();
sync.unlock();
signaler.close();
}
@Override
public String toString()
{
return super.toString() + "[" + name + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy