/src/lsr/paxos/network/NioNetwork.java

http://github.com/JPaxos/JPaxos · Java · 199 lines · 159 code · 37 blank · 3 comment · 21 complexity · 0da752153636badff701d4369affa054 MD5 · raw file

  1. package lsr.paxos.network;
  2. import static lsr.common.ProcessDescriptor.processDescriptor;
  3. import java.io.IOException;
  4. import java.net.InetAddress;
  5. import java.net.InetSocketAddress;
  6. import java.net.SocketException;
  7. import java.nio.ByteBuffer;
  8. import java.nio.channels.ClosedChannelException;
  9. import java.nio.channels.SelectionKey;
  10. import java.nio.channels.Selector;
  11. import java.nio.channels.ServerSocketChannel;
  12. import java.nio.channels.SocketChannel;
  13. import java.nio.channels.spi.SelectorProvider;
  14. import java.util.BitSet;
  15. import java.util.HashMap;
  16. import java.util.Iterator;
  17. import lsr.paxos.messages.Message;
  18. import org.slf4j.Logger;
  19. import org.slf4j.LoggerFactory;
  20. public class NioNetwork extends Network implements Runnable {
  21. private Selector selector;
  22. // input, output
  23. NioConnection[][] connections;
  24. private HashMap<SocketChannel, ByteBuffer> tmpBuffers = new HashMap<SocketChannel, ByteBuffer>();
  25. public NioNetwork()
  26. throws IOException {
  27. selector = SelectorProvider.provider().openSelector();
  28. ServerSocketChannel serverChannel = ServerSocketChannel.open();
  29. serverChannel.configureBlocking(false);
  30. serverChannel.socket().bind(new InetSocketAddress((InetAddress) null,
  31. processDescriptor.getLocalProcess().getReplicaPort()));
  32. serverChannel.register(selector, SelectionKey.OP_ACCEPT);
  33. // input, output
  34. connections = new NioConnection[processDescriptor.numReplicas][2];
  35. // for (int i = localId + 1; i < processDescriptor.numReplicas; i++) {
  36. for (int i = 0; i < processDescriptor.numReplicas; i++) {
  37. if (i != localId)
  38. {
  39. connections[i][1] = new NioOutputConnection(this,
  40. processDescriptor.config.getProcess(i), null);
  41. if (i > localId)
  42. connections[i][1].start();
  43. }
  44. }
  45. }
  46. @Override
  47. public void start() {
  48. Thread t = new Thread(this);
  49. t.setDaemon(true);
  50. t.start();
  51. }
  52. @Override
  53. public void run() {
  54. while (true) {
  55. try {
  56. selector.select();
  57. Iterator<SelectionKey> selectedKeys = this.selector
  58. .selectedKeys().iterator();
  59. while (selectedKeys.hasNext()) {
  60. SelectionKey key = (SelectionKey) selectedKeys.next();
  61. selectedKeys.remove();
  62. if (!key.isValid())
  63. continue;
  64. if (key.isAcceptable())
  65. accept(key);
  66. if (key.isReadable())
  67. read(key);
  68. }
  69. } catch (ClosedChannelException e) {
  70. logger.debug("other process closed a channel when attepting to connect here");
  71. e.printStackTrace();
  72. } catch (IOException e) {
  73. e.printStackTrace();
  74. }
  75. }
  76. }
  77. protected void accept(SelectionKey key) throws IOException,
  78. ClosedChannelException {
  79. ServerSocketChannel serverSocketChannel = (ServerSocketChannel) key
  80. .channel();
  81. SocketChannel channel = serverSocketChannel.accept();
  82. configureChannel(channel);
  83. tmpBuffers.put(channel, ByteBuffer.allocate(8));
  84. channel.register(selector, SelectionKey.OP_READ);
  85. logger.trace("socket accept: {}", channel);
  86. }
  87. protected void configureChannel(SocketChannel channel) throws IOException,
  88. SocketException {
  89. channel.configureBlocking(false);
  90. channel.socket().setTcpNoDelay(true);
  91. channel.socket().setSendBufferSize(NioConnection.TCP_BUFFER_SIZE);
  92. channel.socket().setReceiveBufferSize(NioConnection.TCP_BUFFER_SIZE);
  93. }
  94. private void read(SelectionKey key) throws IOException {
  95. SocketChannel channel = (SocketChannel) key.channel();
  96. ByteBuffer buffer = tmpBuffers.get(channel);
  97. if (logger.isTraceEnabled()) {
  98. logger.trace("socket read: {} {}", channel, (tmpBuffers.get(channel) != null));
  99. }
  100. int numRead = 0;
  101. try {
  102. numRead = channel.read(buffer);
  103. } catch (IOException e) {
  104. logger.trace("client disconnected");
  105. key.cancel();
  106. tmpBuffers.remove(channel);
  107. return;
  108. }
  109. if (numRead == -1) {
  110. logger.trace("client disconnected");
  111. key.cancel();
  112. tmpBuffers.remove(channel);
  113. return;
  114. }
  115. if (buffer.position() != buffer.capacity())
  116. return;
  117. buffer.flip();
  118. buffer.getInt();
  119. int senderId = buffer.getInt();
  120. key.cancel();
  121. tmpBuffers.remove(channel);
  122. if (senderId >= processDescriptor.numReplicas)
  123. {
  124. logger.error("Invalid senderId: {}", senderId);
  125. return;
  126. }
  127. NioInputConnection inputConnection = new NioInputConnection(this,
  128. processDescriptor.config.getProcess(senderId),
  129. channel);
  130. connections[senderId][0] = inputConnection;
  131. inputConnection.start();
  132. NioOutputConnection outputConnection = new NioOutputConnection(this,
  133. processDescriptor.config.getProcess(senderId),
  134. channel);
  135. connections[senderId][1] = outputConnection;
  136. outputConnection.start();
  137. logger.debug("input connection established with: {}", senderId);
  138. }
  139. @Override
  140. protected void send(Message message, int destination) {
  141. boolean sent = ((NioOutputConnection) connections[destination][1])
  142. .send(message.toByteArray());
  143. if (logger.isTraceEnabled()) {
  144. logger.trace("send message to: {} - {}", destination, (sent == true ? "submitted"
  145. : "rejected"));
  146. }
  147. }
  148. @Override
  149. protected void send(Message message, BitSet destinations) {
  150. for (int i = destinations.nextSetBit(0); i >= 0; i = destinations
  151. .nextSetBit(i + 1)) {
  152. send(message, i);
  153. }
  154. }
  155. protected void removeConnection(int senderId, int receiverId) {
  156. logger.trace("connection closed with: {} --> ", senderId, receiverId);
  157. if (receiverId == localId)
  158. ((NioOutputConnection) connections[senderId][1])
  159. .notifyAboutInputDisconnected();
  160. }
  161. private final static Logger logger = LoggerFactory.getLogger(Network.class);
  162. }