/transport-native-epoll/src/main/java/io/netty/channel/epoll/NativeDatagramPacketArray.java

http://github.com/netty/netty · Java · 178 lines · 117 code · 26 blank · 35 comment · 19 complexity · a8f6b75799c1e06a207f5b326419d85d MD5 · raw file

  1. /*
  2. * Copyright 2014 The Netty Project
  3. *
  4. * The Netty Project licenses this file to you under the Apache License,
  5. * version 2.0 (the "License"); you may not use this file except in compliance
  6. * with the License. You may obtain a copy of the License at:
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations
  14. * under the License.
  15. */
  16. package io.netty.channel.epoll;
  17. import io.netty.buffer.ByteBuf;
  18. import io.netty.channel.ChannelOutboundBuffer;
  19. import io.netty.channel.ChannelOutboundBuffer.MessageProcessor;
  20. import io.netty.channel.socket.DatagramPacket;
  21. import io.netty.channel.unix.IovArray;
  22. import io.netty.channel.unix.Limits;
  23. import java.net.Inet6Address;
  24. import java.net.InetAddress;
  25. import java.net.InetSocketAddress;
  26. import java.net.UnknownHostException;
  27. import static io.netty.channel.unix.Limits.UIO_MAX_IOV;
  28. import static io.netty.channel.unix.NativeInetAddress.copyIpv4MappedIpv6Address;
  29. /**
  30. * Support <a href="http://linux.die.net/man/2/sendmmsg">sendmmsg(...)</a> on linux with GLIBC 2.14+
  31. */
  32. final class NativeDatagramPacketArray {
  33. // Use UIO_MAX_IOV as this is the maximum number we can write with one sendmmsg(...) call.
  34. private final NativeDatagramPacket[] packets = new NativeDatagramPacket[UIO_MAX_IOV];
  35. // We share one IovArray for all NativeDatagramPackets to reduce memory overhead. This will allow us to write
  36. // up to IOV_MAX iovec across all messages in one sendmmsg(...) call.
  37. private final IovArray iovArray = new IovArray();
  38. // temporary array to copy the ipv4 part of ipv6-mapped-ipv4 addresses and then create a Inet4Address out of it.
  39. private final byte[] ipv4Bytes = new byte[4];
  40. private final MyMessageProcessor processor = new MyMessageProcessor();
  41. private int count;
  42. NativeDatagramPacketArray() {
  43. for (int i = 0; i < packets.length; i++) {
  44. packets[i] = new NativeDatagramPacket();
  45. }
  46. }
  47. boolean addWritable(ByteBuf buf, int index, int len) {
  48. return add0(buf, index, len, null);
  49. }
  50. private boolean add0(ByteBuf buf, int index, int len, InetSocketAddress recipient) {
  51. if (count == packets.length) {
  52. // We already filled up to UIO_MAX_IOV messages. This is the max allowed per
  53. // recvmmsg(...) / sendmmsg(...) call, we will try again later.
  54. return false;
  55. }
  56. if (len == 0) {
  57. return true;
  58. }
  59. int offset = iovArray.count();
  60. if (offset == Limits.IOV_MAX || !iovArray.add(buf, index, len)) {
  61. // Not enough space to hold the whole content, we will try again later.
  62. return false;
  63. }
  64. NativeDatagramPacket p = packets[count];
  65. p.init(iovArray.memoryAddress(offset), iovArray.count() - offset, recipient);
  66. count++;
  67. return true;
  68. }
  69. void add(ChannelOutboundBuffer buffer, boolean connected) throws Exception {
  70. processor.connected = connected;
  71. buffer.forEachFlushedMessage(processor);
  72. }
  73. /**
  74. * Returns the count
  75. */
  76. int count() {
  77. return count;
  78. }
  79. /**
  80. * Returns an array with {@link #count()} {@link NativeDatagramPacket}s filled.
  81. */
  82. NativeDatagramPacket[] packets() {
  83. return packets;
  84. }
  85. void clear() {
  86. this.count = 0;
  87. this.iovArray.clear();
  88. }
  89. void release() {
  90. iovArray.release();
  91. }
  92. private final class MyMessageProcessor implements MessageProcessor {
  93. private boolean connected;
  94. @Override
  95. public boolean processMessage(Object msg) {
  96. if (msg instanceof DatagramPacket) {
  97. DatagramPacket packet = (DatagramPacket) msg;
  98. ByteBuf buf = packet.content();
  99. return add0(buf, buf.readerIndex(), buf.readableBytes(), packet.recipient());
  100. }
  101. if (msg instanceof ByteBuf && connected) {
  102. ByteBuf buf = (ByteBuf) msg;
  103. return add0(buf, buf.readerIndex(), buf.readableBytes(), null);
  104. }
  105. return false;
  106. }
  107. }
  108. /**
  109. * Used to pass needed data to JNI.
  110. */
  111. @SuppressWarnings("unused")
  112. final class NativeDatagramPacket {
  113. // This is the actual struct iovec*
  114. private long memoryAddress;
  115. private int count;
  116. private final byte[] addr = new byte[16];
  117. private int addrLen;
  118. private int scopeId;
  119. private int port;
  120. private void init(long memoryAddress, int count, InetSocketAddress recipient) {
  121. this.memoryAddress = memoryAddress;
  122. this.count = count;
  123. if (recipient == null) {
  124. this.scopeId = 0;
  125. this.port = 0;
  126. this.addrLen = 0;
  127. } else {
  128. InetAddress address = recipient.getAddress();
  129. if (address instanceof Inet6Address) {
  130. System.arraycopy(address.getAddress(), 0, addr, 0, addr.length);
  131. scopeId = ((Inet6Address) address).getScopeId();
  132. } else {
  133. copyIpv4MappedIpv6Address(address.getAddress(), addr);
  134. scopeId = 0;
  135. }
  136. addrLen = addr.length;
  137. port = recipient.getPort();
  138. }
  139. }
  140. DatagramPacket newDatagramPacket(ByteBuf buffer, InetSocketAddress localAddress) throws UnknownHostException {
  141. final InetAddress address;
  142. if (addrLen == ipv4Bytes.length) {
  143. System.arraycopy(addr, 0, ipv4Bytes, 0, addrLen);
  144. address = InetAddress.getByAddress(ipv4Bytes);
  145. } else {
  146. address = Inet6Address.getByAddress(null, addr, scopeId);
  147. }
  148. return new DatagramPacket(buffer.writerIndex(count),
  149. localAddress, new InetSocketAddress(address, port));
  150. }
  151. }
  152. }