blob: e417d9a82a3adc5f69451f0d6b96e050a812e108 [file] [log] [blame]
sanghoshin94872a12015-10-16 18:04:34 +09001/*
Daniel Park3a06c522016-01-28 20:51:12 +09002* Copyright 2015-2016 Open Networking Laboratory
sanghoshin94872a12015-10-16 18:04:34 +09003*
4* 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
7*
8* 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.
15*/
sangho0c2a3da2016-02-16 13:39:07 +090016package org.onosproject.openstacknetworking.switching;
sanghoshin94872a12015-10-16 18:04:34 +090017
danielbb83ebc2015-10-29 15:13:06 +090018import org.onlab.packet.ARP;
19import org.onlab.packet.Ethernet;
20import org.onlab.packet.Ip4Address;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080021import org.onlab.packet.IpAddress;
danielbb83ebc2015-10-29 15:13:06 +090022import org.onlab.packet.MacAddress;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080023import org.onosproject.net.Host;
danielbb83ebc2015-10-29 15:13:06 +090024import org.onosproject.net.flow.DefaultTrafficTreatment;
25import org.onosproject.net.flow.TrafficTreatment;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080026import org.onosproject.net.host.HostService;
danielbb83ebc2015-10-29 15:13:06 +090027import org.onosproject.net.packet.DefaultOutboundPacket;
sanghoshin94872a12015-10-16 18:04:34 +090028import org.onosproject.net.packet.InboundPacket;
danielbb83ebc2015-10-29 15:13:06 +090029import org.onosproject.net.packet.PacketService;
sangho93447f12016-02-24 00:33:22 +090030import org.onosproject.openstackinterface.OpenstackInterfaceService;
31import org.onosproject.openstackinterface.OpenstackPort;
sangho0c2a3da2016-02-16 13:39:07 +090032import org.onosproject.openstacknetworking.OpenstackPortInfo;
sanghoshin94872a12015-10-16 18:04:34 +090033import org.slf4j.Logger;
34import org.slf4j.LoggerFactory;
danielbb83ebc2015-10-29 15:13:06 +090035import java.nio.ByteBuffer;
Daniel Park3a06c522016-01-28 20:51:12 +090036import java.util.Collection;
sanghoshinf25d2e02015-11-11 23:07:17 +090037
38import static com.google.common.base.Preconditions.checkNotNull;
sanghoshin94872a12015-10-16 18:04:34 +090039
40/**
sanghoshin46297d22015-11-03 17:51:24 +090041 * Handles ARP packet from VMs.
sanghoshin94872a12015-10-16 18:04:34 +090042 */
43public class OpenstackArpHandler {
44
45 private static Logger log = LoggerFactory
46 .getLogger(OpenstackArpHandler.class);
Daniel Park3a06c522016-01-28 20:51:12 +090047 private static final MacAddress GATEWAY_MAC = MacAddress.valueOf("1f:1f:1f:1f:1f:1f");
danielbb83ebc2015-10-29 15:13:06 +090048 private PacketService packetService;
sangho93447f12016-02-24 00:33:22 +090049 private OpenstackInterfaceService openstackService;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080050 private HostService hostService;
sanghoshin94872a12015-10-16 18:04:34 +090051
52 /**
danielbb83ebc2015-10-29 15:13:06 +090053 * Returns OpenstackArpHandler reference.
sanghoshin94872a12015-10-16 18:04:34 +090054 *
sangho0c2a3da2016-02-16 13:39:07 +090055 * @param openstackService OpenstackNetworkingService reference
sanghoshinf25d2e02015-11-11 23:07:17 +090056 * @param packetService PacketService reference
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080057 * @param hostService host service
sanghoshin94872a12015-10-16 18:04:34 +090058 */
sangho93447f12016-02-24 00:33:22 +090059 public OpenstackArpHandler(OpenstackInterfaceService openstackService, PacketService packetService,
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080060 HostService hostService) {
sangho0c2a3da2016-02-16 13:39:07 +090061 this.openstackService = openstackService;
danielbb83ebc2015-10-29 15:13:06 +090062 this.packetService = packetService;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080063 this.hostService = hostService;
sanghoshin94872a12015-10-16 18:04:34 +090064 }
65
66 /**
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080067 * Processes ARP request packets.
68 * It checks if the target IP is owned by a known host first and then ask to
69 * OpenStack if it's not. This ARP proxy does not support overlapping IP.
sanghoshin94872a12015-10-16 18:04:34 +090070 *
71 * @param pkt ARP request packet
72 */
Daniel Park3a06c522016-01-28 20:51:12 +090073 public void processPacketIn(InboundPacket pkt, Collection<OpenstackPortInfo> openstackPortInfoCollection) {
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080074 Ethernet ethRequest = pkt.parsed();
75 ARP arp = (ARP) ethRequest.getPayload();
danielbb83ebc2015-10-29 15:13:06 +090076
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080077 if (arp.getOpCode() != ARP.OP_REQUEST) {
78 return;
79 }
danielbb83ebc2015-10-29 15:13:06 +090080
Daniel Park3a06c522016-01-28 20:51:12 +090081 IpAddress sourceIp = Ip4Address.valueOf(arp.getSenderProtocolAddress());
82 MacAddress srcMac = MacAddress.valueOf(arp.getSenderHardwareAddress());
83 OpenstackPortInfo portInfo = openstackPortInfoCollection.stream()
84 .filter(p -> p.ip().equals(sourceIp) && p.mac().equals(srcMac)).findFirst().orElse(null);
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080085 IpAddress targetIp = Ip4Address.valueOf(arp.getTargetProtocolAddress());
Daniel Park3a06c522016-01-28 20:51:12 +090086
87 MacAddress dstMac;
88
89 if (targetIp.equals(portInfo == null ? null : portInfo.gatewayIP())) {
90 dstMac = GATEWAY_MAC;
91 } else {
92 dstMac = getMacFromHostService(targetIp);
93 if (dstMac == null) {
94 dstMac = getMacFromOpenstack(targetIp);
95 }
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080096 }
danielbb83ebc2015-10-29 15:13:06 +090097
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080098 if (dstMac == null) {
99 log.debug("Failed to find MAC address for {}", targetIp.toString());
100 return;
101 }
danielbb83ebc2015-10-29 15:13:06 +0900102
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800103 Ethernet ethReply = ARP.buildArpReply(targetIp.getIp4Address(),
104 dstMac,
105 ethRequest);
danielbb83ebc2015-10-29 15:13:06 +0900106
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800107 TrafficTreatment treatment = DefaultTrafficTreatment.builder()
108 .setOutput(pkt.receivedFrom().port())
109 .build();
danielbb83ebc2015-10-29 15:13:06 +0900110
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800111 packetService.emit(new DefaultOutboundPacket(
112 pkt.receivedFrom().deviceId(),
113 treatment,
114 ByteBuffer.wrap(ethReply.serialize())));
115 }
danielbb83ebc2015-10-29 15:13:06 +0900116
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800117 /**
118 * Returns MAC address of a host with a given target IP address by asking to
119 * OpenStack. It does not support overlapping IP.
120 *
121 * @param targetIp target ip address
122 * @return mac address, or null if it fails to fetch the mac
123 */
124 private MacAddress getMacFromOpenstack(IpAddress targetIp) {
125 checkNotNull(targetIp);
126
sangho0c2a3da2016-02-16 13:39:07 +0900127 OpenstackPort openstackPort = openstackService.ports()
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800128 .stream()
129 .filter(port -> port.fixedIps().containsValue(targetIp))
130 .findFirst()
131 .orElse(null);
132
133 if (openstackPort != null) {
134 log.debug("Found MAC from OpenStack for {}", targetIp.toString());
135 return openstackPort.macAddress();
136 } else {
137 return null;
138 }
139 }
140
141 /**
142 * Returns MAC address of a host with a given target IP address by asking to
143 * host service. It does not support overlapping IP.
144 *
145 * @param targetIp target ip
146 * @return mac address, or null if it fails to find the mac
147 */
148 private MacAddress getMacFromHostService(IpAddress targetIp) {
149 checkNotNull(targetIp);
150
151 Host host = hostService.getHostsByIp(targetIp)
152 .stream()
153 .findFirst()
154 .orElse(null);
155
156 if (host != null) {
157 log.debug("Found MAC from host service for {}", targetIp.toString());
158 return host.mac();
159 } else {
160 return null;
danielbb83ebc2015-10-29 15:13:06 +0900161 }
sanghoshin94872a12015-10-16 18:04:34 +0900162 }
163}