Add egress split horizon drop flows for external interfaces
[genius.git] / interfacemanager / interfacemanager-impl / src / main / java / org / opendaylight / genius / interfacemanager / servicebindings / flowbased / state / helpers / FlowBasedEgressServicesStateBindHelper.java
1 /*
2  * Copyright (c) 2016 Ericsson India Global Services Pvt Ltd. and others.  All rights reserved.
3  *
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
7  */
8 package org.opendaylight.genius.interfacemanager.servicebindings.flowbased.state.helpers;
9
10 import com.google.common.util.concurrent.ListenableFuture;
11 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
12 import org.opendaylight.controller.md.sal.binding.api.WriteTransaction;
13 import org.opendaylight.genius.interfacemanager.IfmConstants;
14 import org.opendaylight.genius.interfacemanager.IfmUtil;
15 import org.opendaylight.genius.interfacemanager.InterfacemgrProvider;
16 import org.opendaylight.genius.interfacemanager.commons.InterfaceManagerCommonUtils;
17 import org.opendaylight.genius.interfacemanager.servicebindings.flowbased.state.factory.FlowBasedServicesStateAddable;
18 import org.opendaylight.genius.interfacemanager.servicebindings.flowbased.utilities.FlowBasedServicesUtils;
19 import org.opendaylight.genius.mdsalutil.MatchInfo;
20 import org.opendaylight.genius.mdsalutil.NwConstants;
21 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.iana._if.type.rev140508.L2vlan;
22 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.iana._if.type.rev140508.Tunnel;
23 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface;
24 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.ServiceModeBase;
25 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.ServiceModeEgress;
26 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.service.bindings.ServicesInfo;
27 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.service.bindings.services.info.BoundServices;
28 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeConnectorId;
29 import org.slf4j.Logger;
30 import org.slf4j.LoggerFactory;
31
32 import java.math.BigInteger;
33 import java.util.ArrayList;
34 import java.util.Collections;
35 import java.util.Comparator;
36 import java.util.List;
37
38 public class FlowBasedEgressServicesStateBindHelper implements FlowBasedServicesStateAddable{
39     private static final Logger LOG = LoggerFactory.getLogger(FlowBasedEgressServicesStateBindHelper.class);
40
41     private InterfacemgrProvider interfaceMgrProvider;
42     private static volatile FlowBasedServicesStateAddable flowBasedServicesStateAddable;
43
44     private FlowBasedEgressServicesStateBindHelper(InterfacemgrProvider interfaceMgrProvider) {
45         this.interfaceMgrProvider = interfaceMgrProvider;
46     }
47
48     public static void intitializeFlowBasedEgressServicesStateBindHelper(InterfacemgrProvider interfaceMgrProvider) {
49         if (flowBasedServicesStateAddable == null) {
50             synchronized (FlowBasedEgressServicesStateBindHelper.class) {
51                 if (flowBasedServicesStateAddable == null) {
52                     flowBasedServicesStateAddable = new FlowBasedEgressServicesStateBindHelper(interfaceMgrProvider);
53                 }
54             }
55         }
56     }
57
58     public static FlowBasedServicesStateAddable getFlowBasedEgressServicesStateBindHelper() {
59         if (flowBasedServicesStateAddable == null) {
60             LOG.error("OvsInterfaceConfigAdd Renderer is not initialized");
61         }
62         return flowBasedServicesStateAddable;
63     }
64
65     public List<ListenableFuture<Void>> bindServicesOnInterface(Interface ifaceState) {
66         List<ListenableFuture<Void>> futures = new ArrayList<>();
67         if(ifaceState.getType() == null) {
68             return futures;
69         }
70         LOG.debug("binding services on interface {}", ifaceState.getName());
71         DataBroker dataBroker = interfaceMgrProvider.getDataBroker();
72         ServicesInfo servicesInfo = FlowBasedServicesUtils.getServicesInfoForInterface(ifaceState.getName(), ServiceModeEgress.class, dataBroker);
73         if (servicesInfo == null) {
74             LOG.trace("service info is null for interface {}", ifaceState.getName());
75             return futures;
76         }
77
78         List<BoundServices> allServices = servicesInfo.getBoundServices();
79         if (allServices == null || allServices.isEmpty()) {
80             LOG.trace("bound services is empty for interface {}", ifaceState.getName());
81             return futures;
82         }
83
84         if (ifaceState.getType().isAssignableFrom(L2vlan.class)) {
85             return bindServiceOnVlan(allServices, ifaceState, dataBroker);
86         } else if (ifaceState.getType().isAssignableFrom(Tunnel.class)){
87             return bindServiceOnTunnel(allServices, ifaceState, dataBroker);
88         }
89         return futures;
90     }
91
92     private static List<ListenableFuture<Void>> bindServiceOnTunnel(
93             List<BoundServices> allServices, Interface ifState, DataBroker dataBroker) {
94         List<ListenableFuture<Void>> futures = new ArrayList<>();
95         // FIXME : not supported yet
96         return futures;
97     }
98
99     private static List<ListenableFuture<Void>> bindServiceOnVlan(
100             List<BoundServices> allServices,
101             Interface ifState, DataBroker dataBroker) {
102         List<ListenableFuture<Void>> futures = new ArrayList<>();
103         NodeConnectorId nodeConnectorId = FlowBasedServicesUtils.getNodeConnectorIdFromInterface(ifState);
104         BigInteger dpId = IfmUtil.getDpnFromNodeConnectorId(nodeConnectorId);
105         WriteTransaction t = dataBroker.newWriteOnlyTransaction();
106         Collections.sort(allServices, new Comparator<BoundServices>() {
107             @Override
108             public int compare(BoundServices serviceInfo1, BoundServices serviceInfo2) {
109                 return serviceInfo1.getServicePriority().compareTo(serviceInfo2.getServicePriority());
110             }
111         });
112         BoundServices highestPriority = allServices.remove(0);
113         short nextServiceIndex = (short) (allServices.size() > 0 ? allServices.get(0).getServicePriority() : highestPriority.getServicePriority() + 1);
114         org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface iface = InterfaceManagerCommonUtils.getInterfaceFromConfigDS(ifState.getName(), dataBroker);
115         FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, highestPriority, ifState.getName(), t, ifState.getIfIndex(), NwConstants.DEFAULT_SERVICE_INDEX, nextServiceIndex, iface);
116         BoundServices prev = null;
117         for (BoundServices boundService : allServices) {
118             if (prev!=null) {
119                 FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, prev, ifState.getName(), t, ifState.getIfIndex(), prev.getServicePriority(), boundService.getServicePriority(), iface);
120             }
121             prev = boundService;
122         }
123         if (prev!=null) {
124             FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, prev, ifState.getName(), t, ifState.getIfIndex(), prev.getServicePriority(), (short) (prev.getServicePriority()+1), iface);
125         }
126         futures.add(t.submit());
127         return futures;
128
129     }
130
131 }