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.openflowplugin.openflow.md.core.sal;
10 import com.google.common.base.Preconditions;
11 import java.math.BigInteger;
12 import java.net.Inet4Address;
13 import java.net.Inet6Address;
14 import java.net.InetAddress;
15 import java.net.InetSocketAddress;
16 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
17 import org.opendaylight.controller.sal.binding.api.NotificationProviderService;
18 import org.opendaylight.controller.sal.binding.api.RpcProviderRegistry;
19 import org.opendaylight.openflowplugin.api.openflow.md.ModelDrivenSwitch;
20 import org.opendaylight.openflowplugin.api.openflow.md.core.NotificationQueueWrapper;
21 import org.opendaylight.openflowplugin.api.openflow.md.core.session.SessionContext;
22 import org.opendaylight.openflowplugin.api.openflow.md.core.session.SessionListener;
23 import org.opendaylight.openflowplugin.api.openflow.md.core.session.SessionManager;
24 import org.opendaylight.openflowplugin.api.openflow.md.core.session.SwitchSessionKeyOF;
25 import org.opendaylight.openflowplugin.openflow.md.core.session.OFSessionUtil;
26 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.IpAddress;
27 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.Ipv4Address;
28 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.Ipv6Address;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.FlowCapableNodeUpdated;
30 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.FlowCapableNodeUpdatedBuilder;
31 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeId;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeRef;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeRemoved;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeRemovedBuilder;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeUpdated;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeUpdatedBuilder;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.Nodes;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.nodes.Node;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.nodes.NodeKey;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.openflow.protocol.rev130731.GetFeaturesOutput;
41 import org.opendaylight.yangtools.concepts.CompositeObjectRegistration;
42 import org.opendaylight.yangtools.concepts.ListenerRegistration;
43 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
44 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier.InstanceIdentifierBuilder;
45 import org.slf4j.Logger;
46 import org.slf4j.LoggerFactory;
49 * session and inventory listener implementation
51 public class SalRegistrationManager implements SessionListener, AutoCloseable {
53 private static final Logger LOG = LoggerFactory.getLogger(SalRegistrationManager.class);
55 private NotificationProviderService publishService;
57 private DataBroker dataService;
59 private RpcProviderRegistry rpcProviderRegistry;
61 private final SwitchFeaturesUtil swFeaturesUtil;
63 private ListenerRegistration<SessionListener> sessionListenerRegistration;
65 public SalRegistrationManager() {
66 swFeaturesUtil = SwitchFeaturesUtil.getInstance();
69 public NotificationProviderService getPublishService() {
70 return publishService;
73 public void setPublishService(final NotificationProviderService publishService) {
74 this.publishService = publishService;
77 public void setDataService(final DataBroker dataService) {
78 this.dataService = dataService;
81 public void setRpcProviderRegistry(final RpcProviderRegistry rpcProviderRegistry) {
82 this.rpcProviderRegistry = rpcProviderRegistry;
87 sessionListenerRegistration = getSessionManager().registerSessionListener(this);
88 getSessionManager().setNotificationProviderService(publishService);
89 getSessionManager().setDataBroker(dataService);
90 LOG.debug("SalRegistrationManager initialized");
94 public void onSessionAdded(final SwitchSessionKeyOF sessionKey, final SessionContext context) {
95 GetFeaturesOutput features = context.getFeatures();
96 BigInteger datapathId = features.getDatapathId();
97 InstanceIdentifier<Node> identifier = identifierFromDatapathId(datapathId);
98 NodeRef nodeRef = new NodeRef(identifier);
99 NodeId nodeId = nodeIdFromDatapathId(datapathId);
100 ModelDrivenSwitchImpl ofSwitch = new ModelDrivenSwitchImpl(nodeId, identifier, context);
101 CompositeObjectRegistration<ModelDrivenSwitch> registration =
102 ofSwitch.register(rpcProviderRegistry);
103 context.setProviderRegistration(registration);
105 LOG.debug("ModelDrivenSwitch for {} registered to MD-SAL.", datapathId);
107 NotificationQueueWrapper wrappedNotification = new NotificationQueueWrapper(
108 nodeAdded(ofSwitch, features, nodeRef),
109 context.getFeatures().getVersion());
110 context.getNotificationEnqueuer().enqueueNotification(wrappedNotification);
114 public void onSessionRemoved(final SessionContext context) {
115 GetFeaturesOutput features = context.getFeatures();
116 BigInteger datapathId = features.getDatapathId();
117 InstanceIdentifier<Node> identifier = identifierFromDatapathId(datapathId);
118 NodeRef nodeRef = new NodeRef(identifier);
119 NodeRemoved nodeRemoved = nodeRemoved(nodeRef);
121 CompositeObjectRegistration<ModelDrivenSwitch> registration = context.getProviderRegistration();
122 if (null != registration) {
123 registration.close();
124 context.setProviderRegistration(null);
126 LOG.debug("ModelDrivenSwitch for {} unregistered from MD-SAL.", datapathId);
128 NotificationQueueWrapper wrappedNotification = new NotificationQueueWrapper(
129 nodeRemoved, context.getFeatures().getVersion());
130 context.getNotificationEnqueuer().enqueueNotification(wrappedNotification);
133 private NodeUpdated nodeAdded(final ModelDrivenSwitch sw, final GetFeaturesOutput features, final NodeRef nodeRef) {
134 NodeUpdatedBuilder builder = new NodeUpdatedBuilder();
135 builder.setId(sw.getNodeId());
136 builder.setNodeRef(nodeRef);
138 FlowCapableNodeUpdatedBuilder builder2 = new FlowCapableNodeUpdatedBuilder();
140 builder2.setIpAddress(getIpAddressOf(sw));
141 } catch (Exception e) {
142 LOG.warn("IP address of the node {} cannot be obtained.", sw.getNodeId(), e);
144 builder2.setSwitchFeatures(swFeaturesUtil.buildSwitchFeatures(features));
145 builder.addAugmentation(FlowCapableNodeUpdated.class, builder2.build());
147 return builder.build();
150 private static IpAddress getIpAddressOf(final ModelDrivenSwitch sw) {
151 SessionContext sessionContext = sw.getSessionContext();
152 Preconditions.checkNotNull(sessionContext.getPrimaryConductor(),
153 "primary conductor must not be NULL -> " + sw.getNodeId());
154 Preconditions.checkNotNull(sessionContext.getPrimaryConductor().getConnectionAdapter(),
155 "connection adapter of primary conductor must not be NULL -> " + sw.getNodeId());
156 InetSocketAddress remoteAddress = sessionContext.getPrimaryConductor().getConnectionAdapter()
158 if (remoteAddress == null) {
159 LOG.warn("IP address of the node {} cannot be obtained. No connection with switch.", sw.getNodeId());
162 return resolveIpAddress(remoteAddress.getAddress());
165 private static IpAddress resolveIpAddress(final InetAddress address) {
166 String hostAddress = address.getHostAddress();
167 if (address instanceof Inet4Address) {
168 return new IpAddress(new Ipv4Address(hostAddress));
170 if (address instanceof Inet6Address) {
171 return new IpAddress(new Ipv6Address(hostAddress));
173 throw new IllegalArgumentException("Unsupported IP address type!");
176 private static NodeRemoved nodeRemoved(final NodeRef nodeRef) {
177 NodeRemovedBuilder builder = new NodeRemovedBuilder();
178 builder.setNodeRef(nodeRef);
179 return builder.build();
182 public static InstanceIdentifier<Node> identifierFromDatapathId(final BigInteger datapathId) {
183 NodeKey nodeKey = nodeKeyFromDatapathId(datapathId);
184 InstanceIdentifierBuilder<Node> builder = InstanceIdentifier.builder(Nodes.class).child(Node.class, nodeKey);
185 return builder.build();
188 public static NodeKey nodeKeyFromDatapathId(final BigInteger datapathId) {
189 return new NodeKey(nodeIdFromDatapathId(datapathId));
192 public static NodeId nodeIdFromDatapathId(final BigInteger datapathId) {
193 // FIXME: Convert to textual representation of datapathID
194 String current = String.valueOf(datapathId);
195 return new NodeId("openflow:" + current);
198 public SessionManager getSessionManager() {
199 return OFSessionUtil.getSessionManager();
203 public void close() {
206 rpcProviderRegistry = null;
207 publishService = null;
208 if (sessionListenerRegistration != null) {
209 sessionListenerRegistration.close();