1 /* 2 * Copyright (C) 2020 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 com.android.net.module.util; 18 19 import java.net.Inet4Address; 20 import java.net.InetAddress; 21 import java.net.UnknownHostException; 22 23 /** 24 * Collection of utilities to work with IPv4 addresses. 25 * @hide 26 */ 27 public class Inet4AddressUtils { 28 29 /** 30 * Convert a IPv4 address from an integer to an InetAddress (0x04030201 -> 1.2.3.4) 31 * 32 * <p>This method uses the higher-order int bytes as the lower-order IPv4 address bytes, 33 * which is an unusual convention. Consider {@link #intToInet4AddressHTH(int)} instead. 34 * @param hostAddress an int coding for an IPv4 address, where higher-order int byte is 35 * lower-order IPv4 address byte 36 */ intToInet4AddressHTL(int hostAddress)37 public static Inet4Address intToInet4AddressHTL(int hostAddress) { 38 return intToInet4AddressHTH(Integer.reverseBytes(hostAddress)); 39 } 40 41 /** 42 * Convert a IPv4 address from an integer to an InetAddress (0x01020304 -> 1.2.3.4) 43 * @param hostAddress an int coding for an IPv4 address 44 */ intToInet4AddressHTH(int hostAddress)45 public static Inet4Address intToInet4AddressHTH(int hostAddress) { 46 byte[] addressBytes = { (byte) (0xff & (hostAddress >> 24)), 47 (byte) (0xff & (hostAddress >> 16)), 48 (byte) (0xff & (hostAddress >> 8)), 49 (byte) (0xff & hostAddress) }; 50 51 try { 52 return (Inet4Address) InetAddress.getByAddress(addressBytes); 53 } catch (UnknownHostException e) { 54 throw new AssertionError(); 55 } 56 } 57 58 /** 59 * Convert an IPv4 address from an InetAddress to an integer (1.2.3.4 -> 0x01020304) 60 * 61 * <p>This conversion can help order IP addresses: considering the ordering 62 * 192.0.2.1 < 192.0.2.2 < ..., resulting ints will follow that ordering if read as unsigned 63 * integers with {@link Integer#toUnsignedLong}. 64 * @param inetAddr is an InetAddress corresponding to the IPv4 address 65 * @return the IP address as integer 66 */ inet4AddressToIntHTH(Inet4Address inetAddr)67 public static int inet4AddressToIntHTH(Inet4Address inetAddr) 68 throws IllegalArgumentException { 69 byte [] addr = inetAddr.getAddress(); 70 return ((addr[0] & 0xff) << 24) | ((addr[1] & 0xff) << 16) 71 | ((addr[2] & 0xff) << 8) | (addr[3] & 0xff); 72 } 73 74 /** 75 * Convert a IPv4 address from an InetAddress to an integer (1.2.3.4 -> 0x04030201) 76 * 77 * <p>This method stores the higher-order IPv4 address bytes in the lower-order int bytes, 78 * which is an unusual convention. Consider {@link #inet4AddressToIntHTH(Inet4Address)} instead. 79 * @param inetAddr is an InetAddress corresponding to the IPv4 address 80 * @return the IP address as integer 81 */ inet4AddressToIntHTL(Inet4Address inetAddr)82 public static int inet4AddressToIntHTL(Inet4Address inetAddr) { 83 return Integer.reverseBytes(inet4AddressToIntHTH(inetAddr)); 84 } 85 86 /** 87 * Convert a network prefix length to an IPv4 netmask integer (prefixLength 17 -> 0xffff8000) 88 * @return the IPv4 netmask as an integer 89 */ prefixLengthToV4NetmaskIntHTH(int prefixLength)90 public static int prefixLengthToV4NetmaskIntHTH(int prefixLength) 91 throws IllegalArgumentException { 92 if (prefixLength < 0 || prefixLength > 32) { 93 throw new IllegalArgumentException("Invalid prefix length (0 <= prefix <= 32)"); 94 } 95 // (int)a << b is equivalent to a << (b & 0x1f): can't shift by 32 (-1 << 32 == -1) 96 return prefixLength == 0 ? 0 : 0xffffffff << (32 - prefixLength); 97 } 98 99 /** 100 * Convert a network prefix length to an IPv4 netmask integer (prefixLength 17 -> 0x0080ffff). 101 * 102 * <p>This method stores the higher-order IPv4 address bytes in the lower-order int bytes, 103 * which is an unusual convention. Consider {@link #prefixLengthToV4NetmaskIntHTH(int)} instead. 104 * @return the IPv4 netmask as an integer 105 */ prefixLengthToV4NetmaskIntHTL(int prefixLength)106 public static int prefixLengthToV4NetmaskIntHTL(int prefixLength) 107 throws IllegalArgumentException { 108 return Integer.reverseBytes(prefixLengthToV4NetmaskIntHTH(prefixLength)); 109 } 110 111 /** 112 * Convert an IPv4 netmask to a prefix length, checking that the netmask is contiguous. 113 * @param netmask as a {@code Inet4Address}. 114 * @return the network prefix length 115 * @throws IllegalArgumentException the specified netmask was not contiguous. 116 * @hide 117 */ netmaskToPrefixLength(Inet4Address netmask)118 public static int netmaskToPrefixLength(Inet4Address netmask) { 119 // inetAddressToInt returns an int in *network* byte order. 120 int i = inet4AddressToIntHTH(netmask); 121 int prefixLength = Integer.bitCount(i); 122 int trailingZeros = Integer.numberOfTrailingZeros(i); 123 if (trailingZeros != 32 - prefixLength) { 124 throw new IllegalArgumentException("Non-contiguous netmask: " + Integer.toHexString(i)); 125 } 126 return prefixLength; 127 } 128 129 /** 130 * Returns the implicit netmask of an IPv4 address, as was the custom before 1993. 131 */ getImplicitNetmask(Inet4Address address)132 public static int getImplicitNetmask(Inet4Address address) { 133 int firstByte = address.getAddress()[0] & 0xff; // Convert to an unsigned value. 134 if (firstByte < 128) { 135 return 8; 136 } else if (firstByte < 192) { 137 return 16; 138 } else if (firstByte < 224) { 139 return 24; 140 } else { 141 return 32; // Will likely not end well for other reasons. 142 } 143 } 144 145 /** 146 * Get the broadcast address for a given prefix. 147 * 148 * <p>For example 192.168.0.1/24 -> 192.168.0.255 149 */ getBroadcastAddress(Inet4Address addr, int prefixLength)150 public static Inet4Address getBroadcastAddress(Inet4Address addr, int prefixLength) 151 throws IllegalArgumentException { 152 final int intBroadcastAddr = inet4AddressToIntHTH(addr) 153 | ~prefixLengthToV4NetmaskIntHTH(prefixLength); 154 return intToInet4AddressHTH(intBroadcastAddr); 155 } 156 157 /** 158 * Get a prefix mask as Inet4Address for a given prefix length. 159 * 160 * <p>For example 20 -> 255.255.240.0 161 */ getPrefixMaskAsInet4Address(int prefixLength)162 public static Inet4Address getPrefixMaskAsInet4Address(int prefixLength) 163 throws IllegalArgumentException { 164 return intToInet4AddressHTH(prefixLengthToV4NetmaskIntHTH(prefixLength)); 165 } 166 167 /** 168 * Trim leading zeros from IPv4 address strings 169 * Non-v4 addresses and host names remain unchanged. 170 * For example, 192.168.000.010 -> 192.168.0.10 171 * @param addr a string representing an ip address 172 * @return a string properly trimmed 173 */ trimAddressZeros(String addr)174 public static String trimAddressZeros(String addr) { 175 if (addr == null) return null; 176 String[] octets = addr.split("\\."); 177 if (octets.length != 4) return addr; 178 StringBuilder builder = new StringBuilder(16); 179 String result = null; 180 for (int i = 0; i < 4; i++) { 181 try { 182 if (octets[i].length() > 3) return addr; 183 builder.append(Integer.parseInt(octets[i])); 184 } catch (NumberFormatException e) { 185 return addr; 186 } 187 if (i < 3) builder.append('.'); 188 } 189 result = builder.toString(); 190 return result; 191 } 192 } 193