Add abstract IP header class to unify IPv4 and IPv6 header classes

Change-Id: Ia932dad67f64595b52b6fbc7dc43a13f64d53796
diff --git a/utils/misc/src/main/java/org/onlab/packet/IP.java b/utils/misc/src/main/java/org/onlab/packet/IP.java
new file mode 100644
index 0000000..a25ad54
--- /dev/null
+++ b/utils/misc/src/main/java/org/onlab/packet/IP.java
@@ -0,0 +1,60 @@
+/*
+ * Copyright 2016-present Open Networking Laboratory
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.onlab.packet;
+
+import java.nio.ByteBuffer;
+
+/**
+ * Implements IP packet format.
+ */
+public abstract class IP extends BasePacket {
+
+    /**
+     * Gets IP version number.
+     *
+     * @return IP version number
+     */
+    public abstract byte getVersion();
+
+    /**
+     * Sets IP version number.
+     *
+     * @param version the version to set
+     * @return IP class
+     */
+    public abstract IP setVersion(final byte version);
+
+    /**
+     * Deserializer function for IP packets.
+     *
+     * @return deserializer function
+     */
+    public static Deserializer<? extends IP> deserializer() {
+        return (data, offset, length) -> {
+            final ByteBuffer bb = ByteBuffer.wrap(data, offset, length);
+            byte version = (byte) (bb.get() >> 4 & 0xf);
+
+            switch (version) {
+                case 4:
+                    return IPv4.deserializer().deserialize(data, offset, length);
+                case 6:
+                    return IPv6.deserializer().deserialize(data, offset, length);
+                default:
+                    throw new DeserializationException("Invalid IP version");
+            }
+        };
+    }
+}
diff --git a/utils/misc/src/main/java/org/onlab/packet/IPv4.java b/utils/misc/src/main/java/org/onlab/packet/IPv4.java
index dc51689..ac27918 100644
--- a/utils/misc/src/main/java/org/onlab/packet/IPv4.java
+++ b/utils/misc/src/main/java/org/onlab/packet/IPv4.java
@@ -14,9 +14,6 @@
  * limitations under the License.
  */
 
-/**
- *
- */
 package org.onlab.packet;
 
 import java.nio.ByteBuffer;
@@ -31,7 +28,7 @@
 /**
  * Implements IPv4 packet format.
  */
-public class IPv4 extends BasePacket {
+public class IPv4 extends IP {
     public static final byte PROTOCOL_ICMP = 0x1;
     public static final byte PROTOCOL_IGMP = 0x2;
     public static final byte PROTOCOL_TCP = 0x6;
@@ -79,18 +76,12 @@
         this.isTruncated = false;
     }
 
-    /**
-     * @return the version
-     */
+    @Override
     public byte getVersion() {
         return this.version;
     }
 
-    /**
-     * @param version
-     *            the version to set
-     * @return this
-     */
+    @Override
     public IPv4 setVersion(final byte version) {
         this.version = version;
         return this;
diff --git a/utils/misc/src/main/java/org/onlab/packet/IPv6.java b/utils/misc/src/main/java/org/onlab/packet/IPv6.java
index 414e724..ed26268 100644
--- a/utils/misc/src/main/java/org/onlab/packet/IPv6.java
+++ b/utils/misc/src/main/java/org/onlab/packet/IPv6.java
@@ -14,8 +14,6 @@
  * limitations under the License.
  */
 
-
-
 package org.onlab.packet;
 
 import org.onlab.packet.ipv6.Authentication;
@@ -37,7 +35,7 @@
 /**
  * Implements IPv6 packet format. (RFC 2460)
  */
-public class IPv6 extends BasePacket implements IExtensionHeader {
+public class IPv6 extends IP implements IExtensionHeader {
     public static final byte FIXED_HEADER_LENGTH = 40; // bytes
 
     public static final byte PROTOCOL_TCP = 0x6;
@@ -83,21 +81,12 @@
         this.version = 6;
     }
 
-    /**
-     * Gets IP version.
-     *
-     * @return the IP version
-     */
+    @Override
     public byte getVersion() {
         return this.version;
     }
 
-    /**
-     * Sets IP version.
-     *
-     * @param version the IP version to set
-     * @return this
-     */
+    @Override
     public IPv6 setVersion(final byte version) {
         this.version = version;
         return this;