Bug-6562: Support add-path in base BGP NLRI
[bgpcep.git] / bgp / parser-impl / src / main / java / org / opendaylight / protocol / bgp / parser / impl / message / open / GracefulCapabilityHandler.java
index cda8c5fac3f780798e76761a4d319d61f3d3948e..5519b7a001975c084fe183b75ff842198ea75d0e 100644 (file)
@@ -7,8 +7,10 @@
  */
 package org.opendaylight.protocol.bgp.parser.impl.message.open;
 
+import static java.util.Objects.requireNonNull;
+import static org.opendaylight.protocol.util.ByteBufWriteUtil.writeUnsignedShort;
+
 import com.google.common.base.Preconditions;
-import com.google.common.primitives.UnsignedBytes;
 import io.netty.buffer.ByteBuf;
 import io.netty.buffer.Unpooled;
 import java.util.ArrayList;
@@ -20,15 +22,16 @@ import org.opendaylight.protocol.bgp.parser.spi.CapabilityParser;
 import org.opendaylight.protocol.bgp.parser.spi.CapabilitySerializer;
 import org.opendaylight.protocol.bgp.parser.spi.CapabilityUtil;
 import org.opendaylight.protocol.bgp.parser.spi.SubsequentAddressFamilyRegistry;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.open.bgp.parameters.CParameters;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.GracefulRestartCase;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.GracefulRestartCaseBuilder;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.GracefulRestartCapability;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.GracefulRestartCapability.RestartFlags;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.GracefulRestartCapabilityBuilder;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.graceful.restart.capability.Tables;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.graceful.restart.capability.Tables.AfiFlags;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.graceful.restart.capability.TablesBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev171207.open.message.bgp.parameters.optional.capabilities.CParameters;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev171207.open.message.bgp.parameters.optional.capabilities.CParametersBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.CParameters1;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.CParameters1Builder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.GracefulRestartCapability;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.GracefulRestartCapability.RestartFlags;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.GracefulRestartCapabilityBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.graceful.restart.capability.Tables;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.graceful.restart.capability.Tables.AfiFlags;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev171207.mp.capabilities.graceful.restart.capability.TablesBuilder;
 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.AddressFamily;
 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.SubsequentAddressFamily;
 import org.slf4j.Logger;
@@ -41,7 +44,7 @@ public final class GracefulCapabilityHandler implements CapabilityParser, Capabi
 
     // Restart flag size, in bits
     private static final int RESTART_FLAGS_SIZE = 4;
-    private static final int RESTART_FLAG_STATE = 0x80;
+    private static final int RESTART_FLAG_STATE = 0x8000;
 
     // Restart timer size, in bits
     private static final int TIMER_SIZE = 12;
@@ -53,55 +56,71 @@ public final class GracefulCapabilityHandler implements CapabilityParser, Capabi
     // Length of each AFI/SAFI array member, in bytes
     private static final int PER_AFI_SAFI_SIZE = 4;
 
-    private static final int AFI_FLAG_FORWARDING_STATE = 0x80;
+    private static final short AFI_FLAG_FORWARDING_STATE = 0x80;
+
+    private static final int MAX_RESTART_TIME = 4095;
 
     private final AddressFamilyRegistry afiReg;
     private final SubsequentAddressFamilyRegistry safiReg;
 
     public GracefulCapabilityHandler(final AddressFamilyRegistry afiReg, final SubsequentAddressFamilyRegistry safiReg) {
-        this.afiReg = Preconditions.checkNotNull(afiReg);
-        this.safiReg = Preconditions.checkNotNull(safiReg);
+        this.afiReg = requireNonNull(afiReg);
+        this.safiReg = requireNonNull(safiReg);
     }
 
-    @Override
-    public void serializeCapability(final CParameters capability, final ByteBuf byteAggregator) {
-        Preconditions.checkArgument(capability instanceof GracefulRestartCase);
-        final GracefulRestartCapability grace = ((GracefulRestartCase) capability).getGracefulRestartCapability();
-        final List<Tables> tables = grace.getTables();
-
-        final ByteBuf bytes = Unpooled.buffer(HEADER_SIZE + PER_AFI_SAFI_SIZE * tables.size());
-
-        int flagBits = 0;
-        final RestartFlags flags = grace.getRestartFlags();
-        if (flags != null && flags.isRestartState()) {
-            flagBits |= RESTART_FLAG_STATE;
-        }
-        int timeval = 0;
-        final Integer time = grace.getRestartTime();
-        if (time != null) {
-            Preconditions.checkArgument(time >= 0 && time <= 4095);
-            timeval = time;
+    private void serializeTables(final List<Tables> tables, final ByteBuf bytes) {
+        if (tables == null) {
+            return;
         }
-        bytes.writeByte(flagBits + timeval / 256);
-        bytes.writeByte(timeval % 256);
-
         for (final Tables t : tables) {
             final Class<? extends AddressFamily> afi = t.getAfi();
             final Integer afival = this.afiReg.numberForClass(afi);
             Preconditions.checkArgument(afival != null, "Unhandled address family " + afi);
-
+            bytes.writeShort(afival);
             final Class<? extends SubsequentAddressFamily> safi = t.getSafi();
             final Integer safival = this.safiReg.numberForClass(safi);
             Preconditions.checkArgument(safival != null, "Unhandled subsequent address family " + safi);
-
-            bytes.writeByte(afival / 256);
-            bytes.writeByte(afival % 256);
             bytes.writeByte(safival);
-            if (t.getAfiFlags().isForwardingState()) {
+            if (t.getAfiFlags() != null && t.getAfiFlags().isForwardingState()) {
                 bytes.writeByte(AFI_FLAG_FORWARDING_STATE);
+            } else {
+                bytes.writeZero(1);
             }
         }
-        CapabilityUtil.formatCapability(CODE, bytes,byteAggregator);
+    }
+
+    private ByteBuf serializeCapability(final GracefulRestartCapability grace) {
+        final List<Tables> tables = grace.getTables();
+        final int tablesSize = (tables != null) ? tables.size() : 0;
+        final ByteBuf bytes = Unpooled.buffer(HEADER_SIZE + (PER_AFI_SAFI_SIZE * tablesSize));
+        int timeval = 0;
+        Integer time = grace.getRestartTime();
+        if (time == null) {
+            time = 0;
+        }
+        Preconditions.checkArgument(time >= 0 && time <= MAX_RESTART_TIME, "Restart time is " + time);
+        timeval = time;
+        final GracefulRestartCapability.RestartFlags flags = grace.getRestartFlags();
+        if (flags != null && flags.isRestartState()) {
+            writeUnsignedShort(RESTART_FLAG_STATE | timeval, bytes);
+        } else {
+            writeUnsignedShort(timeval, bytes);
+        }
+        serializeTables(tables, bytes);
+        return bytes;
+    }
+
+    @Override
+    public void serializeCapability(final CParameters capability, final ByteBuf byteAggregator) {
+        if (capability.getAugmentation(CParameters1.class) == null
+            || capability.getAugmentation(CParameters1.class).getGracefulRestartCapability() == null) {
+            return;
+        }
+        final GracefulRestartCapability grace = capability.getAugmentation(CParameters1.class).getGracefulRestartCapability();
+
+        final ByteBuf bytes = serializeCapability(grace);
+
+        CapabilityUtil.formatCapability(CODE, bytes, byteAggregator);
     }
 
     @Override
@@ -109,28 +128,32 @@ public final class GracefulCapabilityHandler implements CapabilityParser, Capabi
         final GracefulRestartCapabilityBuilder cb = new GracefulRestartCapabilityBuilder();
 
         final int flagBits = (buffer.getByte(0) >> RESTART_FLAGS_SIZE);
-        cb.setRestartFlags(new RestartFlags((flagBits & 8) != 0));
+        cb.setRestartFlags(new RestartFlags((flagBits & Byte.SIZE) != 0));
 
-        final int timer = ((buffer.readByte() & TIMER_TOPBITS_MASK) << RESTART_FLAGS_SIZE) + UnsignedBytes.toInt(buffer.readByte());
+        final int timer = ((buffer.readUnsignedByte() & TIMER_TOPBITS_MASK) << Byte.SIZE) + buffer.readUnsignedByte();
         cb.setRestartTime(timer);
 
         final List<Tables> tables = new ArrayList<>();
         while (buffer.readableBytes() != 0) {
-            final int afiVal = UnsignedBytes.toInt(buffer.readByte()) * 256 + UnsignedBytes.toInt(buffer.readByte());
+            final int afiVal = buffer.readShort();
             final Class<? extends AddressFamily> afi = this.afiReg.classForFamily(afiVal);
             if (afi == null) {
                 LOG.debug("Ignoring GR capability for unknown address family {}", afiVal);
+                buffer.skipBytes(PER_AFI_SAFI_SIZE - 2);
                 continue;
             }
-            final int safiVal = UnsignedBytes.toInt(buffer.readByte());
+            final int safiVal = buffer.readUnsignedByte();
             final Class<? extends SubsequentAddressFamily> safi = this.safiReg.classForFamily(safiVal);
             if (safi == null) {
                 LOG.debug("Ignoring GR capability for unknown subsequent address family {}", safiVal);
+                buffer.skipBytes(1);
                 continue;
             }
-            final int flags = UnsignedBytes.toInt(buffer.readByte());
+            final int flags = buffer.readUnsignedByte();
             tables.add(new TablesBuilder().setAfi(afi).setSafi(safi).setAfiFlags(new AfiFlags((flags & AFI_FLAG_FORWARDING_STATE) != 0)).build());
         }
-        return new GracefulRestartCaseBuilder().setGracefulRestartCapability(cb.build()).build();
+        cb.setTables(tables);
+        return new CParametersBuilder().addAugmentation(CParameters1.class, new CParameters1Builder()
+            .setGracefulRestartCapability(cb.build()).build()).build();
     }
 }