1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * 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, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.net.util; 18 19 import static android.system.OsConstants.IPPROTO_ICMPV6; 20 import static android.system.OsConstants.IPPROTO_UDP; 21 22 import static com.android.net.module.util.NetworkStackConstants.ARP_HWTYPE_ETHER; 23 import static com.android.net.module.util.NetworkStackConstants.ARP_PAYLOAD_LEN; 24 import static com.android.net.module.util.NetworkStackConstants.ARP_REPLY; 25 import static com.android.net.module.util.NetworkStackConstants.ARP_REQUEST; 26 import static com.android.net.module.util.NetworkStackConstants.DHCP4_CLIENT_PORT; 27 import static com.android.net.module.util.NetworkStackConstants.ETHER_ADDR_LEN; 28 import static com.android.net.module.util.NetworkStackConstants.ETHER_DST_ADDR_OFFSET; 29 import static com.android.net.module.util.NetworkStackConstants.ETHER_HEADER_LEN; 30 import static com.android.net.module.util.NetworkStackConstants.ETHER_SRC_ADDR_OFFSET; 31 import static com.android.net.module.util.NetworkStackConstants.ETHER_TYPE_ARP; 32 import static com.android.net.module.util.NetworkStackConstants.ETHER_TYPE_IPV4; 33 import static com.android.net.module.util.NetworkStackConstants.ETHER_TYPE_IPV6; 34 import static com.android.net.module.util.NetworkStackConstants.ETHER_TYPE_OFFSET; 35 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_HEADER_MIN_LEN; 36 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ND_OPTION_LENGTH_SCALING_FACTOR; 37 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ND_OPTION_MIN_LENGTH; 38 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ND_OPTION_MTU; 39 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ND_OPTION_SLLA; 40 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ND_OPTION_TLLA; 41 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_NEIGHBOR_ADVERTISEMENT; 42 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_NEIGHBOR_SOLICITATION; 43 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ROUTER_ADVERTISEMENT; 44 import static com.android.net.module.util.NetworkStackConstants.ICMPV6_ROUTER_SOLICITATION; 45 import static com.android.net.module.util.NetworkStackConstants.IPV4_ADDR_LEN; 46 import static com.android.net.module.util.NetworkStackConstants.IPV4_DST_ADDR_OFFSET; 47 import static com.android.net.module.util.NetworkStackConstants.IPV4_FLAGS_OFFSET; 48 import static com.android.net.module.util.NetworkStackConstants.IPV4_FRAGMENT_MASK; 49 import static com.android.net.module.util.NetworkStackConstants.IPV4_HEADER_MIN_LEN; 50 import static com.android.net.module.util.NetworkStackConstants.IPV4_IHL_MASK; 51 import static com.android.net.module.util.NetworkStackConstants.IPV4_PROTOCOL_OFFSET; 52 import static com.android.net.module.util.NetworkStackConstants.IPV4_SRC_ADDR_OFFSET; 53 import static com.android.net.module.util.NetworkStackConstants.IPV6_ADDR_LEN; 54 import static com.android.net.module.util.NetworkStackConstants.IPV6_HEADER_LEN; 55 import static com.android.net.module.util.NetworkStackConstants.IPV6_PROTOCOL_OFFSET; 56 import static com.android.net.module.util.NetworkStackConstants.IPV6_SRC_ADDR_OFFSET; 57 import static com.android.net.module.util.NetworkStackConstants.UDP_HEADER_LEN; 58 59 import android.net.MacAddress; 60 import android.net.dhcp.DhcpPacket; 61 62 import java.net.InetAddress; 63 import java.net.UnknownHostException; 64 import java.nio.ByteBuffer; 65 import java.nio.ByteOrder; 66 import java.util.StringJoiner; 67 68 69 /** 70 * Critical connectivity packet summarizing class. 71 * 72 * Outputs short descriptions of ARP, DHCPv4, and IPv6 RS/RA/NS/NA packets. 73 * 74 * @hide 75 */ 76 public class ConnectivityPacketSummary { 77 private static final String TAG = ConnectivityPacketSummary.class.getSimpleName(); 78 79 private final byte[] mHwAddr; 80 private final byte[] mBytes; 81 private final int mLength; 82 private final ByteBuffer mPacket; 83 private final String mSummary; 84 85 /** 86 * Create a string summary of a received packet. 87 * @param hwaddr MacAddress of the interface sending/receiving the packet. 88 * @param buffer The packet bytes. Length is assumed to be the buffer length. 89 * @return A summary of the packet. 90 */ summarize(MacAddress hwaddr, byte[] buffer)91 public static String summarize(MacAddress hwaddr, byte[] buffer) { 92 return summarize(hwaddr, buffer, buffer.length); 93 } 94 95 // Methods called herein perform some but by no means all error checking. 96 // They may throw runtime exceptions on malformed packets. 97 98 /** 99 * Create a string summary of a received packet. 100 * @param macAddr MacAddress of the interface sending/receiving the packet. 101 * @param buffer The packet bytes. 102 * @param length Length of the packet. 103 * @return A summary of the packet. 104 */ summarize(MacAddress macAddr, byte[] buffer, int length)105 public static String summarize(MacAddress macAddr, byte[] buffer, int length) { 106 if ((macAddr == null) || (buffer == null)) return null; 107 length = Math.min(length, buffer.length); 108 return (new ConnectivityPacketSummary(macAddr, buffer, length)).toString(); 109 } 110 ConnectivityPacketSummary(MacAddress macAddr, byte[] buffer, int length)111 private ConnectivityPacketSummary(MacAddress macAddr, byte[] buffer, int length) { 112 mHwAddr = macAddr.toByteArray(); 113 mBytes = buffer; 114 mLength = Math.min(length, mBytes.length); 115 mPacket = ByteBuffer.wrap(mBytes, 0, mLength); 116 mPacket.order(ByteOrder.BIG_ENDIAN); 117 118 final StringJoiner sj = new StringJoiner(" "); 119 // TODO: support other link-layers, or even no link-layer header. 120 parseEther(sj); 121 mSummary = sj.toString(); 122 } 123 toString()124 public String toString() { 125 return mSummary; 126 } 127 parseEther(StringJoiner sj)128 private void parseEther(StringJoiner sj) { 129 if (mPacket.remaining() < ETHER_HEADER_LEN) { 130 sj.add("runt:").add(asString(mPacket.remaining())); 131 return; 132 } 133 134 mPacket.position(ETHER_SRC_ADDR_OFFSET); 135 final ByteBuffer srcMac = (ByteBuffer) mPacket.slice().limit(ETHER_ADDR_LEN); 136 sj.add(ByteBuffer.wrap(mHwAddr).equals(srcMac) ? "TX" : "RX"); 137 sj.add(getMacAddressString(srcMac)); 138 139 mPacket.position(ETHER_DST_ADDR_OFFSET); 140 final ByteBuffer dstMac = (ByteBuffer) mPacket.slice().limit(ETHER_ADDR_LEN); 141 sj.add(">").add(getMacAddressString(dstMac)); 142 143 mPacket.position(ETHER_TYPE_OFFSET); 144 final int etherType = asUint(mPacket.getShort()); 145 switch (etherType) { 146 case ETHER_TYPE_ARP: 147 sj.add("arp"); 148 parseARP(sj); 149 break; 150 case ETHER_TYPE_IPV4: 151 sj.add("ipv4"); 152 parseIPv4(sj); 153 break; 154 case ETHER_TYPE_IPV6: 155 sj.add("ipv6"); 156 parseIPv6(sj); 157 break; 158 default: 159 // Unknown ether type. 160 sj.add("ethtype").add(asString(etherType)); 161 break; 162 } 163 } 164 parseARP(StringJoiner sj)165 private void parseARP(StringJoiner sj) { 166 if (mPacket.remaining() < ARP_PAYLOAD_LEN) { 167 sj.add("runt:").add(asString(mPacket.remaining())); 168 return; 169 } 170 171 if (asUint(mPacket.getShort()) != ARP_HWTYPE_ETHER || 172 asUint(mPacket.getShort()) != ETHER_TYPE_IPV4 || 173 asUint(mPacket.get()) != ETHER_ADDR_LEN || 174 asUint(mPacket.get()) != IPV4_ADDR_LEN) { 175 sj.add("unexpected header"); 176 return; 177 } 178 179 final int opCode = asUint(mPacket.getShort()); 180 181 final String senderHwAddr = getMacAddressString(mPacket); 182 final String senderIPv4 = getIPv4AddressString(mPacket); 183 getMacAddressString(mPacket); // target hardware address, unused 184 final String targetIPv4 = getIPv4AddressString(mPacket); 185 186 if (opCode == ARP_REQUEST) { 187 sj.add("who-has").add(targetIPv4); 188 } else if (opCode == ARP_REPLY) { 189 sj.add("reply").add(senderIPv4).add(senderHwAddr); 190 } else { 191 sj.add("unknown opcode").add(asString(opCode)); 192 } 193 } 194 parseIPv4(StringJoiner sj)195 private void parseIPv4(StringJoiner sj) { 196 if (!mPacket.hasRemaining()) { 197 sj.add("runt"); 198 return; 199 } 200 201 final int startOfIpLayer = mPacket.position(); 202 final int ipv4HeaderLength = (mPacket.get(startOfIpLayer) & IPV4_IHL_MASK) * 4; 203 if (mPacket.remaining() < ipv4HeaderLength || 204 mPacket.remaining() < IPV4_HEADER_MIN_LEN) { 205 sj.add("runt:").add(asString(mPacket.remaining())); 206 return; 207 } 208 final int startOfTransportLayer = startOfIpLayer + ipv4HeaderLength; 209 210 mPacket.position(startOfIpLayer + IPV4_FLAGS_OFFSET); 211 final int flagsAndFragment = asUint(mPacket.getShort()); 212 final boolean isFragment = (flagsAndFragment & IPV4_FRAGMENT_MASK) != 0; 213 214 mPacket.position(startOfIpLayer + IPV4_PROTOCOL_OFFSET); 215 final int protocol = asUint(mPacket.get()); 216 217 mPacket.position(startOfIpLayer + IPV4_SRC_ADDR_OFFSET); 218 final String srcAddr = getIPv4AddressString(mPacket); 219 220 mPacket.position(startOfIpLayer + IPV4_DST_ADDR_OFFSET); 221 final String dstAddr = getIPv4AddressString(mPacket); 222 223 sj.add(srcAddr).add(">").add(dstAddr); 224 225 mPacket.position(startOfTransportLayer); 226 if (protocol == IPPROTO_UDP) { 227 sj.add("udp"); 228 if (isFragment) sj.add("fragment"); 229 else parseUDP(sj); 230 } else { 231 sj.add("proto").add(asString(protocol)); 232 if (isFragment) sj.add("fragment"); 233 } 234 } 235 parseIPv6(StringJoiner sj)236 private void parseIPv6(StringJoiner sj) { 237 if (mPacket.remaining() < IPV6_HEADER_LEN) { 238 sj.add("runt:").add(asString(mPacket.remaining())); 239 return; 240 } 241 242 final int startOfIpLayer = mPacket.position(); 243 244 mPacket.position(startOfIpLayer + IPV6_PROTOCOL_OFFSET); 245 final int protocol = asUint(mPacket.get()); 246 247 mPacket.position(startOfIpLayer + IPV6_SRC_ADDR_OFFSET); 248 final String srcAddr = getIPv6AddressString(mPacket); 249 final String dstAddr = getIPv6AddressString(mPacket); 250 251 sj.add(srcAddr).add(">").add(dstAddr); 252 253 mPacket.position(startOfIpLayer + IPV6_HEADER_LEN); 254 if (protocol == IPPROTO_ICMPV6) { 255 sj.add("icmp6"); 256 parseICMPv6(sj); 257 } else { 258 sj.add("proto").add(asString(protocol)); 259 } 260 } 261 parseICMPv6(StringJoiner sj)262 private void parseICMPv6(StringJoiner sj) { 263 if (mPacket.remaining() < ICMPV6_HEADER_MIN_LEN) { 264 sj.add("runt:").add(asString(mPacket.remaining())); 265 return; 266 } 267 268 final int icmp6Type = asUint(mPacket.get()); 269 final int icmp6Code = asUint(mPacket.get()); 270 mPacket.getShort(); // checksum, unused 271 272 switch (icmp6Type) { 273 case ICMPV6_ROUTER_SOLICITATION: 274 sj.add("rs"); 275 parseICMPv6RouterSolicitation(sj); 276 break; 277 case ICMPV6_ROUTER_ADVERTISEMENT: 278 sj.add("ra"); 279 parseICMPv6RouterAdvertisement(sj); 280 break; 281 case ICMPV6_NEIGHBOR_SOLICITATION: 282 sj.add("ns"); 283 parseICMPv6NeighborMessage(sj); 284 break; 285 case ICMPV6_NEIGHBOR_ADVERTISEMENT: 286 sj.add("na"); 287 parseICMPv6NeighborMessage(sj); 288 break; 289 default: 290 sj.add("type").add(asString(icmp6Type)); 291 sj.add("code").add(asString(icmp6Code)); 292 break; 293 } 294 } 295 parseICMPv6RouterSolicitation(StringJoiner sj)296 private void parseICMPv6RouterSolicitation(StringJoiner sj) { 297 final int RESERVED = 4; 298 if (mPacket.remaining() < RESERVED) { 299 sj.add("runt:").add(asString(mPacket.remaining())); 300 return; 301 } 302 303 mPacket.position(mPacket.position() + RESERVED); 304 parseICMPv6NeighborDiscoveryOptions(sj); 305 } 306 parseICMPv6RouterAdvertisement(StringJoiner sj)307 private void parseICMPv6RouterAdvertisement(StringJoiner sj) { 308 final int FLAGS_AND_TIMERS = 3 * 4; 309 if (mPacket.remaining() < FLAGS_AND_TIMERS) { 310 sj.add("runt:").add(asString(mPacket.remaining())); 311 return; 312 } 313 314 mPacket.position(mPacket.position() + FLAGS_AND_TIMERS); 315 parseICMPv6NeighborDiscoveryOptions(sj); 316 } 317 parseICMPv6NeighborMessage(StringJoiner sj)318 private void parseICMPv6NeighborMessage(StringJoiner sj) { 319 final int RESERVED = 4; 320 final int minReq = RESERVED + IPV6_ADDR_LEN; 321 if (mPacket.remaining() < minReq) { 322 sj.add("runt:").add(asString(mPacket.remaining())); 323 return; 324 } 325 326 mPacket.position(mPacket.position() + RESERVED); 327 sj.add(getIPv6AddressString(mPacket)); 328 parseICMPv6NeighborDiscoveryOptions(sj); 329 } 330 parseICMPv6NeighborDiscoveryOptions(StringJoiner sj)331 private void parseICMPv6NeighborDiscoveryOptions(StringJoiner sj) { 332 // All ND options are TLV, where T is one byte and L is one byte equal 333 // to the length of T + L + V in units of 8 octets. 334 while (mPacket.remaining() >= ICMPV6_ND_OPTION_MIN_LENGTH) { 335 final int ndType = asUint(mPacket.get()); 336 final int ndLength = asUint(mPacket.get()); 337 final int ndBytes = ndLength * ICMPV6_ND_OPTION_LENGTH_SCALING_FACTOR - 2; 338 if (ndBytes < 0 || ndBytes > mPacket.remaining()) { 339 sj.add("<malformed>"); 340 break; 341 } 342 final int position = mPacket.position(); 343 344 switch (ndType) { 345 case ICMPV6_ND_OPTION_SLLA: 346 sj.add("slla"); 347 sj.add(getMacAddressString(mPacket)); 348 break; 349 case ICMPV6_ND_OPTION_TLLA: 350 sj.add("tlla"); 351 sj.add(getMacAddressString(mPacket)); 352 break; 353 case ICMPV6_ND_OPTION_MTU: 354 sj.add("mtu"); 355 final short reserved = mPacket.getShort(); 356 sj.add(asString(mPacket.getInt())); 357 break; 358 default: 359 // Skip. 360 break; 361 } 362 363 mPacket.position(position + ndBytes); 364 } 365 } 366 parseUDP(StringJoiner sj)367 private void parseUDP(StringJoiner sj) { 368 if (mPacket.remaining() < UDP_HEADER_LEN) { 369 sj.add("runt:").add(asString(mPacket.remaining())); 370 return; 371 } 372 373 final int previous = mPacket.position(); 374 final int srcPort = asUint(mPacket.getShort()); 375 final int dstPort = asUint(mPacket.getShort()); 376 sj.add(asString(srcPort)).add(">").add(asString(dstPort)); 377 378 mPacket.position(previous + UDP_HEADER_LEN); 379 if (srcPort == DHCP4_CLIENT_PORT || dstPort == DHCP4_CLIENT_PORT) { 380 sj.add("dhcp4"); 381 parseDHCPv4(sj); 382 } 383 } 384 parseDHCPv4(StringJoiner sj)385 private void parseDHCPv4(StringJoiner sj) { 386 final DhcpPacket dhcpPacket; 387 try { 388 dhcpPacket = DhcpPacket.decodeFullPacket(mBytes, mLength, DhcpPacket.ENCAP_L2); 389 sj.add(dhcpPacket.toString()); 390 } catch (DhcpPacket.ParseException e) { 391 sj.add("parse error: " + e); 392 } 393 } 394 getIPv4AddressString(ByteBuffer ipv4)395 private static String getIPv4AddressString(ByteBuffer ipv4) { 396 return getIpAddressString(ipv4, IPV4_ADDR_LEN); 397 } 398 getIPv6AddressString(ByteBuffer ipv6)399 private static String getIPv6AddressString(ByteBuffer ipv6) { 400 return getIpAddressString(ipv6, IPV6_ADDR_LEN); 401 } 402 getIpAddressString(ByteBuffer ip, int byteLength)403 private static String getIpAddressString(ByteBuffer ip, int byteLength) { 404 if (ip == null || ip.remaining() < byteLength) return "invalid"; 405 406 byte[] bytes = new byte[byteLength]; 407 ip.get(bytes, 0, byteLength); 408 try { 409 InetAddress addr = InetAddress.getByAddress(bytes); 410 return addr.getHostAddress(); 411 } catch (UnknownHostException uhe) { 412 return "unknown"; 413 } 414 } 415 getMacAddressString(ByteBuffer mac)416 private static String getMacAddressString(ByteBuffer mac) { 417 if (mac == null || mac.remaining() < ETHER_ADDR_LEN) return "invalid"; 418 419 byte[] bytes = new byte[ETHER_ADDR_LEN]; 420 mac.get(bytes, 0, bytes.length); 421 Object[] printableBytes = new Object[bytes.length]; 422 int i = 0; 423 for (byte b : bytes) printableBytes[i++] = new Byte(b); 424 425 final String MAC48_FORMAT = "%02x:%02x:%02x:%02x:%02x:%02x"; 426 return String.format(MAC48_FORMAT, printableBytes); 427 } 428 429 /** 430 * Convenience method to convert an int to a String. 431 */ asString(int i)432 public static String asString(int i) { 433 return Integer.toString(i); 434 } 435 436 /** 437 * Convenience method to read a byte as an unsigned int. 438 */ asUint(byte b)439 public static int asUint(byte b) { 440 return (b & 0xff); 441 } 442 443 /** 444 * Convenience method to read a short as an unsigned int. 445 */ asUint(short s)446 public static int asUint(short s) { 447 return (s & 0xffff); 448 } 449 } 450