blob: b0a4c438b181c4f67371987da5622b9130678c41 [file] [log] [blame]
sanghoshin94872a12015-10-16 18:04:34 +09001/*
2* Copyright 2015 Open Networking Laboratory
3*
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*/
16package org.onosproject.openstackswitching;
17
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;
sanghoshin94872a12015-10-16 18:04:34 +090030import org.slf4j.Logger;
31import org.slf4j.LoggerFactory;
danielbb83ebc2015-10-29 15:13:06 +090032import java.nio.ByteBuffer;
sanghoshinf25d2e02015-11-11 23:07:17 +090033
34import static com.google.common.base.Preconditions.checkNotNull;
sanghoshin94872a12015-10-16 18:04:34 +090035
36/**
sanghoshin46297d22015-11-03 17:51:24 +090037 * Handles ARP packet from VMs.
sanghoshin94872a12015-10-16 18:04:34 +090038 */
39public class OpenstackArpHandler {
40
41 private static Logger log = LoggerFactory
42 .getLogger(OpenstackArpHandler.class);
danielbb83ebc2015-10-29 15:13:06 +090043 private PacketService packetService;
sanghoshinf25d2e02015-11-11 23:07:17 +090044 private OpenstackRestHandler restHandler;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080045 private HostService hostService;
sanghoshin94872a12015-10-16 18:04:34 +090046
47 /**
danielbb83ebc2015-10-29 15:13:06 +090048 * Returns OpenstackArpHandler reference.
sanghoshin94872a12015-10-16 18:04:34 +090049 *
sanghoshinf25d2e02015-11-11 23:07:17 +090050 * @param restHandler rest API handler reference
51 * @param packetService PacketService reference
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080052 * @param hostService host service
sanghoshin94872a12015-10-16 18:04:34 +090053 */
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080054 public OpenstackArpHandler(OpenstackRestHandler restHandler, PacketService packetService,
55 HostService hostService) {
sanghoshinf25d2e02015-11-11 23:07:17 +090056 this.restHandler = checkNotNull(restHandler);
danielbb83ebc2015-10-29 15:13:06 +090057 this.packetService = packetService;
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080058 this.hostService = hostService;
sanghoshin94872a12015-10-16 18:04:34 +090059 }
60
61 /**
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080062 * Processes ARP request packets.
63 * It checks if the target IP is owned by a known host first and then ask to
64 * OpenStack if it's not. This ARP proxy does not support overlapping IP.
sanghoshin94872a12015-10-16 18:04:34 +090065 *
66 * @param pkt ARP request packet
67 */
68 public void processPacketIn(InboundPacket pkt) {
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080069 Ethernet ethRequest = pkt.parsed();
70 ARP arp = (ARP) ethRequest.getPayload();
danielbb83ebc2015-10-29 15:13:06 +090071
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080072 if (arp.getOpCode() != ARP.OP_REQUEST) {
73 return;
74 }
danielbb83ebc2015-10-29 15:13:06 +090075
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080076 IpAddress targetIp = Ip4Address.valueOf(arp.getTargetProtocolAddress());
77 MacAddress dstMac = getMacFromHostService(targetIp);
78 if (dstMac == null) {
79 dstMac = getMacFromOpenstack(targetIp);
80 }
danielbb83ebc2015-10-29 15:13:06 +090081
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080082 if (dstMac == null) {
83 log.debug("Failed to find MAC address for {}", targetIp.toString());
84 return;
85 }
danielbb83ebc2015-10-29 15:13:06 +090086
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080087 Ethernet ethReply = ARP.buildArpReply(targetIp.getIp4Address(),
88 dstMac,
89 ethRequest);
danielbb83ebc2015-10-29 15:13:06 +090090
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080091 TrafficTreatment treatment = DefaultTrafficTreatment.builder()
92 .setOutput(pkt.receivedFrom().port())
93 .build();
danielbb83ebc2015-10-29 15:13:06 +090094
Hyunsun Moonc98e7c52016-01-11 02:54:27 -080095 packetService.emit(new DefaultOutboundPacket(
96 pkt.receivedFrom().deviceId(),
97 treatment,
98 ByteBuffer.wrap(ethReply.serialize())));
99 }
danielbb83ebc2015-10-29 15:13:06 +0900100
Hyunsun Moonc98e7c52016-01-11 02:54:27 -0800101 /**
102 * Returns MAC address of a host with a given target IP address by asking to
103 * OpenStack. It does not support overlapping IP.
104 *
105 * @param targetIp target ip address
106 * @return mac address, or null if it fails to fetch the mac
107 */
108 private MacAddress getMacFromOpenstack(IpAddress targetIp) {
109 checkNotNull(targetIp);
110
111 OpenstackPort openstackPort = restHandler.getPorts()
112 .stream()
113 .filter(port -> port.fixedIps().containsValue(targetIp))
114 .findFirst()
115 .orElse(null);
116
117 if (openstackPort != null) {
118 log.debug("Found MAC from OpenStack for {}", targetIp.toString());
119 return openstackPort.macAddress();
120 } else {
121 return null;
122 }
123 }
124
125 /**
126 * Returns MAC address of a host with a given target IP address by asking to
127 * host service. It does not support overlapping IP.
128 *
129 * @param targetIp target ip
130 * @return mac address, or null if it fails to find the mac
131 */
132 private MacAddress getMacFromHostService(IpAddress targetIp) {
133 checkNotNull(targetIp);
134
135 Host host = hostService.getHostsByIp(targetIp)
136 .stream()
137 .findFirst()
138 .orElse(null);
139
140 if (host != null) {
141 log.debug("Found MAC from host service for {}", targetIp.toString());
142 return host.mac();
143 } else {
144 return null;
danielbb83ebc2015-10-29 15:13:06 +0900145 }
sanghoshin94872a12015-10-16 18:04:34 +0900146 }
147}