blob: d9b792d2e5ddf57c93b3d4650f12c47357a5ce76 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Jonathan Hartf4bd0482017-01-27 15:11:18 -08002 * Copyright 2017-present Open Networking Laboratory
Thomas Vachuska781d18b2014-10-27 10:31:25 -07003 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07004 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
Thomas Vachuska781d18b2014-10-27 10:31:25 -07007 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07008 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
Thomas Vachuska781d18b2014-10-27 10:31:25 -070015 */
Jonathan Hartf4bd0482017-01-27 15:11:18 -080016
Jonathan Hart2da1e602015-02-18 19:09:24 -080017package org.onosproject.routing.config.impl;
Jonathan Hart90a02c22015-02-13 11:52:07 -080018
Pingping Lin9a445c82016-04-07 11:40:29 -070019import com.google.common.collect.ImmutableSet;
Pingping Line28ae4c2015-03-13 11:37:03 -070020import com.googlecode.concurrenttrees.radix.node.concrete.DefaultByteArrayNodeFactory;
21import com.googlecode.concurrenttrees.radixinverted.ConcurrentInvertedRadixTree;
22import com.googlecode.concurrenttrees.radixinverted.InvertedRadixTree;
Jonathan Hart90a02c22015-02-13 11:52:07 -080023import org.apache.felix.scr.annotations.Activate;
24import org.apache.felix.scr.annotations.Component;
Jonathan Hart66018992015-07-31 11:19:27 -070025import org.apache.felix.scr.annotations.Deactivate;
Jonathan Hart90a02c22015-02-13 11:52:07 -080026import org.apache.felix.scr.annotations.Reference;
27import org.apache.felix.scr.annotations.ReferenceCardinality;
28import org.apache.felix.scr.annotations.Service;
Pingping Line28ae4c2015-03-13 11:37:03 -070029import org.onlab.packet.Ip4Address;
30import org.onlab.packet.Ip6Address;
Jonathan Hart90a02c22015-02-13 11:52:07 -080031import org.onlab.packet.IpAddress;
Pingping Line28ae4c2015-03-13 11:37:03 -070032import org.onlab.packet.IpPrefix;
Pingping Linc9e16bf2015-04-10 14:42:41 -070033import org.onlab.packet.MacAddress;
Jonathan Hart4cb39882015-08-12 23:50:55 -040034import org.onosproject.core.ApplicationId;
35import org.onosproject.core.CoreService;
36import org.onosproject.incubator.net.intf.InterfaceService;
37import org.onosproject.net.ConnectPoint;
Ray Milkeya4122362015-08-18 15:19:08 -070038import org.onosproject.net.config.ConfigFactory;
Pingping Lin9a445c82016-04-07 11:40:29 -070039import org.onosproject.net.config.NetworkConfigEvent;
40import org.onosproject.net.config.NetworkConfigListener;
Ray Milkeya4122362015-08-18 15:19:08 -070041import org.onosproject.net.config.NetworkConfigRegistry;
Jonathan Hart4cb39882015-08-12 23:50:55 -040042import org.onosproject.net.config.NetworkConfigService;
Thomas Vachuska4998caa2015-08-26 13:28:38 -070043import org.onosproject.net.config.basics.SubjectFactories;
Jonathan Hart1ad75f22016-04-13 21:24:13 -070044import org.onosproject.routing.RoutingService;
Jonathan Hart4cb39882015-08-12 23:50:55 -040045import org.onosproject.routing.config.BgpConfig;
Pingping Line28ae4c2015-03-13 11:37:03 -070046import org.onosproject.routing.config.LocalIpPrefixEntry;
Pingping Lin9a445c82016-04-07 11:40:29 -070047import org.onosproject.routing.config.ReactiveRoutingConfig;
48import org.onosproject.routing.config.RouterConfig;
Jonathan Hart2da1e602015-02-18 19:09:24 -080049import org.onosproject.routing.config.RoutingConfigurationService;
Jonathan Hart90a02c22015-02-13 11:52:07 -080050import org.slf4j.Logger;
51import org.slf4j.LoggerFactory;
Jonathan Hartbac07a02014-10-13 21:29:54 -070052
Jonathan Hart1ad75f22016-04-13 21:24:13 -070053import java.util.HashSet;
54import java.util.Objects;
55import java.util.Set;
56import java.util.stream.Collectors;
57
Jonathan Hartbac07a02014-10-13 21:29:54 -070058/**
Jonathan Hart90a02c22015-02-13 11:52:07 -080059 * Implementation of RoutingConfigurationService which reads routing
Jonathan Hartde6e2032016-04-15 13:28:02 -070060 * configuration from the network configuration service.
Jonathan Hartbac07a02014-10-13 21:29:54 -070061 */
Jonathan Hart90a02c22015-02-13 11:52:07 -080062@Component(immediate = true)
63@Service
64public class RoutingConfigurationImpl implements RoutingConfigurationService {
Jonathan Hartbac07a02014-10-13 21:29:54 -070065
Pavlin Radoslavov0a297b12014-11-11 16:03:27 -080066 private final Logger log = LoggerFactory.getLogger(getClass());
Jonathan Hartbac07a02014-10-13 21:29:54 -070067
Jonathan Hart90a02c22015-02-13 11:52:07 -080068 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart66018992015-07-31 11:19:27 -070069 protected NetworkConfigRegistry registry;
70
Jonathan Hart4cb39882015-08-12 23:50:55 -040071 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
72 protected NetworkConfigService configService;
73
74 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
75 protected CoreService coreService;
76
77 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
78 protected InterfaceService interfaceService;
79
Pingping Linc9e16bf2015-04-10 14:42:41 -070080 private Set<IpAddress> gatewayIpAddresses = new HashSet<>();
Pingping Lin8a524712015-06-24 14:58:24 -070081 private Set<ConnectPoint> bgpPeerConnectPoints = new HashSet<>();
Jonathan Hartbac07a02014-10-13 21:29:54 -070082
Pingping Line28ae4c2015-03-13 11:37:03 -070083 private InvertedRadixTree<LocalIpPrefixEntry>
84 localPrefixTable4 = new ConcurrentInvertedRadixTree<>(
85 new DefaultByteArrayNodeFactory());
86 private InvertedRadixTree<LocalIpPrefixEntry>
87 localPrefixTable6 = new ConcurrentInvertedRadixTree<>(
88 new DefaultByteArrayNodeFactory());
89
Pingping Linc9e16bf2015-04-10 14:42:41 -070090 private MacAddress virtualGatewayMacAddress;
Pingping Lin9a445c82016-04-07 11:40:29 -070091 private final InternalNetworkConfigListener configListener =
92 new InternalNetworkConfigListener();
Jonathan Hart90a02c22015-02-13 11:52:07 -080093
Jonathan Hart6344f572015-12-15 08:26:25 -080094 private ConfigFactory<ApplicationId, BgpConfig> bgpConfigFactory =
95 new ConfigFactory<ApplicationId, BgpConfig>(
96 SubjectFactories.APP_SUBJECT_FACTORY, BgpConfig.class, "bgp") {
Jonathan Hart66018992015-07-31 11:19:27 -070097 @Override
98 public BgpConfig createConfig() {
99 return new BgpConfig();
100 }
101 };
102
Jonathan Hart6344f572015-12-15 08:26:25 -0800103 private ConfigFactory<ApplicationId, RouterConfig> routerConfigFactory =
104 new ConfigFactory<ApplicationId, RouterConfig>(
105 SubjectFactories.APP_SUBJECT_FACTORY, RouterConfig.class, "router") {
Pingping Lin9a445c82016-04-07 11:40:29 -0700106 @Override
107 public RouterConfig createConfig() {
108 return new RouterConfig();
109 }
110 };
111
112 private ConfigFactory<ApplicationId, ReactiveRoutingConfig>
113 reactiveRoutingConfigFactory =
114 new ConfigFactory<ApplicationId, ReactiveRoutingConfig>(
115 SubjectFactories.APP_SUBJECT_FACTORY,
116 ReactiveRoutingConfig.class, "reactiveRouting") {
117 @Override
118 public ReactiveRoutingConfig createConfig() {
119 return new ReactiveRoutingConfig();
120 }
121 };
Jonathan Hart6344f572015-12-15 08:26:25 -0800122
Jonathan Hart90a02c22015-02-13 11:52:07 -0800123 @Activate
124 public void activate() {
Pingping Lin9a445c82016-04-07 11:40:29 -0700125 configService.addListener(configListener);
Jonathan Hart6344f572015-12-15 08:26:25 -0800126 registry.registerConfigFactory(bgpConfigFactory);
127 registry.registerConfigFactory(routerConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700128 registry.registerConfigFactory(reactiveRoutingConfigFactory);
129 setUpConfiguration();
Jonathan Hart90a02c22015-02-13 11:52:07 -0800130 log.info("Routing configuration service started");
131 }
132
Jonathan Hart66018992015-07-31 11:19:27 -0700133 @Deactivate
134 public void deactivate() {
Jonathan Hart6344f572015-12-15 08:26:25 -0800135 registry.unregisterConfigFactory(bgpConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700136 registry.unregisterConfigFactory(routerConfigFactory);
137 registry.unregisterConfigFactory(reactiveRoutingConfigFactory);
138 configService.removeListener(configListener);
Jonathan Hart66018992015-07-31 11:19:27 -0700139 log.info("Routing configuration service stopped");
140 }
141
Jonathan Hartbac07a02014-10-13 21:29:54 -0700142 /**
Pingping Lin9a445c82016-04-07 11:40:29 -0700143 * Set up reactive routing information from configuration.
Jonathan Hartbac07a02014-10-13 21:29:54 -0700144 */
Pingping Lin9a445c82016-04-07 11:40:29 -0700145 private void setUpConfiguration() {
146 ReactiveRoutingConfig config = configService.getConfig(
147 coreService.registerApplication(RoutingConfigurationService
148 .REACTIVE_ROUTING_APP_ID),
149 RoutingConfigurationService.CONFIG_CLASS);
150 if (config == null) {
151 log.warn("No reactive routing config available!");
152 return;
Jonathan Hartbac07a02014-10-13 21:29:54 -0700153 }
Pingping Lin9a445c82016-04-07 11:40:29 -0700154 for (LocalIpPrefixEntry entry : config.localIp4PrefixEntries()) {
155 localPrefixTable4.put(createBinaryString(entry.ipPrefix()), entry);
156 gatewayIpAddresses.add(entry.getGatewayIpAddress());
157 }
158 for (LocalIpPrefixEntry entry : config.localIp6PrefixEntries()) {
159 localPrefixTable6.put(createBinaryString(entry.ipPrefix()), entry);
160 gatewayIpAddresses.add(entry.getGatewayIpAddress());
161 }
Jonathan Hartbac07a02014-10-13 21:29:54 -0700162
Pingping Lin9a445c82016-04-07 11:40:29 -0700163 virtualGatewayMacAddress = config.virtualGatewayMacAddress();
Jonathan Hartbac07a02014-10-13 21:29:54 -0700164
Pingping Lin9a445c82016-04-07 11:40:29 -0700165 // Setup BGP peer connect points
Jonathan Hart1ad75f22016-04-13 21:24:13 -0700166 ApplicationId routerAppId = coreService.getAppId(RoutingService.ROUTER_APP_ID);
Jonathan Hart4cb39882015-08-12 23:50:55 -0400167 if (routerAppId == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700168 log.info("Router application ID is null!");
169 return;
Jonathan Hart4cb39882015-08-12 23:50:55 -0400170 }
171
172 BgpConfig bgpConfig = configService.getConfig(routerAppId, BgpConfig.class);
Pingping Lin9a445c82016-04-07 11:40:29 -0700173
Pingping Lin9b85c032015-10-05 18:16:27 -0700174 if (bgpConfig == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700175 log.info("BGP config is null!");
176 return;
Pingping Lin9b85c032015-10-05 18:16:27 -0700177 } else {
Pingping Lin9a445c82016-04-07 11:40:29 -0700178 bgpPeerConnectPoints =
179 bgpConfig.bgpSpeakers().stream()
Pingping Lin9b85c032015-10-05 18:16:27 -0700180 .flatMap(speaker -> speaker.peers().stream())
181 .map(peer -> interfaceService.getMatchingInterface(peer))
Sho SHIMIZU45906042016-01-13 23:05:54 -0800182 .filter(Objects::nonNull)
Pingping Lin9b85c032015-10-05 18:16:27 -0700183 .map(intf -> intf.connectPoint())
184 .collect(Collectors.toSet());
185 }
Pingping Lin8a524712015-06-24 14:58:24 -0700186 }
187
188 @Override
Pingping Line28ae4c2015-03-13 11:37:03 -0700189 public boolean isIpAddressLocal(IpAddress ipAddress) {
190 if (ipAddress.isIp4()) {
191 return localPrefixTable4.getValuesForKeysPrefixing(
192 createBinaryString(
193 IpPrefix.valueOf(ipAddress, Ip4Address.BIT_LENGTH)))
194 .iterator().hasNext();
195 } else {
196 return localPrefixTable6.getValuesForKeysPrefixing(
197 createBinaryString(
198 IpPrefix.valueOf(ipAddress, Ip6Address.BIT_LENGTH)))
199 .iterator().hasNext();
200 }
201 }
202
203 @Override
204 public boolean isIpPrefixLocal(IpPrefix ipPrefix) {
205 return (localPrefixTable4.getValueForExactKey(
206 createBinaryString(ipPrefix)) != null ||
207 localPrefixTable6.getValueForExactKey(
208 createBinaryString(ipPrefix)) != null);
209 }
210
Pingping Linc9e16bf2015-04-10 14:42:41 -0700211 @Override
212 public boolean isVirtualGatewayIpAddress(IpAddress ipAddress) {
213 return gatewayIpAddresses.contains(ipAddress);
214 }
215
216 @Override
217 public MacAddress getVirtualGatewayMacAddress() {
218 return virtualGatewayMacAddress;
219 }
220
Jonathan Hartc9e36c52017-01-05 09:53:33 +1300221 @Override
222 public Set<ConnectPoint> getBgpPeerConnectPoints() {
223 return ImmutableSet.copyOf(bgpPeerConnectPoints);
224 }
225
226 /**
227 * Creates the binary string representation of an IP prefix.
228 * The prefix can be either IPv4 or IPv6.
229 * The string length is equal to the prefix length + 1.
230 *
231 * For each string, we put a extra "0" in the front. The purpose of
232 * doing this is to store the default route inside InvertedRadixTree.
233 *
234 * @param ipPrefix the IP prefix to use
235 * @return the binary string representation
236 */
237 private static String createBinaryString(IpPrefix ipPrefix) {
238 if (ipPrefix.prefixLength() == 0) {
239 return "0";
240 }
241
242 byte[] octets = ipPrefix.address().toOctets();
243 StringBuilder result = new StringBuilder(ipPrefix.prefixLength());
244 for (int i = 0; i < ipPrefix.prefixLength(); i++) {
245 int byteOffset = i / Byte.SIZE;
246 int bitOffset = i % Byte.SIZE;
247 int mask = 1 << (Byte.SIZE - 1 - bitOffset);
248 byte value = octets[byteOffset];
249 boolean isSet = ((value & mask) != 0);
250 result.append(isSet ? "1" : "0");
251 }
252
253 return "0" + result.toString();
254 }
255
Pingping Lin9a445c82016-04-07 11:40:29 -0700256 private class InternalNetworkConfigListener implements NetworkConfigListener {
257
258 @Override
259 public void event(NetworkConfigEvent event) {
260 switch (event.type()) {
261 case CONFIG_REGISTERED:
262 break;
263 case CONFIG_UNREGISTERED:
264 break;
265 case CONFIG_ADDED:
266 case CONFIG_UPDATED:
267 case CONFIG_REMOVED:
268 if (event.configClass() == RoutingConfigurationService.CONFIG_CLASS) {
269 setUpConfiguration();
270 }
271 break;
272 default:
273 break;
274 }
275 }
276 }
Jonathan Hartbac07a02014-10-13 21:29:54 -0700277}