2 * Copyright (c) 2013 Cisco Systems, Inc. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
8 package org.opendaylight.protocol.bgp.parser.impl.message.open;
10 import java.util.ArrayList;
11 import java.util.List;
13 import org.opendaylight.protocol.bgp.parser.BGPDocumentedException;
14 import org.opendaylight.protocol.bgp.parser.BGPParsingException;
15 import org.opendaylight.protocol.bgp.parser.spi.AddressFamilyRegistry;
16 import org.opendaylight.protocol.bgp.parser.spi.CapabilityParser;
17 import org.opendaylight.protocol.bgp.parser.spi.CapabilitySerializer;
18 import org.opendaylight.protocol.bgp.parser.spi.CapabilityUtil;
19 import org.opendaylight.protocol.bgp.parser.spi.SubsequentAddressFamilyRegistry;
20 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.open.bgp.parameters.CParameters;
21 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.GracefulRestartCase;
22 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.GracefulRestartCaseBuilder;
23 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.GracefulRestartCapability;
24 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;
25 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.c.parameters.graceful.restart._case.GracefulRestartCapabilityBuilder;
26 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;
27 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;
28 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;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.AddressFamily;
30 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.SubsequentAddressFamily;
31 import org.slf4j.Logger;
32 import org.slf4j.LoggerFactory;
34 import com.google.common.base.Preconditions;
35 import com.google.common.primitives.UnsignedBytes;
37 public final class GracefulCapabilityHandler implements CapabilityParser, CapabilitySerializer {
38 public static final int CODE = 64;
40 private static final Logger LOG = LoggerFactory.getLogger(GracefulCapabilityHandler.class);
42 // Restart flag size, in bits
43 private static final int RESTART_FLAGS_SIZE = 4;
44 private static final int RESTART_FLAG_STATE = 0x80;
46 // Restart timer size, in bits
47 private static final int TIMER_SIZE = 12;
48 private static final int TIMER_TOPBITS_MASK = 0x0F;
50 // Size of the capability header
51 private static final int HEADER_SIZE = (RESTART_FLAGS_SIZE + TIMER_SIZE) / Byte.SIZE;
53 // Length of each AFI/SAFI array member, in bytes
54 private static final int PER_AFI_SAFI_SIZE = 4;
56 private static final int AFI_FLAG_FORWARDING_STATE = 0x80;
58 private final AddressFamilyRegistry afiReg;
59 private final SubsequentAddressFamilyRegistry safiReg;
61 public GracefulCapabilityHandler(final AddressFamilyRegistry afiReg, final SubsequentAddressFamilyRegistry safiReg) {
62 this.afiReg = Preconditions.checkNotNull(afiReg);
63 this.safiReg = Preconditions.checkNotNull(safiReg);
67 public byte[] serializeCapability(final CParameters capability) {
68 final GracefulRestartCapability grace = ((GracefulRestartCase) capability).getGracefulRestartCapability();
69 final List<Tables> tables = grace.getTables();
71 final byte[] bytes = new byte[HEADER_SIZE + PER_AFI_SAFI_SIZE * tables.size()];
74 final RestartFlags flags = grace.getRestartFlags();
76 if (flags.isRestartState()) {
77 flagBits |= RESTART_FLAG_STATE;
81 final Integer time = grace.getRestartTime();
83 Preconditions.checkArgument(time >= 0 && time <= 4095);
86 bytes[0] = UnsignedBytes.checkedCast(flagBits + timeval / 256);
87 bytes[1] = UnsignedBytes.checkedCast(timeval % 256);
89 int index = HEADER_SIZE;
90 for (final Tables t : tables) {
91 final Class<? extends AddressFamily> afi = t.getAfi();
92 final Integer afival = this.afiReg.numberForClass(afi);
93 Preconditions.checkArgument(afival != null, "Unhandled address family " + afi);
95 final Class<? extends SubsequentAddressFamily> safi = t.getSafi();
96 final Integer safival = this.safiReg.numberForClass(safi);
97 Preconditions.checkArgument(safival != null, "Unhandled subsequent address family " + safi);
99 bytes[index] = UnsignedBytes.checkedCast(afival / 256);
100 bytes[index + 1] = UnsignedBytes.checkedCast(afival % 256);
101 bytes[index + 2] = UnsignedBytes.checkedCast(safival);
102 if (t.getAfiFlags().isForwardingState()) {
103 bytes[index + 3] = UnsignedBytes.checkedCast(AFI_FLAG_FORWARDING_STATE);
105 index += PER_AFI_SAFI_SIZE;
107 return CapabilityUtil.formatCapability(CODE, bytes);
111 public CParameters parseCapability(final byte[] bytes) throws BGPDocumentedException, BGPParsingException {
112 final GracefulRestartCapabilityBuilder cb = new GracefulRestartCapabilityBuilder();
114 final int flagBits = (bytes[0] >> RESTART_FLAGS_SIZE);
115 cb.setRestartFlags(new RestartFlags((flagBits & 8) != 0));
117 final int timer = ((bytes[0] & TIMER_TOPBITS_MASK) << RESTART_FLAGS_SIZE) + UnsignedBytes.toInt(bytes[1]);
118 cb.setRestartTime(timer);
120 final List<Tables> tables = new ArrayList<>();
121 for (int offset = HEADER_SIZE; offset < bytes.length; offset += PER_AFI_SAFI_SIZE) {
122 final int afiVal = UnsignedBytes.toInt(bytes[offset]) * 256 + UnsignedBytes.toInt(bytes[offset + 1]);
123 final Class<? extends AddressFamily> afi = this.afiReg.classForFamily(afiVal);
125 LOG.debug("Ignoring GR capability for unknown address family {}", afiVal);
128 final int safiVal = UnsignedBytes.toInt(bytes[offset + 2]);
129 final Class<? extends SubsequentAddressFamily> safi = this.safiReg.classForFamily(safiVal);
131 LOG.debug("Ignoring GR capability for unknown subsequent address family {}", safiVal);
134 final int flags = UnsignedBytes.toInt(bytes[offset + 3]);
135 tables.add(new TablesBuilder().setAfi(afi).setSafi(safi).setAfiFlags(new AfiFlags((flags & AFI_FLAG_FORWARDING_STATE) != 0)).build());
137 return new GracefulRestartCaseBuilder().setGracefulRestartCapability(cb.build()).build();