blob: 5fbfe7f48f0946c5dd7f4cb61f46dd383b79c0b0 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-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 Hart2da1e602015-02-18 19:09:24 -080016package org.onosproject.routing.config.impl;
Jonathan Hart90a02c22015-02-13 11:52:07 -080017
Pingping Lin9a445c82016-04-07 11:40:29 -070018import com.google.common.collect.ImmutableSet;
Pingping Line28ae4c2015-03-13 11:37:03 -070019import com.googlecode.concurrenttrees.radix.node.concrete.DefaultByteArrayNodeFactory;
20import com.googlecode.concurrenttrees.radixinverted.ConcurrentInvertedRadixTree;
21import com.googlecode.concurrenttrees.radixinverted.InvertedRadixTree;
Jonathan Hart90a02c22015-02-13 11:52:07 -080022import org.apache.felix.scr.annotations.Activate;
23import org.apache.felix.scr.annotations.Component;
Jonathan Hart66018992015-07-31 11:19:27 -070024import org.apache.felix.scr.annotations.Deactivate;
Jonathan Hart90a02c22015-02-13 11:52:07 -080025import org.apache.felix.scr.annotations.Reference;
26import org.apache.felix.scr.annotations.ReferenceCardinality;
27import org.apache.felix.scr.annotations.Service;
Pingping Line28ae4c2015-03-13 11:37:03 -070028import org.onlab.packet.Ip4Address;
29import org.onlab.packet.Ip6Address;
Jonathan Hart90a02c22015-02-13 11:52:07 -080030import org.onlab.packet.IpAddress;
Pingping Line28ae4c2015-03-13 11:37:03 -070031import org.onlab.packet.IpPrefix;
Pingping Linc9e16bf2015-04-10 14:42:41 -070032import org.onlab.packet.MacAddress;
Jonathan Hart4cb39882015-08-12 23:50:55 -040033import org.onosproject.core.ApplicationId;
34import org.onosproject.core.CoreService;
35import org.onosproject.incubator.net.intf.InterfaceService;
36import org.onosproject.net.ConnectPoint;
Ray Milkeya4122362015-08-18 15:19:08 -070037import org.onosproject.net.config.ConfigFactory;
Pingping Lin9a445c82016-04-07 11:40:29 -070038import org.onosproject.net.config.NetworkConfigEvent;
39import org.onosproject.net.config.NetworkConfigListener;
Ray Milkeya4122362015-08-18 15:19:08 -070040import org.onosproject.net.config.NetworkConfigRegistry;
Jonathan Hart4cb39882015-08-12 23:50:55 -040041import org.onosproject.net.config.NetworkConfigService;
Thomas Vachuska4998caa2015-08-26 13:28:38 -070042import org.onosproject.net.config.basics.SubjectFactories;
Jonathan Hart1ad75f22016-04-13 21:24:13 -070043import org.onosproject.routing.RoutingService;
Jonathan Hart4cb39882015-08-12 23:50:55 -040044import org.onosproject.routing.config.BgpConfig;
Pingping Line28ae4c2015-03-13 11:37:03 -070045import org.onosproject.routing.config.LocalIpPrefixEntry;
Pingping Lin9a445c82016-04-07 11:40:29 -070046import org.onosproject.routing.config.ReactiveRoutingConfig;
47import org.onosproject.routing.config.RouterConfig;
Jonathan Hart2da1e602015-02-18 19:09:24 -080048import org.onosproject.routing.config.RoutingConfigurationService;
Jonathan Hart90a02c22015-02-13 11:52:07 -080049import org.slf4j.Logger;
50import org.slf4j.LoggerFactory;
Jonathan Hartbac07a02014-10-13 21:29:54 -070051
Jonathan Hart1ad75f22016-04-13 21:24:13 -070052import java.util.HashSet;
53import java.util.Objects;
54import java.util.Set;
55import java.util.stream.Collectors;
56
Jonathan Hartbac07a02014-10-13 21:29:54 -070057/**
Jonathan Hart90a02c22015-02-13 11:52:07 -080058 * Implementation of RoutingConfigurationService which reads routing
Jonathan Hartde6e2032016-04-15 13:28:02 -070059 * configuration from the network configuration service.
Jonathan Hartbac07a02014-10-13 21:29:54 -070060 */
Jonathan Hart90a02c22015-02-13 11:52:07 -080061@Component(immediate = true)
62@Service
63public class RoutingConfigurationImpl implements RoutingConfigurationService {
Jonathan Hartbac07a02014-10-13 21:29:54 -070064
Pavlin Radoslavov0a297b12014-11-11 16:03:27 -080065 private final Logger log = LoggerFactory.getLogger(getClass());
Jonathan Hartbac07a02014-10-13 21:29:54 -070066
Jonathan Hart90a02c22015-02-13 11:52:07 -080067 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart66018992015-07-31 11:19:27 -070068 protected NetworkConfigRegistry registry;
69
Jonathan Hart4cb39882015-08-12 23:50:55 -040070 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
71 protected NetworkConfigService configService;
72
73 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
74 protected CoreService coreService;
75
76 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
77 protected InterfaceService interfaceService;
78
Pingping Linc9e16bf2015-04-10 14:42:41 -070079 private Set<IpAddress> gatewayIpAddresses = new HashSet<>();
Pingping Lin8a524712015-06-24 14:58:24 -070080 private Set<ConnectPoint> bgpPeerConnectPoints = new HashSet<>();
Jonathan Hartbac07a02014-10-13 21:29:54 -070081
Pingping Line28ae4c2015-03-13 11:37:03 -070082 private InvertedRadixTree<LocalIpPrefixEntry>
83 localPrefixTable4 = new ConcurrentInvertedRadixTree<>(
84 new DefaultByteArrayNodeFactory());
85 private InvertedRadixTree<LocalIpPrefixEntry>
86 localPrefixTable6 = new ConcurrentInvertedRadixTree<>(
87 new DefaultByteArrayNodeFactory());
88
Pingping Linc9e16bf2015-04-10 14:42:41 -070089 private MacAddress virtualGatewayMacAddress;
Pingping Lin9a445c82016-04-07 11:40:29 -070090 private final InternalNetworkConfigListener configListener =
91 new InternalNetworkConfigListener();
Jonathan Hart90a02c22015-02-13 11:52:07 -080092
Jonathan Hart6344f572015-12-15 08:26:25 -080093 private ConfigFactory<ApplicationId, BgpConfig> bgpConfigFactory =
94 new ConfigFactory<ApplicationId, BgpConfig>(
95 SubjectFactories.APP_SUBJECT_FACTORY, BgpConfig.class, "bgp") {
Jonathan Hart66018992015-07-31 11:19:27 -070096 @Override
97 public BgpConfig createConfig() {
98 return new BgpConfig();
99 }
100 };
101
Jonathan Hart6344f572015-12-15 08:26:25 -0800102 private ConfigFactory<ApplicationId, RouterConfig> routerConfigFactory =
103 new ConfigFactory<ApplicationId, RouterConfig>(
104 SubjectFactories.APP_SUBJECT_FACTORY, RouterConfig.class, "router") {
Pingping Lin9a445c82016-04-07 11:40:29 -0700105 @Override
106 public RouterConfig createConfig() {
107 return new RouterConfig();
108 }
109 };
110
111 private ConfigFactory<ApplicationId, ReactiveRoutingConfig>
112 reactiveRoutingConfigFactory =
113 new ConfigFactory<ApplicationId, ReactiveRoutingConfig>(
114 SubjectFactories.APP_SUBJECT_FACTORY,
115 ReactiveRoutingConfig.class, "reactiveRouting") {
116 @Override
117 public ReactiveRoutingConfig createConfig() {
118 return new ReactiveRoutingConfig();
119 }
120 };
Jonathan Hart6344f572015-12-15 08:26:25 -0800121
Jonathan Hart90a02c22015-02-13 11:52:07 -0800122 @Activate
123 public void activate() {
Pingping Lin9a445c82016-04-07 11:40:29 -0700124 configService.addListener(configListener);
Jonathan Hart6344f572015-12-15 08:26:25 -0800125 registry.registerConfigFactory(bgpConfigFactory);
126 registry.registerConfigFactory(routerConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700127 registry.registerConfigFactory(reactiveRoutingConfigFactory);
128 setUpConfiguration();
Jonathan Hart90a02c22015-02-13 11:52:07 -0800129 log.info("Routing configuration service started");
130 }
131
Jonathan Hart66018992015-07-31 11:19:27 -0700132 @Deactivate
133 public void deactivate() {
Jonathan Hart6344f572015-12-15 08:26:25 -0800134 registry.unregisterConfigFactory(bgpConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700135 registry.unregisterConfigFactory(routerConfigFactory);
136 registry.unregisterConfigFactory(reactiveRoutingConfigFactory);
137 configService.removeListener(configListener);
Jonathan Hart66018992015-07-31 11:19:27 -0700138 log.info("Routing configuration service stopped");
139 }
140
Jonathan Hartbac07a02014-10-13 21:29:54 -0700141 /**
Pingping Lin9a445c82016-04-07 11:40:29 -0700142 * Set up reactive routing information from configuration.
Jonathan Hartbac07a02014-10-13 21:29:54 -0700143 */
Pingping Lin9a445c82016-04-07 11:40:29 -0700144 private void setUpConfiguration() {
145 ReactiveRoutingConfig config = configService.getConfig(
146 coreService.registerApplication(RoutingConfigurationService
147 .REACTIVE_ROUTING_APP_ID),
148 RoutingConfigurationService.CONFIG_CLASS);
149 if (config == null) {
150 log.warn("No reactive routing config available!");
151 return;
Jonathan Hartbac07a02014-10-13 21:29:54 -0700152 }
Pingping Lin9a445c82016-04-07 11:40:29 -0700153 for (LocalIpPrefixEntry entry : config.localIp4PrefixEntries()) {
154 localPrefixTable4.put(createBinaryString(entry.ipPrefix()), entry);
155 gatewayIpAddresses.add(entry.getGatewayIpAddress());
156 }
157 for (LocalIpPrefixEntry entry : config.localIp6PrefixEntries()) {
158 localPrefixTable6.put(createBinaryString(entry.ipPrefix()), entry);
159 gatewayIpAddresses.add(entry.getGatewayIpAddress());
160 }
Jonathan Hartbac07a02014-10-13 21:29:54 -0700161
Pingping Lin9a445c82016-04-07 11:40:29 -0700162 virtualGatewayMacAddress = config.virtualGatewayMacAddress();
Jonathan Hartbac07a02014-10-13 21:29:54 -0700163
Pingping Lin9a445c82016-04-07 11:40:29 -0700164 // Setup BGP peer connect points
Jonathan Hart1ad75f22016-04-13 21:24:13 -0700165 ApplicationId routerAppId = coreService.getAppId(RoutingService.ROUTER_APP_ID);
Jonathan Hart4cb39882015-08-12 23:50:55 -0400166 if (routerAppId == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700167 log.info("Router application ID is null!");
168 return;
Jonathan Hart4cb39882015-08-12 23:50:55 -0400169 }
170
171 BgpConfig bgpConfig = configService.getConfig(routerAppId, BgpConfig.class);
Pingping Lin9a445c82016-04-07 11:40:29 -0700172
Pingping Lin9b85c032015-10-05 18:16:27 -0700173 if (bgpConfig == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700174 log.info("BGP config is null!");
175 return;
Pingping Lin9b85c032015-10-05 18:16:27 -0700176 } else {
Pingping Lin9a445c82016-04-07 11:40:29 -0700177 bgpPeerConnectPoints =
178 bgpConfig.bgpSpeakers().stream()
Pingping Lin9b85c032015-10-05 18:16:27 -0700179 .flatMap(speaker -> speaker.peers().stream())
180 .map(peer -> interfaceService.getMatchingInterface(peer))
Sho SHIMIZU45906042016-01-13 23:05:54 -0800181 .filter(Objects::nonNull)
Pingping Lin9b85c032015-10-05 18:16:27 -0700182 .map(intf -> intf.connectPoint())
183 .collect(Collectors.toSet());
184 }
Pingping Lin8a524712015-06-24 14:58:24 -0700185 }
186
187 @Override
Pingping Line28ae4c2015-03-13 11:37:03 -0700188 public boolean isIpAddressLocal(IpAddress ipAddress) {
189 if (ipAddress.isIp4()) {
190 return localPrefixTable4.getValuesForKeysPrefixing(
191 createBinaryString(
192 IpPrefix.valueOf(ipAddress, Ip4Address.BIT_LENGTH)))
193 .iterator().hasNext();
194 } else {
195 return localPrefixTable6.getValuesForKeysPrefixing(
196 createBinaryString(
197 IpPrefix.valueOf(ipAddress, Ip6Address.BIT_LENGTH)))
198 .iterator().hasNext();
199 }
200 }
201
202 @Override
203 public boolean isIpPrefixLocal(IpPrefix ipPrefix) {
204 return (localPrefixTable4.getValueForExactKey(
205 createBinaryString(ipPrefix)) != null ||
206 localPrefixTable6.getValueForExactKey(
207 createBinaryString(ipPrefix)) != null);
208 }
209
Pingping Linc9e16bf2015-04-10 14:42:41 -0700210 @Override
211 public boolean isVirtualGatewayIpAddress(IpAddress ipAddress) {
212 return gatewayIpAddresses.contains(ipAddress);
213 }
214
215 @Override
216 public MacAddress getVirtualGatewayMacAddress() {
217 return virtualGatewayMacAddress;
218 }
219
Jonathan Hartc9e36c52017-01-05 09:53:33 +1300220 @Override
221 public Set<ConnectPoint> getBgpPeerConnectPoints() {
222 return ImmutableSet.copyOf(bgpPeerConnectPoints);
223 }
224
225 /**
226 * Creates the binary string representation of an IP prefix.
227 * The prefix can be either IPv4 or IPv6.
228 * The string length is equal to the prefix length + 1.
229 *
230 * For each string, we put a extra "0" in the front. The purpose of
231 * doing this is to store the default route inside InvertedRadixTree.
232 *
233 * @param ipPrefix the IP prefix to use
234 * @return the binary string representation
235 */
236 private static String createBinaryString(IpPrefix ipPrefix) {
237 if (ipPrefix.prefixLength() == 0) {
238 return "0";
239 }
240
241 byte[] octets = ipPrefix.address().toOctets();
242 StringBuilder result = new StringBuilder(ipPrefix.prefixLength());
243 for (int i = 0; i < ipPrefix.prefixLength(); i++) {
244 int byteOffset = i / Byte.SIZE;
245 int bitOffset = i % Byte.SIZE;
246 int mask = 1 << (Byte.SIZE - 1 - bitOffset);
247 byte value = octets[byteOffset];
248 boolean isSet = ((value & mask) != 0);
249 result.append(isSet ? "1" : "0");
250 }
251
252 return "0" + result.toString();
253 }
254
Pingping Lin9a445c82016-04-07 11:40:29 -0700255 private class InternalNetworkConfigListener implements NetworkConfigListener {
256
257 @Override
258 public void event(NetworkConfigEvent event) {
259 switch (event.type()) {
260 case CONFIG_REGISTERED:
261 break;
262 case CONFIG_UNREGISTERED:
263 break;
264 case CONFIG_ADDED:
265 case CONFIG_UPDATED:
266 case CONFIG_REMOVED:
267 if (event.configClass() == RoutingConfigurationService.CONFIG_CLASS) {
268 setUpConfiguration();
269 }
270 break;
271 default:
272 break;
273 }
274 }
275 }
Jonathan Hartbac07a02014-10-13 21:29:54 -0700276}