blob: abb217a7498711874a31e2065a5a39a74c0c7633 [file] [log] [blame]
Yotam Harcholf3f11152013-09-05 16:47:16 -07001package org.projectfloodlight.openflow.types;
2
Ronald Liffa80792014-07-04 17:50:49 -07003import java.net.Inet4Address;
Ronald Liaf8d3e72014-07-05 01:26:28 -07004import java.net.InetAddress;
Andreas Wundsam5f71b412014-02-18 12:56:35 -08005import java.util.Arrays;
6
Andreas Wundsam3700d162014-03-11 04:43:38 -07007import javax.annotation.Nonnull;
8
Yotam Harcholf3f11152013-09-05 16:47:16 -07009import org.jboss.netty.buffer.ChannelBuffer;
10
Sovietaced9dfc1ef2014-06-27 11:13:57 -070011import com.google.common.base.Preconditions;
Andreas Wundsam22ba3af2013-10-04 16:00:30 -070012import com.google.common.hash.PrimitiveSink;
Andreas Wundsam85c961f2013-09-29 21:22:12 -070013import com.google.common.primitives.UnsignedInts;
14
Yotam Harcholf3f11152013-09-05 16:47:16 -070015
16
17/**
Yotam Harchola289d552013-09-16 10:10:40 -070018 * Wrapper around an IPv4Address address
Yotam Harcholf3f11152013-09-05 16:47:16 -070019 *
20 * @author Andreas Wundsam <andreas.wundsam@bigswitch.com>
21 */
Yotam Harchol4d634682013-09-26 13:21:06 -070022public class IPv4Address extends IPAddress<IPv4Address> {
Yotam Harcholf3f11152013-09-05 16:47:16 -070023 static final int LENGTH = 4;
24 private final int rawValue;
Yotam Harcholf3f11152013-09-05 16:47:16 -070025
Gregor Maier1acb4502013-12-12 11:25:07 -080026 private static final int NOT_A_CIDR_MASK = -1;
27 private static final int CIDR_MASK_CACHE_UNSET = -2;
Gregor Maier5615b6c2013-12-11 22:29:07 -080028 // Must appear before the static IPv4Address constant assignments
29 private volatile int cidrMaskLengthCache = CIDR_MASK_CACHE_UNSET;
30
Andreas Wundsamb75c4ad2013-09-23 14:45:35 -070031 private final static int NONE_VAL = 0x0;
32 public final static IPv4Address NONE = new IPv4Address(NONE_VAL);
33
Yotam Harchola289d552013-09-16 10:10:40 -070034 public static final IPv4Address NO_MASK = IPv4Address.of(0xFFFFFFFF);
35 public static final IPv4Address FULL_MASK = IPv4Address.of(0x00000000);
36
37 private IPv4Address(final int rawValue) {
Yotam Harcholf3f11152013-09-05 16:47:16 -070038 this.rawValue = rawValue;
39 }
40
Yotam Harchol4d634682013-09-26 13:21:06 -070041 @Override
Yotam Harcholeb023dc2013-09-26 15:45:44 -070042 public IPVersion getIpVersion() {
43 return IPVersion.IPv4;
Yotam Harchol4d634682013-09-26 13:21:06 -070044 }
45
Gregor Maier5615b6c2013-12-11 22:29:07 -080046 private int asCidrMaskLengthInternal() {
47 if (cidrMaskLengthCache == CIDR_MASK_CACHE_UNSET) {
48 // No lock required. We only write cidrMaskLengthCache once
49 int maskint = getInt();
50 if (maskint == 0) {
51 cidrMaskLengthCache = 0;
52 } else if (Integer.bitCount((~maskint) + 1) == 1) {
53 // IP represents a true CIDR prefix length
54 cidrMaskLengthCache = Integer.bitCount(maskint);
55 } else {
56 cidrMaskLengthCache = NOT_A_CIDR_MASK;
57 }
58 }
59 return cidrMaskLengthCache;
60 }
61
62 @Override
63 public boolean isCidrMask() {
64 return asCidrMaskLengthInternal() != NOT_A_CIDR_MASK;
65 }
Gregor Maier7f987e62013-12-10 19:34:18 -080066
67 @Override
68 public int asCidrMaskLength() {
Gregor Maier5615b6c2013-12-11 22:29:07 -080069 if (!isCidrMask()) {
70 throw new IllegalStateException("IP is not a valid CIDR prefix " +
71 "mask " + toString());
Gregor Maier7f987e62013-12-10 19:34:18 -080072 } else {
Gregor Maier5615b6c2013-12-11 22:29:07 -080073 return asCidrMaskLengthInternal();
Gregor Maier7f987e62013-12-10 19:34:18 -080074 }
75 }
76
Aditya Vaja56b8b182014-03-11 13:13:58 -070077 @Override
78 public boolean isBroadcast() {
79 return this.equals(NO_MASK);
80 }
81
82 @Override
Aditya Vaja98c96e72014-03-11 15:19:01 -070083 public IPv4Address and(IPv4Address other) {
Sovietaced9dfc1ef2014-06-27 11:13:57 -070084 Preconditions.checkNotNull(other, "other must not be null");
85
86 IPv4Address otherIp = other;
Aditya Vaja56b8b182014-03-11 13:13:58 -070087 return IPv4Address.of(rawValue & otherIp.rawValue);
88 }
89
90 @Override
Aditya Vaja98c96e72014-03-11 15:19:01 -070091 public IPv4Address or(IPv4Address other) {
Sovietaced9dfc1ef2014-06-27 11:13:57 -070092 Preconditions.checkNotNull(other, "other must not be null");
93
94 IPv4Address otherIp = other;
Aditya Vaja56b8b182014-03-11 13:13:58 -070095 return IPv4Address.of(rawValue | otherIp.rawValue);
96 }
97
98 @Override
99 public IPv4Address not() {
100 return IPv4Address.of(~rawValue);
101 }
102
Ronald Liaf8d3e72014-07-05 01:26:28 -0700103 /**
104 * Returns an {@code IPv4Address} object that represents the given
Ronald Li097cda52014-07-06 23:17:05 -0700105 * IP address. The argument is in network byte order: the highest
Ronald Liaf8d3e72014-07-05 01:26:28 -0700106 * order byte of the address is in {@code address[0]}.
107 * <p>
108 * The address byte array must be 4 bytes long (32 bits long).
109 * <p>
110 * Similar to {@link InetAddress#getByAddress(byte[])}.
111 *
112 * @param address the raw IP address in network byte order
113 * @return an {@code IPv4Address} object that represents the given
114 * raw IP address
115 * @throws NullPointerException if the given address was {@code null}
116 * @throws IllegalArgumentException if the given address was of an invalid
117 * byte array length
118 * @see InetAddress#getByAddress(byte[])
119 */
Ronald Liffa80792014-07-04 17:50:49 -0700120 @Nonnull
121 public static IPv4Address of(@Nonnull final byte[] address) {
Sovietaced9dfc1ef2014-06-27 11:13:57 -0700122 Preconditions.checkNotNull(address, "address must not be null");
123
Yotam Harcholf3f11152013-09-05 16:47:16 -0700124 if (address.length != LENGTH) {
125 throw new IllegalArgumentException(
Andreas Wundsamc85b5c52013-09-24 13:01:43 -0700126 "Invalid byte array length for IPv4Address address: " + address.length);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700127 }
128
129 int raw =
130 (address[0] & 0xFF) << 24 | (address[1] & 0xFF) << 16
131 | (address[2] & 0xFF) << 8 | (address[3] & 0xFF) << 0;
Yotam Harchola289d552013-09-16 10:10:40 -0700132 return IPv4Address.of(raw);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700133 }
134
Ronald Liaf8d3e72014-07-05 01:26:28 -0700135 /**
136 * Returns an {@code IPv4Address} object that represents the given
Ronald Li097cda52014-07-06 23:17:05 -0700137 * IP address.
Andreas Wundsam3700d162014-03-11 04:43:38 -0700138 *
Ronald Liaf8d3e72014-07-05 01:26:28 -0700139 * @param raw the raw IP address represented as a 32-bit integer
140 * @return an {@code IPv4Address} object that represents the given
141 * raw IP address
Andreas Wundsam3700d162014-03-11 04:43:38 -0700142 */
Ronald Liffa80792014-07-04 17:50:49 -0700143 @Nonnull
Yotam Harchola289d552013-09-16 10:10:40 -0700144 public static IPv4Address of(final int raw) {
Andreas Wundsamb75c4ad2013-09-23 14:45:35 -0700145 if(raw == NONE_VAL)
146 return NONE;
Yotam Harchola289d552013-09-16 10:10:40 -0700147 return new IPv4Address(raw);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700148 }
149
Ronald Liaf8d3e72014-07-05 01:26:28 -0700150 /**
151 * Returns an {@code IPv4Address} object that represents the given
Ronald Li5b41da92014-07-06 22:59:28 -0700152 * IP address. The argument is in the canonical quad-dotted notation.
153 * For example, {@code 1.2.3.4}.
Andreas Wundsam3700d162014-03-11 04:43:38 -0700154 *
Ronald Li5b41da92014-07-06 22:59:28 -0700155 * @param string the IP address in the canonical quad-dotted notation
Ronald Liaf8d3e72014-07-05 01:26:28 -0700156 * @return an {@code IPv4Address} object that represents the given
Ronald Li5b41da92014-07-06 22:59:28 -0700157 * IP address
Ronald Li097cda52014-07-06 23:17:05 -0700158 * @throws NullPointerException if the given string was {@code null}
159 * @throws IllegalArgumentException if the given string was not a valid
Ronald Liaf8d3e72014-07-05 01:26:28 -0700160 * IPv4 address
Andreas Wundsam3700d162014-03-11 04:43:38 -0700161 */
162 @Nonnull
163 public static IPv4Address of(@Nonnull final String string) throws IllegalArgumentException {
Sovietaced9dfc1ef2014-06-27 11:13:57 -0700164 Preconditions.checkNotNull(string, "string must not be null");
165
Yotam Harcholf3f11152013-09-05 16:47:16 -0700166 int start = 0;
167 int shift = 24;
168
169 int raw = 0;
170 while (shift >= 0) {
171 int end = string.indexOf('.', start);
172 if (end == start || !((shift > 0) ^ (end < 0)))
173 throw new IllegalArgumentException("IP Address not well formed: " + string);
174
175 String substr =
176 end > 0 ? string.substring(start, end) : string.substring(start);
177 int val = Integer.parseInt(substr);
178 if (val < 0 || val > 255)
179 throw new IllegalArgumentException("IP Address not well formed: " + string);
180
181 raw |= val << shift;
182
183 shift -= 8;
184 start = end + 1;
185 }
Yotam Harchola289d552013-09-16 10:10:40 -0700186 return IPv4Address.of(raw);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700187 }
188
Ronald Liaf8d3e72014-07-05 01:26:28 -0700189 /**
Ronald Li5b41da92014-07-06 22:59:28 -0700190 * Returns an {@code IPv4Address} object that represents the given
191 * IP address. The argument is given as an {@code Inet4Address} object.
Ronald Liaf8d3e72014-07-05 01:26:28 -0700192 *
193 * @param address the {@code Inet4Address} object
194 * @return an {@code IPv4Address} object that represents the
195 * given IP address
Ronald Li5b41da92014-07-06 22:59:28 -0700196 * @throws NullPointerException if the given {@code Inet4Address} was
197 * {@code null}
Ronald Liaf8d3e72014-07-05 01:26:28 -0700198 */
Ronald Liffa80792014-07-04 17:50:49 -0700199 @Nonnull
200 public static IPv4Address of(@Nonnull final Inet4Address address) {
201 Preconditions.checkNotNull(address, "address must not be null");
202 return IPv4Address.of(address.getAddress());
203 }
204
Ronald Liaf8d3e72014-07-05 01:26:28 -0700205 /**
206 * Returns an {@code IPv4Address} object that represents the
207 * CIDR subnet mask of the given prefix length, that is, the
208 * number of leading one-bits.
209 *
210 * @param cidrMaskLength the prefix length of the CIDR subnet mask,
211 * where {@code 0 <= cidrMaskLength <= 32}
212 * @return an {@code IPv4Address} object that represents the
213 * CIDR subnet mask of the given prefix length
214 * @throws IllegalArgumentException if the given prefix length was invalid
215 */
Ronald Liffa80792014-07-04 17:50:49 -0700216 @Nonnull
Ronald Lia7484222014-07-03 17:14:12 -0700217 public static IPv4Address ofCidrMaskLength(final int cidrMaskLength) {
218 Preconditions.checkArgument(
219 cidrMaskLength >= 0 && cidrMaskLength <= 32,
220 "Invalid IPv4 CIDR mask length: %s", cidrMaskLength);
221
222 if (cidrMaskLength == 32) {
223 return IPv4Address.NO_MASK;
224 } else if (cidrMaskLength == 0) {
225 return IPv4Address.FULL_MASK;
226 } else {
227 int mask = (-1) << (32 - cidrMaskLength);
228 return IPv4Address.of(mask);
229 }
230 }
231
Ronald Liaf8d3e72014-07-05 01:26:28 -0700232 /**
233 * Returns an {@code IPv4AddressWithMask} object that represents this
Ronald Lica08d5f2014-07-05 02:43:18 -0700234 * IP address masked by the given IP address mask.
235 *
236 * @param mask the {@code IPv4Address} object that represents the mask
237 * @return an {@code IPv4AddressWithMask} object that represents this
238 * IP address masked by the given mask
239 * @throws NullPointerException if the given mask was {@code null}
240 */
241 @Nonnull
242 public IPv4AddressWithMask withMask(@Nonnull final IPv4Address mask) {
243 return IPv4AddressWithMask.of(this, mask);
244 }
245
246 /**
247 * Returns an {@code IPv4AddressWithMask} object that represents this
Ronald Li097cda52014-07-06 23:17:05 -0700248 * IP address masked by the given IP address mask. The argument is in
Ronald Liaf8d3e72014-07-05 01:26:28 -0700249 * network byte order: the highest order byte of the address is in
250 * {@code mask[0]}.
251 * <p>
252 * The address byte array must be 4 bytes long (32 bits long).
253 *
254 * @param mask the raw IP address mask in network byte order
255 * @return an {@code IPv4AddressWithMask} object that represents this
256 * IP address masked by the given raw IP address mask
257 * @throws NullPointerException if the given mask was {@code null}
258 * @throws IllegalArgumentException if the given mask was of an invalid
259 * byte array length
260 * @see #of(byte[])
261 */
262 @Nonnull
263 public IPv4AddressWithMask withMask(@Nonnull final byte[] mask) {
Ronald Liffa80792014-07-04 17:50:49 -0700264 return IPv4AddressWithMask.of(this, IPv4Address.of(mask));
265 }
266
Ronald Liaf8d3e72014-07-05 01:26:28 -0700267 /**
268 * Returns an {@code IPv4AddressWithMask} object that represents this
Ronald Li097cda52014-07-06 23:17:05 -0700269 * IP address masked by the given IP address mask.
Ronald Liaf8d3e72014-07-05 01:26:28 -0700270 *
271 * @param mask the raw IP address mask represented as a 32-bit integer
272 * @return an {@code IPv4AddressWithMask} object that represents this
273 * IP address masked by the given raw IP address mask
274 * @see #of(int)
275 */
276 @Nonnull
277 public IPv4AddressWithMask withMask(final int mask) {
Ronald Liffa80792014-07-04 17:50:49 -0700278 return IPv4AddressWithMask.of(this, IPv4Address.of(mask));
279 }
280
Ronald Liaf8d3e72014-07-05 01:26:28 -0700281 /**
282 * Returns an {@code IPv4AddressWithMask} object that represents this
Ronald Li5b41da92014-07-06 22:59:28 -0700283 * IP address masked by the given IP address mask. The argument is in
284 * the canonical quad-dotted notation. For example, {@code 255.255.255.0}.
Ronald Liaf8d3e72014-07-05 01:26:28 -0700285 *
Ronald Li5b41da92014-07-06 22:59:28 -0700286 * @param mask the IP address mask in the canonical quad-dotted notation
Ronald Liaf8d3e72014-07-05 01:26:28 -0700287 * @return an {@code IPv4AddressWithMask} object that represents this
Ronald Li5b41da92014-07-06 22:59:28 -0700288 * IP address masked by the given IP address mask
Ronald Liaf8d3e72014-07-05 01:26:28 -0700289 * @throws NullPointerException if the given string was {@code null}
290 * @throws IllegalArgumentException if the given string was not a valid
Ronald Li5b41da92014-07-06 22:59:28 -0700291 * IPv4 address
Ronald Liaf8d3e72014-07-05 01:26:28 -0700292 * @see #of(String)
293 */
294 @Nonnull
295 public IPv4AddressWithMask withMask(@Nonnull final String mask) {
Ronald Liffa80792014-07-04 17:50:49 -0700296 return IPv4AddressWithMask.of(this, IPv4Address.of(mask));
297 }
298
Ronald Liaf8d3e72014-07-05 01:26:28 -0700299 /**
300 * Returns an {@code IPv4AddressWithMask} object that represents this
Ronald Li097cda52014-07-06 23:17:05 -0700301 * IP address masked by the given IP address mask. The argument is given as
Ronald Li5b41da92014-07-06 22:59:28 -0700302 * an {@code Inet4Address} object.
Ronald Liaf8d3e72014-07-05 01:26:28 -0700303 *
304 * @param mask the {@code Inet4Address} object
305 * @return an {@code IPv4AddressWithMask} object that represents this
306 * IP address masked by the given IP address mask
Ronald Li5b41da92014-07-06 22:59:28 -0700307 * @throws NullPointerException if the given {@code Inet4Address} was
308 * {@code null}
Ronald Liaf8d3e72014-07-05 01:26:28 -0700309 * @see #of(Inet4Address)
310 */
311 @Nonnull
312 public IPv4AddressWithMask withMask(@Nonnull final Inet4Address mask) {
Ronald Liffa80792014-07-04 17:50:49 -0700313 return IPv4AddressWithMask.of(this, IPv4Address.of(mask));
314 }
315
Ronald Liaf8d3e72014-07-05 01:26:28 -0700316 /**
317 * Returns an {@code IPv4AddressWithMask} object that represents this
318 * IP address masked by the CIDR subnet mask of the given prefix length,
319 * that is, the number of leading one-bits.
320 *
321 * @param cidrMaskLength the prefix length of the CIDR subnet mask,
322 * where {@code 0 <= cidrMaskLength <= 32}
323 * @return an {@code IPv4AddressWithMask} object that
324 * represents this IP address masked by the CIDR
325 * subnet mask of the given prefix length
326 * @throws IllegalArgumentException if the given prefix length was invalid
327 * @see #ofCidrMaskLength(int)
328 */
329 @Nonnull
330 public IPv4AddressWithMask withMaskOfLength(final int cidrMaskLength) {
Ronald Liffa80792014-07-04 17:50:49 -0700331 return IPv4AddressWithMask.of(this,
332 IPv4Address.ofCidrMaskLength(cidrMaskLength));
333 }
334
Yotam Harcholf3f11152013-09-05 16:47:16 -0700335 public int getInt() {
336 return rawValue;
337 }
338
Andreas Wundsam4e2469e2014-02-17 15:32:43 -0800339 private volatile byte[] bytesCache = null;
Yotam Harcholf3f11152013-09-05 16:47:16 -0700340
Ronald Liaf8d3e72014-07-05 01:26:28 -0700341 @Override
Yotam Harcholf3f11152013-09-05 16:47:16 -0700342 public byte[] getBytes() {
343 if (bytesCache == null) {
344 synchronized (this) {
345 if (bytesCache == null) {
346 bytesCache =
347 new byte[] { (byte) ((rawValue >>> 24) & 0xFF),
348 (byte) ((rawValue >>> 16) & 0xFF),
349 (byte) ((rawValue >>> 8) & 0xFF),
350 (byte) ((rawValue >>> 0) & 0xFF) };
351 }
352 }
353 }
Andreas Wundsam5f71b412014-02-18 12:56:35 -0800354 return Arrays.copyOf(bytesCache, bytesCache.length);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700355 }
356
357 @Override
358 public int getLength() {
359 return LENGTH;
360 }
361
362 @Override
363 public String toString() {
364 StringBuilder res = new StringBuilder();
365 res.append((rawValue >> 24) & 0xFF).append('.');
366 res.append((rawValue >> 16) & 0xFF).append('.');
367 res.append((rawValue >> 8) & 0xFF).append('.');
368 res.append((rawValue >> 0) & 0xFF);
369 return res.toString();
370 }
371
Yotam Harcholf3f11152013-09-05 16:47:16 -0700372 public void write4Bytes(ChannelBuffer c) {
373 c.writeInt(rawValue);
374 }
Yotam Harchola289d552013-09-16 10:10:40 -0700375
376 public static IPv4Address read4Bytes(ChannelBuffer c) {
377 return IPv4Address.of(c.readInt());
Yotam Harcholf3f11152013-09-05 16:47:16 -0700378 }
379
380 @Override
Yotam Harchola289d552013-09-16 10:10:40 -0700381 public IPv4Address applyMask(IPv4Address mask) {
Aditya Vaja98c96e72014-03-11 15:19:01 -0700382 return and(mask);
Yotam Harcholf3f11152013-09-05 16:47:16 -0700383 }
384
Andreas Wundsam85c961f2013-09-29 21:22:12 -0700385 @Override
386 public int hashCode() {
387 final int prime = 31;
388 int result = 1;
Andreas Wundsam85c961f2013-09-29 21:22:12 -0700389 result = prime * result + rawValue;
390 return result;
391 }
Yotam Harchola289d552013-09-16 10:10:40 -0700392
Andreas Wundsam85c961f2013-09-29 21:22:12 -0700393 @Override
394 public boolean equals(Object obj) {
395 if (this == obj)
396 return true;
397 if (obj == null)
398 return false;
399 if (getClass() != obj.getClass())
400 return false;
401 IPv4Address other = (IPv4Address) obj;
Andreas Wundsam85c961f2013-09-29 21:22:12 -0700402 if (rawValue != other.rawValue)
403 return false;
404 return true;
405 }
406
407 @Override
408 public int compareTo(IPv4Address o) {
409 return UnsignedInts.compare(rawValue, o.rawValue);
410 }
Andreas Wundsam22ba3af2013-10-04 16:00:30 -0700411
412 @Override
413 public void putTo(PrimitiveSink sink) {
414 sink.putInt(rawValue);
415 }
416
Yotam Harcholf3f11152013-09-05 16:47:16 -0700417}