blob: b8e16a85f8e3301c1f35e8b61fa9be9d19357be3 [file] [log] [blame]
sanghob35a6192015-04-01 13:05:26 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-present Open Networking Laboratory
sanghob35a6192015-04-01 13:05:26 -07003 *
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.segmentrouting;
17
18import org.onlab.packet.Ethernet;
19import org.onlab.packet.IPv4;
20import org.onlab.packet.Ip4Address;
Pier Ventref4b5fce2016-11-28 16:48:06 -080021import org.onlab.packet.Ip6Address;
sanghob35a6192015-04-01 13:05:26 -070022import org.onosproject.net.ConnectPoint;
23import org.onosproject.net.DeviceId;
24import org.onosproject.net.Host;
25import org.onosproject.net.flow.DefaultTrafficTreatment;
26import org.onosproject.net.flow.TrafficTreatment;
27import org.onosproject.net.packet.DefaultOutboundPacket;
28import org.onosproject.net.packet.InboundPacket;
29import org.onosproject.net.packet.OutboundPacket;
Charles Chan0b4e6182015-11-03 10:42:14 -080030import org.onosproject.segmentrouting.config.DeviceConfigNotFoundException;
31import org.onosproject.segmentrouting.config.DeviceConfiguration;
sanghob35a6192015-04-01 13:05:26 -070032import org.slf4j.Logger;
33import org.slf4j.LoggerFactory;
34
35import java.nio.ByteBuffer;
36import java.util.concurrent.ConcurrentHashMap;
37import java.util.concurrent.ConcurrentLinkedQueue;
38
39import static com.google.common.base.Preconditions.checkNotNull;
40
Charles Chane849c192016-01-11 18:28:54 -080041/**
42 * Handler of IP packets that forwards IP packets that are sent to the controller,
43 * except the ICMP packets which are processed by @link{IcmpHandler}.
44 */
sanghob35a6192015-04-01 13:05:26 -070045public class IpHandler {
46
47 private static Logger log = LoggerFactory.getLogger(IpHandler.class);
48 private SegmentRoutingManager srManager;
sangho666cd6d2015-04-14 16:27:13 -070049 private DeviceConfiguration config;
sanghob35a6192015-04-01 13:05:26 -070050 private ConcurrentHashMap<Ip4Address, ConcurrentLinkedQueue<IPv4>> ipPacketQueue;
51
52 /**
53 * Creates an IpHandler object.
54 *
55 * @param srManager SegmentRoutingManager object
56 */
57 public IpHandler(SegmentRoutingManager srManager) {
58 this.srManager = srManager;
sangho666cd6d2015-04-14 16:27:13 -070059 this.config = checkNotNull(srManager.deviceConfiguration);
Sho SHIMIZU6cfc02d2015-09-11 11:19:11 -070060 ipPacketQueue = new ConcurrentHashMap<>();
sanghob35a6192015-04-01 13:05:26 -070061 }
62
63 /**
64 * Processes incoming IP packets.
65 *
66 * If it is an IP packet for known host, then forward it to the host.
67 * If it is an IP packet for unknown host in subnet, then send an ARP request
68 * to the subnet.
69 *
70 * @param pkt incoming packet
71 */
72 public void processPacketIn(InboundPacket pkt) {
73 Ethernet ethernet = pkt.parsed();
74 IPv4 ipv4 = (IPv4) ethernet.getPayload();
75
76 ConnectPoint connectPoint = pkt.receivedFrom();
77 DeviceId deviceId = connectPoint.deviceId();
78 Ip4Address destinationAddress =
79 Ip4Address.valueOf(ipv4.getDestinationAddress());
80
81 // IP packet for know hosts
82 if (!srManager.hostService.getHostsByIp(destinationAddress).isEmpty()) {
83 forwardPackets(deviceId, destinationAddress);
84
85 // IP packet for unknown host in the subnet of the router
86 } else if (config.inSameSubnet(deviceId, destinationAddress)) {
87 srManager.arpHandler.sendArpRequest(deviceId, destinationAddress, connectPoint);
88
89 // IP packets for unknown host
90 } else {
Charles Chanda5dc8c2017-01-13 14:29:29 -080091 log.debug("IP request for unknown host {} which is not in the subnet",
sanghob35a6192015-04-01 13:05:26 -070092 destinationAddress);
93 // Do nothing
94 }
95 }
96
97 /**
98 * Adds the IP packet to a buffer.
99 * The packets are forwarded to corresponding destination when the destination
100 * MAC address is known via ARP response.
101 *
102 * @param ipPacket IP packet to add to the buffer
103 */
104 public void addToPacketBuffer(IPv4 ipPacket) {
105
Saurav Das4ce45962015-11-24 23:21:05 -0800106 // Better not buffer TCP packets due to out-of-order packet transfer
sanghob35a6192015-04-01 13:05:26 -0700107 if (ipPacket.getProtocol() == IPv4.PROTOCOL_TCP) {
108 return;
109 }
110
111 Ip4Address destIpAddress = Ip4Address.valueOf(ipPacket.getDestinationAddress());
112
Yuta HIGUCHIa26109f2016-12-23 13:39:08 -0800113 ipPacketQueue
114 .computeIfAbsent(destIpAddress, a -> new ConcurrentLinkedQueue<>())
115 .add(ipPacket);
sanghob35a6192015-04-01 13:05:26 -0700116 }
117
118 /**
119 * Forwards IP packets in the buffer to the destination IP address.
120 * It is called when the controller finds the destination MAC address
Saurav Dasa07f2032015-10-19 14:37:36 -0700121 * via ARP responses.
sanghob35a6192015-04-01 13:05:26 -0700122 *
123 * @param deviceId switch device ID
124 * @param destIpAddress destination IP address
125 */
126 public void forwardPackets(DeviceId deviceId, Ip4Address destIpAddress) {
Srikanth Vavilapalli78baf582015-06-05 11:40:14 -0700127 if (ipPacketQueue.get(destIpAddress) == null) {
128 return;
129 }
130
sanghob35a6192015-04-01 13:05:26 -0700131 for (IPv4 ipPacket : ipPacketQueue.get(destIpAddress)) {
132 Ip4Address destAddress = Ip4Address.valueOf(ipPacket.getDestinationAddress());
Charles Chan0b4e6182015-11-03 10:42:14 -0800133 if (config.inSameSubnet(deviceId, destAddress)) {
sanghob35a6192015-04-01 13:05:26 -0700134 ipPacket.setTtl((byte) (ipPacket.getTtl() - 1));
135 ipPacket.setChecksum((short) 0);
136 for (Host dest: srManager.hostService.getHostsByIp(destIpAddress)) {
137 Ethernet eth = new Ethernet();
138 eth.setDestinationMACAddress(dest.mac());
Charles Chan0b4e6182015-11-03 10:42:14 -0800139 try {
140 eth.setSourceMACAddress(config.getDeviceMac(deviceId));
141 } catch (DeviceConfigNotFoundException e) {
142 log.warn(e.getMessage()
143 + " Skipping forwardPackets for this destination.");
144 continue;
145 }
sanghob35a6192015-04-01 13:05:26 -0700146 eth.setEtherType(Ethernet.TYPE_IPV4);
147 eth.setPayload(ipPacket);
148
149 TrafficTreatment treatment = DefaultTrafficTreatment.builder().
150 setOutput(dest.location().port()).build();
151 OutboundPacket packet = new DefaultOutboundPacket(deviceId,
152 treatment, ByteBuffer.wrap(eth.serialize()));
153 srManager.packetService.emit(packet);
sangho20eff1d2015-04-13 15:15:58 -0700154 ipPacketQueue.get(destIpAddress).remove(ipPacket);
sanghob35a6192015-04-01 13:05:26 -0700155 }
sangho666cd6d2015-04-14 16:27:13 -0700156 ipPacketQueue.get(destIpAddress).remove(ipPacket);
sanghob35a6192015-04-01 13:05:26 -0700157 }
158 }
159 }
sangho666cd6d2015-04-14 16:27:13 -0700160
Pier Ventref4b5fce2016-11-28 16:48:06 -0800161 //////////////////////
162 // IPv6 Handling //
163 ////////////////////
164
165 /**
166 * Forwards IP packets in the buffer to the destination IP address.
167 * It is called when the controller finds the destination MAC address
168 * via NDP replies.
169 *
170 * @param deviceId the target device
171 * @param destIpAddress the destination ip address
172 */
173 public void forwardPackets(DeviceId deviceId, Ip6Address destIpAddress) {
174 /*
175 * TODO in the following commit.
176 */
177 }
178
sanghob35a6192015-04-01 13:05:26 -0700179}