/L2EmuProject-Commons/src/main/java/net/l2emuproject/network/mmocore/MMOConnection.java

http://l2emu.googlecode.com/ · Java · 308 lines · 230 code · 52 blank · 26 comment · 18 complexity · 83585b1e8966079f4eae615a757bce99 MD5 · raw file

  1. /*
  2. * This program is free software: you can redistribute it and/or modify it under
  3. * the terms of the GNU General Public License as published by the Free Software
  4. * Foundation, either version 3 of the License, or (at your option) any later
  5. * version.
  6. *
  7. * This program is distributed in the hope that it will be useful, but WITHOUT
  8. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  9. * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
  10. * details.
  11. *
  12. * You should have received a copy of the GNU General Public License along with
  13. * this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. package net.l2emuproject.network.mmocore;
  16. import java.net.InetAddress;
  17. import java.net.Socket;
  18. import java.net.UnknownHostException;
  19. import java.nio.ByteBuffer;
  20. import java.nio.channels.CancelledKeyException;
  21. import java.nio.channels.ClosedChannelException;
  22. import java.nio.channels.ReadableByteChannel;
  23. import java.nio.channels.SelectionKey;
  24. import java.nio.channels.SocketChannel;
  25. import java.nio.channels.WritableByteChannel;
  26. import javolution.util.FastList;
  27. /**
  28. * @author KenM
  29. */
  30. public abstract class MMOConnection<T extends MMOConnection<T, RP, SP>, RP extends ReceivablePacket<T, RP, SP>, SP extends SendablePacket<T, RP, SP>>
  31. {
  32. private final SelectorThread<T, RP, SP> _selectorThread;
  33. private final ReadWriteThread<T, RP, SP> _readWriteThread;
  34. private final Socket _socket;
  35. private InetAddress _inetAddress;
  36. private String _hostAddress;
  37. private FastList<SP> _sendQueue;
  38. private final SelectionKey _selectionKey;
  39. private ByteBuffer _readBuffer;
  40. private ByteBuffer _primaryWriteBuffer;
  41. private ByteBuffer _secondaryWriteBuffer;
  42. private long _closeTimeout = -1;
  43. protected MMOConnection(SelectorThread<T, RP, SP> selectorThread, SocketChannel socketChannel)
  44. throws ClosedChannelException
  45. {
  46. _selectorThread = selectorThread;
  47. _readWriteThread = getSelectorThread().getReadWriteThread();
  48. _socket = socketChannel.socket();
  49. _inetAddress = _socket.getInetAddress();
  50. _hostAddress = _inetAddress.getHostAddress();
  51. _selectionKey = socketChannel.register(getReadWriteThread().getSelector(), SelectionKey.OP_READ);
  52. _selectionKey.attach(this);
  53. }
  54. public synchronized void sendPacket(SP sp)
  55. {
  56. if (isClosed())
  57. return;
  58. try
  59. {
  60. getSelectionKey().interestOps(getSelectionKey().interestOps() | SelectionKey.OP_WRITE);
  61. getSendQueue2().addLast(sp);
  62. }
  63. catch (CancelledKeyException e)
  64. {
  65. // ignore
  66. }
  67. }
  68. final SelectorThread<T, RP, SP> getSelectorThread()
  69. {
  70. return _selectorThread;
  71. }
  72. private ReadWriteThread<T, RP, SP> getReadWriteThread()
  73. {
  74. return _readWriteThread;
  75. }
  76. final SelectionKey getSelectionKey()
  77. {
  78. return _selectionKey;
  79. }
  80. final void disableReadInterest()
  81. {
  82. try
  83. {
  84. getSelectionKey().interestOps(getSelectionKey().interestOps() & ~SelectionKey.OP_READ);
  85. }
  86. catch (CancelledKeyException e)
  87. {
  88. // ignore
  89. }
  90. }
  91. final void disableWriteInterest()
  92. {
  93. try
  94. {
  95. getSelectionKey().interestOps(getSelectionKey().interestOps() & ~SelectionKey.OP_WRITE);
  96. }
  97. catch (CancelledKeyException e)
  98. {
  99. // ignore
  100. }
  101. }
  102. final Socket getSocket()
  103. {
  104. return _socket;
  105. }
  106. public final InetAddress getInetAddress()
  107. {
  108. return _inetAddress;
  109. }
  110. public final String getHostAddress()
  111. {
  112. return _hostAddress;
  113. }
  114. public final void setHostAddress(String hostAddress)
  115. {
  116. try
  117. {
  118. _inetAddress = InetAddress.getByName(hostAddress);
  119. _hostAddress = _inetAddress.getHostAddress();
  120. }
  121. catch (UnknownHostException e)
  122. {
  123. SelectorThread._log.warn("", e);
  124. }
  125. }
  126. final WritableByteChannel getWritableChannel()
  127. {
  128. return _socket.getChannel();
  129. }
  130. final ReadableByteChannel getReadableByteChannel()
  131. {
  132. return _socket.getChannel();
  133. }
  134. final FastList<SP> getSendQueue2()
  135. {
  136. if (_sendQueue == null)
  137. _sendQueue = new FastList<SP>();
  138. return _sendQueue;
  139. }
  140. final void createWriteBuffer(ByteBuffer buf)
  141. {
  142. if (_primaryWriteBuffer == null)
  143. {
  144. // APPENDING FOR NULL
  145. _primaryWriteBuffer = getReadWriteThread().getPooledBuffer();
  146. _primaryWriteBuffer.put(buf);
  147. }
  148. else
  149. {
  150. // PREPENDING ON EXISTING
  151. ByteBuffer temp = getReadWriteThread().getPooledBuffer();
  152. temp.put(buf);
  153. int remaining = temp.remaining();
  154. _primaryWriteBuffer.flip();
  155. int limit = _primaryWriteBuffer.limit();
  156. if (remaining >= _primaryWriteBuffer.remaining())
  157. {
  158. temp.put(_primaryWriteBuffer);
  159. getReadWriteThread().recycleBuffer(_primaryWriteBuffer);
  160. _primaryWriteBuffer = temp;
  161. }
  162. else
  163. {
  164. _primaryWriteBuffer.limit(remaining);
  165. temp.put(_primaryWriteBuffer);
  166. _primaryWriteBuffer.limit(limit);
  167. _primaryWriteBuffer.compact();
  168. _secondaryWriteBuffer = _primaryWriteBuffer;
  169. _primaryWriteBuffer = temp;
  170. }
  171. }
  172. }
  173. final boolean hasPendingWriteBuffer()
  174. {
  175. return _primaryWriteBuffer != null;
  176. }
  177. final void movePendingWriteBufferTo(ByteBuffer dest)
  178. {
  179. _primaryWriteBuffer.flip();
  180. dest.put(_primaryWriteBuffer);
  181. getReadWriteThread().recycleBuffer(_primaryWriteBuffer);
  182. _primaryWriteBuffer = _secondaryWriteBuffer;
  183. _secondaryWriteBuffer = null;
  184. }
  185. final void setReadBuffer(ByteBuffer buf)
  186. {
  187. _readBuffer = buf;
  188. }
  189. final ByteBuffer getReadBuffer()
  190. {
  191. return _readBuffer;
  192. }
  193. final boolean isClosed()
  194. {
  195. return _closeTimeout != -1;
  196. }
  197. final boolean closeTimeouted()
  198. {
  199. return System.currentTimeMillis() > _closeTimeout;
  200. }
  201. @SuppressWarnings("unchecked")
  202. public synchronized void closeNow()
  203. {
  204. if (isClosed())
  205. return;
  206. getSendQueue2().clear();
  207. sendPacket(getDefaultClosePacket());
  208. // close the client as soon as possible
  209. // for a normal connection 100 msec should be enough to write the pending packets
  210. _closeTimeout = System.currentTimeMillis() + 100;
  211. disableReadInterest();
  212. getReadWriteThread().closeConnection((T)this);
  213. }
  214. @SuppressWarnings("unchecked")
  215. public synchronized void close(SP sp)
  216. {
  217. if (isClosed())
  218. return;
  219. getSendQueue2().clear();
  220. sendPacket(sp);
  221. // let the client have the chance to get all of the packets
  222. // even for a connection with issues 10'000 msec should be enough to write the pending packets
  223. _closeTimeout = System.currentTimeMillis() + 10000;
  224. disableReadInterest();
  225. getReadWriteThread().closeConnection((T)this);
  226. }
  227. final void releaseBuffers()
  228. {
  229. if (_primaryWriteBuffer != null)
  230. {
  231. getReadWriteThread().recycleBuffer(_primaryWriteBuffer);
  232. _primaryWriteBuffer = null;
  233. if (_secondaryWriteBuffer != null)
  234. {
  235. getReadWriteThread().recycleBuffer(_secondaryWriteBuffer);
  236. _secondaryWriteBuffer = null;
  237. }
  238. }
  239. if (_readBuffer != null)
  240. {
  241. getReadWriteThread().recycleBuffer(_readBuffer);
  242. _readBuffer = null;
  243. }
  244. }
  245. protected abstract void onDisconnection();
  246. protected abstract void onForcedDisconnection();
  247. protected abstract boolean decrypt(ByteBuffer buf, int size);
  248. protected abstract boolean encrypt(ByteBuffer buf, int size);
  249. protected abstract SP getDefaultClosePacket();
  250. protected abstract String getUID();
  251. final String getValidUID()
  252. {
  253. final String UID = getUID();
  254. return UID == null || UID.isEmpty() ? getHostAddress() : UID;
  255. }
  256. }