blob: 9a5f0968183d485fd1daa95aeb21d7ab2baa097f [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Ray Milkey34c95902015-04-15 09:47:53 -07002 * Copyright 2014-2015 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 static org.onosproject.routing.RouteEntry.createBinaryString;
19
20import com.google.common.collect.ImmutableSet;
Pingping Line28ae4c2015-03-13 11:37:03 -070021import com.googlecode.concurrenttrees.radix.node.concrete.DefaultByteArrayNodeFactory;
22import com.googlecode.concurrenttrees.radixinverted.ConcurrentInvertedRadixTree;
23import com.googlecode.concurrenttrees.radixinverted.InvertedRadixTree;
Pingping Lin9a445c82016-04-07 11:40:29 -070024
25import java.util.HashSet;
26import java.util.Objects;
27import java.util.Set;
28import java.util.stream.Collectors;
29
Jonathan Hart90a02c22015-02-13 11:52:07 -080030import org.apache.felix.scr.annotations.Activate;
31import org.apache.felix.scr.annotations.Component;
Jonathan Hart66018992015-07-31 11:19:27 -070032import org.apache.felix.scr.annotations.Deactivate;
Jonathan Hart90a02c22015-02-13 11:52:07 -080033import org.apache.felix.scr.annotations.Reference;
34import org.apache.felix.scr.annotations.ReferenceCardinality;
35import org.apache.felix.scr.annotations.Service;
Pingping Line28ae4c2015-03-13 11:37:03 -070036import org.onlab.packet.Ip4Address;
37import org.onlab.packet.Ip6Address;
Jonathan Hart90a02c22015-02-13 11:52:07 -080038import org.onlab.packet.IpAddress;
Pingping Line28ae4c2015-03-13 11:37:03 -070039import org.onlab.packet.IpPrefix;
Pingping Linc9e16bf2015-04-10 14:42:41 -070040import org.onlab.packet.MacAddress;
Jonathan Hart4cb39882015-08-12 23:50:55 -040041import org.onosproject.core.ApplicationId;
42import org.onosproject.core.CoreService;
43import org.onosproject.incubator.net.intf.InterfaceService;
44import org.onosproject.net.ConnectPoint;
Ray Milkeya4122362015-08-18 15:19:08 -070045import org.onosproject.net.config.ConfigFactory;
Pingping Lin9a445c82016-04-07 11:40:29 -070046import org.onosproject.net.config.NetworkConfigEvent;
47import org.onosproject.net.config.NetworkConfigListener;
Ray Milkeya4122362015-08-18 15:19:08 -070048import org.onosproject.net.config.NetworkConfigRegistry;
Jonathan Hart4cb39882015-08-12 23:50:55 -040049import org.onosproject.net.config.NetworkConfigService;
Thomas Vachuska4998caa2015-08-26 13:28:38 -070050import org.onosproject.net.config.basics.SubjectFactories;
Jonathan Hart4cb39882015-08-12 23:50:55 -040051import org.onosproject.routing.config.BgpConfig;
Pingping Line28ae4c2015-03-13 11:37:03 -070052import org.onosproject.routing.config.LocalIpPrefixEntry;
Pingping Lin9a445c82016-04-07 11:40:29 -070053import org.onosproject.routing.config.ReactiveRoutingConfig;
54import org.onosproject.routing.config.RouterConfig;
Jonathan Hart2da1e602015-02-18 19:09:24 -080055import org.onosproject.routing.config.RoutingConfigurationService;
Jonathan Hart4cb39882015-08-12 23:50:55 -040056import org.onosproject.routing.impl.Router;
Jonathan Hart90a02c22015-02-13 11:52:07 -080057import org.slf4j.Logger;
58import org.slf4j.LoggerFactory;
Jonathan Hartbac07a02014-10-13 21:29:54 -070059
Jonathan Hartbac07a02014-10-13 21:29:54 -070060/**
Jonathan Hart90a02c22015-02-13 11:52:07 -080061 * Implementation of RoutingConfigurationService which reads routing
62 * configuration from a file.
Jonathan Hartbac07a02014-10-13 21:29:54 -070063 */
Jonathan Hart90a02c22015-02-13 11:52:07 -080064@Component(immediate = true)
65@Service
66public class RoutingConfigurationImpl implements RoutingConfigurationService {
Jonathan Hartbac07a02014-10-13 21:29:54 -070067
Pavlin Radoslavov0a297b12014-11-11 16:03:27 -080068 private final Logger log = LoggerFactory.getLogger(getClass());
Jonathan Hartbac07a02014-10-13 21:29:54 -070069
Jonathan Hart90a02c22015-02-13 11:52:07 -080070 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart66018992015-07-31 11:19:27 -070071 protected NetworkConfigRegistry registry;
72
Jonathan Hart4cb39882015-08-12 23:50:55 -040073 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
74 protected NetworkConfigService configService;
75
76 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
77 protected CoreService coreService;
78
79 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
80 protected InterfaceService interfaceService;
81
Pingping Linc9e16bf2015-04-10 14:42:41 -070082 private Set<IpAddress> gatewayIpAddresses = new HashSet<>();
Pingping Lin8a524712015-06-24 14:58:24 -070083 private Set<ConnectPoint> bgpPeerConnectPoints = new HashSet<>();
Jonathan Hartbac07a02014-10-13 21:29:54 -070084
Pingping Line28ae4c2015-03-13 11:37:03 -070085 private InvertedRadixTree<LocalIpPrefixEntry>
86 localPrefixTable4 = new ConcurrentInvertedRadixTree<>(
87 new DefaultByteArrayNodeFactory());
88 private InvertedRadixTree<LocalIpPrefixEntry>
89 localPrefixTable6 = new ConcurrentInvertedRadixTree<>(
90 new DefaultByteArrayNodeFactory());
91
Pingping Linc9e16bf2015-04-10 14:42:41 -070092 private MacAddress virtualGatewayMacAddress;
Pingping Lin9a445c82016-04-07 11:40:29 -070093 private final InternalNetworkConfigListener configListener =
94 new InternalNetworkConfigListener();
Jonathan Hart90a02c22015-02-13 11:52:07 -080095
Jonathan Hart6344f572015-12-15 08:26:25 -080096 private ConfigFactory<ApplicationId, BgpConfig> bgpConfigFactory =
97 new ConfigFactory<ApplicationId, BgpConfig>(
98 SubjectFactories.APP_SUBJECT_FACTORY, BgpConfig.class, "bgp") {
Jonathan Hart66018992015-07-31 11:19:27 -070099 @Override
100 public BgpConfig createConfig() {
101 return new BgpConfig();
102 }
103 };
104
Jonathan Hart6344f572015-12-15 08:26:25 -0800105 private ConfigFactory<ApplicationId, RouterConfig> routerConfigFactory =
106 new ConfigFactory<ApplicationId, RouterConfig>(
107 SubjectFactories.APP_SUBJECT_FACTORY, RouterConfig.class, "router") {
Pingping Lin9a445c82016-04-07 11:40:29 -0700108 @Override
109 public RouterConfig createConfig() {
110 return new RouterConfig();
111 }
112 };
113
114 private ConfigFactory<ApplicationId, ReactiveRoutingConfig>
115 reactiveRoutingConfigFactory =
116 new ConfigFactory<ApplicationId, ReactiveRoutingConfig>(
117 SubjectFactories.APP_SUBJECT_FACTORY,
118 ReactiveRoutingConfig.class, "reactiveRouting") {
119 @Override
120 public ReactiveRoutingConfig createConfig() {
121 return new ReactiveRoutingConfig();
122 }
123 };
Jonathan Hart6344f572015-12-15 08:26:25 -0800124
Jonathan Hart90a02c22015-02-13 11:52:07 -0800125 @Activate
126 public void activate() {
Pingping Lin9a445c82016-04-07 11:40:29 -0700127 configService.addListener(configListener);
Jonathan Hart6344f572015-12-15 08:26:25 -0800128 registry.registerConfigFactory(bgpConfigFactory);
129 registry.registerConfigFactory(routerConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700130 registry.registerConfigFactory(reactiveRoutingConfigFactory);
131 setUpConfiguration();
Jonathan Hart90a02c22015-02-13 11:52:07 -0800132 log.info("Routing configuration service started");
133 }
134
Jonathan Hart66018992015-07-31 11:19:27 -0700135 @Deactivate
136 public void deactivate() {
Jonathan Hart6344f572015-12-15 08:26:25 -0800137 registry.unregisterConfigFactory(bgpConfigFactory);
Pingping Lin9a445c82016-04-07 11:40:29 -0700138 registry.unregisterConfigFactory(routerConfigFactory);
139 registry.unregisterConfigFactory(reactiveRoutingConfigFactory);
140 configService.removeListener(configListener);
Jonathan Hart66018992015-07-31 11:19:27 -0700141 log.info("Routing configuration service stopped");
142 }
143
Jonathan Hartbac07a02014-10-13 21:29:54 -0700144 /**
Pingping Lin9a445c82016-04-07 11:40:29 -0700145 * Set up reactive routing information from configuration.
Jonathan Hartbac07a02014-10-13 21:29:54 -0700146 */
Pingping Lin9a445c82016-04-07 11:40:29 -0700147 private void setUpConfiguration() {
148 ReactiveRoutingConfig config = configService.getConfig(
149 coreService.registerApplication(RoutingConfigurationService
150 .REACTIVE_ROUTING_APP_ID),
151 RoutingConfigurationService.CONFIG_CLASS);
152 if (config == null) {
153 log.warn("No reactive routing config available!");
154 return;
Jonathan Hartbac07a02014-10-13 21:29:54 -0700155 }
Pingping Lin9a445c82016-04-07 11:40:29 -0700156 for (LocalIpPrefixEntry entry : config.localIp4PrefixEntries()) {
157 localPrefixTable4.put(createBinaryString(entry.ipPrefix()), entry);
158 gatewayIpAddresses.add(entry.getGatewayIpAddress());
159 }
160 for (LocalIpPrefixEntry entry : config.localIp6PrefixEntries()) {
161 localPrefixTable6.put(createBinaryString(entry.ipPrefix()), entry);
162 gatewayIpAddresses.add(entry.getGatewayIpAddress());
163 }
Jonathan Hartbac07a02014-10-13 21:29:54 -0700164
Pingping Lin9a445c82016-04-07 11:40:29 -0700165 virtualGatewayMacAddress = config.virtualGatewayMacAddress();
Jonathan Hartbac07a02014-10-13 21:29:54 -0700166
Pingping Lin9a445c82016-04-07 11:40:29 -0700167 // Setup BGP peer connect points
Jonathan Hart4cb39882015-08-12 23:50:55 -0400168 ApplicationId routerAppId = coreService.getAppId(Router.ROUTER_APP_ID);
169 if (routerAppId == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700170 log.info("Router application ID is null!");
171 return;
Jonathan Hart4cb39882015-08-12 23:50:55 -0400172 }
173
174 BgpConfig bgpConfig = configService.getConfig(routerAppId, BgpConfig.class);
Pingping Lin9a445c82016-04-07 11:40:29 -0700175
Pingping Lin9b85c032015-10-05 18:16:27 -0700176 if (bgpConfig == null) {
Pingping Lin9a445c82016-04-07 11:40:29 -0700177 log.info("BGP config is null!");
178 return;
Pingping Lin9b85c032015-10-05 18:16:27 -0700179 } else {
Pingping Lin9a445c82016-04-07 11:40:29 -0700180 bgpPeerConnectPoints =
181 bgpConfig.bgpSpeakers().stream()
Pingping Lin9b85c032015-10-05 18:16:27 -0700182 .flatMap(speaker -> speaker.peers().stream())
183 .map(peer -> interfaceService.getMatchingInterface(peer))
Sho SHIMIZU45906042016-01-13 23:05:54 -0800184 .filter(Objects::nonNull)
Pingping Lin9b85c032015-10-05 18:16:27 -0700185 .map(intf -> intf.connectPoint())
186 .collect(Collectors.toSet());
187 }
Pingping Lin8a524712015-06-24 14:58:24 -0700188 }
189
190 @Override
Pingping Line28ae4c2015-03-13 11:37:03 -0700191 public boolean isIpAddressLocal(IpAddress ipAddress) {
192 if (ipAddress.isIp4()) {
193 return localPrefixTable4.getValuesForKeysPrefixing(
194 createBinaryString(
195 IpPrefix.valueOf(ipAddress, Ip4Address.BIT_LENGTH)))
196 .iterator().hasNext();
197 } else {
198 return localPrefixTable6.getValuesForKeysPrefixing(
199 createBinaryString(
200 IpPrefix.valueOf(ipAddress, Ip6Address.BIT_LENGTH)))
201 .iterator().hasNext();
202 }
203 }
204
205 @Override
206 public boolean isIpPrefixLocal(IpPrefix ipPrefix) {
207 return (localPrefixTable4.getValueForExactKey(
208 createBinaryString(ipPrefix)) != null ||
209 localPrefixTable6.getValueForExactKey(
210 createBinaryString(ipPrefix)) != null);
211 }
212
Pingping Linc9e16bf2015-04-10 14:42:41 -0700213 @Override
214 public boolean isVirtualGatewayIpAddress(IpAddress ipAddress) {
215 return gatewayIpAddresses.contains(ipAddress);
216 }
217
218 @Override
219 public MacAddress getVirtualGatewayMacAddress() {
220 return virtualGatewayMacAddress;
221 }
222
Pingping Lin9a445c82016-04-07 11:40:29 -0700223 private class InternalNetworkConfigListener implements NetworkConfigListener {
224
225 @Override
226 public void event(NetworkConfigEvent event) {
227 switch (event.type()) {
228 case CONFIG_REGISTERED:
229 break;
230 case CONFIG_UNREGISTERED:
231 break;
232 case CONFIG_ADDED:
233 case CONFIG_UPDATED:
234 case CONFIG_REMOVED:
235 if (event.configClass() == RoutingConfigurationService.CONFIG_CLASS) {
236 setUpConfiguration();
237 }
238 break;
239 default:
240 break;
241 }
242 }
243 }
244
245 @Override
246 public Set<ConnectPoint> getBgpPeerConnectPoints() {
247 return ImmutableSet.copyOf(bgpPeerConnectPoints);
248 }
249
Jonathan Hartbac07a02014-10-13 21:29:54 -0700250}