blob: 2505be5dd934c66e8faaec361813390857ee4d6f [file] [log] [blame]
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07001/*
2 * Copyright 2014 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 */
Brian O'Connor66630c82014-10-02 21:08:19 -070016package org.onlab.onos.net.intent.impl;
17
Brian O'Connora4cab072014-10-03 18:46:39 -070018import java.util.Iterator;
alshabib902d41b2014-10-07 16:52:05 -070019import java.util.List;
Brian O'Connora4cab072014-10-03 18:46:39 -070020
Brian O'Connor66630c82014-10-02 21:08:19 -070021import org.apache.felix.scr.annotations.Activate;
22import org.apache.felix.scr.annotations.Component;
23import org.apache.felix.scr.annotations.Deactivate;
24import org.apache.felix.scr.annotations.Reference;
25import org.apache.felix.scr.annotations.ReferenceCardinality;
Thomas Vachuskae0f804a2014-10-27 23:40:48 -070026import org.onlab.onos.core.ApplicationId;
27import org.onlab.onos.core.CoreService;
Brian O'Connor66630c82014-10-02 21:08:19 -070028import org.onlab.onos.net.ConnectPoint;
29import org.onlab.onos.net.Link;
30import org.onlab.onos.net.flow.DefaultFlowRule;
31import org.onlab.onos.net.flow.DefaultTrafficSelector;
Brian O'Connor66630c82014-10-02 21:08:19 -070032import org.onlab.onos.net.flow.FlowRule;
alshabib902d41b2014-10-07 16:52:05 -070033import org.onlab.onos.net.flow.FlowRuleBatchEntry;
34import org.onlab.onos.net.flow.FlowRuleBatchEntry.FlowRuleOperation;
35import org.onlab.onos.net.flow.FlowRuleBatchOperation;
Brian O'Connor66630c82014-10-02 21:08:19 -070036import org.onlab.onos.net.flow.TrafficSelector;
37import org.onlab.onos.net.flow.TrafficTreatment;
Thomas Vachuskaedc944c2014-11-04 15:42:25 -080038import org.onlab.onos.net.intent.Constraint;
Brian O'Connor66630c82014-10-02 21:08:19 -070039import org.onlab.onos.net.intent.IntentExtensionService;
40import org.onlab.onos.net.intent.IntentInstaller;
41import org.onlab.onos.net.intent.PathIntent;
Thomas Vachuskaedc944c2014-11-04 15:42:25 -080042import org.onlab.onos.net.resource.DefaultLinkResourceRequest;
Ray Milkeycaa450b2014-10-29 15:54:24 -070043import org.onlab.onos.net.resource.LinkResourceAllocations;
Thomas Vachuskaedc944c2014-11-04 15:42:25 -080044import org.onlab.onos.net.resource.LinkResourceRequest;
Ray Milkeycaa450b2014-10-29 15:54:24 -070045import org.onlab.onos.net.resource.LinkResourceService;
alshabib8ca53902014-10-07 13:11:17 -070046import org.slf4j.Logger;
Brian O'Connor66630c82014-10-02 21:08:19 -070047
alshabib902d41b2014-10-07 16:52:05 -070048import com.google.common.collect.Lists;
49
Ray Milkeycaa450b2014-10-29 15:54:24 -070050import static org.onlab.onos.net.flow.DefaultTrafficTreatment.builder;
51import static org.slf4j.LoggerFactory.getLogger;
52
Brian O'Connor66630c82014-10-02 21:08:19 -070053/**
Thomas Vachuska425a2d72014-10-29 11:28:28 -070054 * Installer for {@link PathIntent packet path connectivity intents}.
Brian O'Connor66630c82014-10-02 21:08:19 -070055 */
56@Component(immediate = true)
tom9a693fd2014-10-03 11:32:19 -070057public class PathIntentInstaller implements IntentInstaller<PathIntent> {
58
alshabib8ca53902014-10-07 13:11:17 -070059 private final Logger log = getLogger(getClass());
60
Brian O'Connor66630c82014-10-02 21:08:19 -070061 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
62 protected IntentExtensionService intentManager;
63
64 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabib92c65ad2014-10-08 21:56:05 -070065 protected CoreService coreService;
66
Ray Milkeycaa450b2014-10-29 15:54:24 -070067 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
68 protected LinkResourceService resourceService;
69
Ray Milkey8d3ce432014-11-07 16:21:10 -080070 protected ApplicationId appId;
Brian O'Connor66630c82014-10-02 21:08:19 -070071
72 @Activate
73 public void activate() {
alshabib92c65ad2014-10-08 21:56:05 -070074 appId = coreService.registerApplication("org.onlab.onos.net.intent");
Brian O'Connor66630c82014-10-02 21:08:19 -070075 intentManager.registerInstaller(PathIntent.class, this);
76 }
77
78 @Deactivate
79 public void deactivate() {
80 intentManager.unregisterInstaller(PathIntent.class);
81 }
82
83 @Override
Brian O'Connorf2dbde52014-10-10 16:20:24 -070084 public List<FlowRuleBatchOperation> install(PathIntent intent) {
Thomas Vachuskaedc944c2014-11-04 15:42:25 -080085 LinkResourceAllocations allocations = allocateResources(intent);
Thomas Vachuskae3784c92014-10-29 21:09:10 -070086
tom9a693fd2014-10-03 11:32:19 -070087 TrafficSelector.Builder builder =
tom85258ee2014-10-07 00:10:02 -070088 DefaultTrafficSelector.builder(intent.selector());
89 Iterator<Link> links = intent.path().links().iterator();
Brian O'Connor66630c82014-10-02 21:08:19 -070090 ConnectPoint prev = links.next().dst();
alshabib902d41b2014-10-07 16:52:05 -070091 List<FlowRuleBatchEntry> rules = Lists.newLinkedList();
Brian O'Connorf2dbde52014-10-10 16:20:24 -070092 // TODO Generate multiple batches
Brian O'Connor66630c82014-10-02 21:08:19 -070093 while (links.hasNext()) {
94 builder.matchInport(prev.port());
95 Link link = links.next();
tomf5c9d922014-10-03 15:22:03 -070096 TrafficTreatment treatment = builder()
97 .setOutput(link.src().port()).build();
alshabib902d41b2014-10-07 16:52:05 -070098
tom9a693fd2014-10-03 11:32:19 -070099 FlowRule rule = new DefaultFlowRule(link.src().deviceId(),
Brian O'Connora4cab072014-10-03 18:46:39 -0700100 builder.build(), treatment,
Jonathan Hartbc4a7932014-10-21 11:46:00 -0700101 123, appId, 0, true);
alshabib902d41b2014-10-07 16:52:05 -0700102 rules.add(new FlowRuleBatchEntry(FlowRuleOperation.ADD, rule));
Brian O'Connor66630c82014-10-02 21:08:19 -0700103 prev = link.dst();
104 }
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700105 return Lists.newArrayList(new FlowRuleBatchOperation(rules));
Brian O'Connor66630c82014-10-02 21:08:19 -0700106 }
107
108 @Override
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700109 public List<FlowRuleBatchOperation> uninstall(PathIntent intent) {
Thomas Vachuskaedc944c2014-11-04 15:42:25 -0800110 LinkResourceAllocations allocatedResources = resourceService.getAllocations(intent.id());
111 if (allocatedResources != null) {
112 resourceService.releaseResources(allocatedResources);
113 }
Brian O'Connora4cab072014-10-03 18:46:39 -0700114 TrafficSelector.Builder builder =
tom85258ee2014-10-07 00:10:02 -0700115 DefaultTrafficSelector.builder(intent.selector());
116 Iterator<Link> links = intent.path().links().iterator();
Brian O'Connora4cab072014-10-03 18:46:39 -0700117 ConnectPoint prev = links.next().dst();
alshabib902d41b2014-10-07 16:52:05 -0700118 List<FlowRuleBatchEntry> rules = Lists.newLinkedList();
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700119 // TODO Generate multiple batches
Brian O'Connora4cab072014-10-03 18:46:39 -0700120 while (links.hasNext()) {
121 builder.matchInport(prev.port());
122 Link link = links.next();
123 TrafficTreatment treatment = builder()
124 .setOutput(link.src().port()).build();
125 FlowRule rule = new DefaultFlowRule(link.src().deviceId(),
126 builder.build(), treatment,
Jonathan Hartbc4a7932014-10-21 11:46:00 -0700127 123, appId, 0, true);
alshabib902d41b2014-10-07 16:52:05 -0700128 rules.add(new FlowRuleBatchEntry(FlowRuleOperation.REMOVE, rule));
Brian O'Connora4cab072014-10-03 18:46:39 -0700129 prev = link.dst();
130 }
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700131 return Lists.newArrayList(new FlowRuleBatchOperation(rules));
Brian O'Connorcb900f42014-10-07 21:55:33 -0700132 }
133
Thomas Vachuskaedc944c2014-11-04 15:42:25 -0800134 /**
135 * Allocate resources required for an intent.
136 *
137 * @param intent intent to allocate resource for
138 * @return allocated resources if any are required, null otherwise
139 */
140 private LinkResourceAllocations allocateResources(PathIntent intent) {
Thomas Vachuskaedc944c2014-11-04 15:42:25 -0800141 LinkResourceRequest.Builder builder =
142 DefaultLinkResourceRequest.builder(intent.id(), intent.path().links());
143 for (Constraint constraint : intent.constraints()) {
144 builder.addConstraint(constraint);
145 }
146 LinkResourceRequest request = builder.build();
147 return request.resources().isEmpty() ? null : resourceService.requestResources(request);
Ray Milkeycaa450b2014-10-29 15:54:24 -0700148 }
149
Brian O'Connorcb900f42014-10-07 21:55:33 -0700150 // TODO refactor below this line... ----------------------------
151
152 /**
153 * Generates the series of MatchActionOperations from the
154 * {@link FlowBatchOperation}.
155 * <p>
156 * FIXME: Currently supporting PacketPathFlow and SingleDstTreeFlow only.
157 * <p>
158 * FIXME: MatchActionOperations should have dependency field to the other
159 * match action operations, and this method should use this.
160 *
161 * @param op the {@link FlowBatchOperation} object
162 * @return the list of {@link MatchActionOperations} objects
163 */
164 /*
165 private List<MatchActionOperations>
166 generateMatchActionOperationsList(FlowBatchOperation op) {
167
168 // MatchAction operations at head (ingress) switches.
169 MatchActionOperations headOps = matchActionService.createOperationsList();
170
171 // MatchAction operations at rest of the switches.
172 MatchActionOperations tailOps = matchActionService.createOperationsList();
173
174 MatchActionOperations removeOps = matchActionService.createOperationsList();
175
176 for (BatchOperationEntry<Operator, ?> e : op.getOperations()) {
177
178 if (e.getOperator() == FlowBatchOperation.Operator.ADD) {
179 generateInstallMatchActionOperations(e, tailOps, headOps);
180 } else if (e.getOperator() == FlowBatchOperation.Operator.REMOVE) {
181 generateRemoveMatchActionOperations(e, removeOps);
182 } else {
183 throw new UnsupportedOperationException(
184 "FlowManager supports ADD and REMOVE operations only.");
185 }
186
187 }
188
189 return Arrays.asList(tailOps, headOps, removeOps);
190 }
191 */
192
193 /**
194 * Generates MatchActionOperations for an INSTALL FlowBatchOperation.
195 * <p/>
196 * FIXME: Currently only supports flows that generate exactly two match
197 * action operation sets.
198 *
199 * @param e Flow BatchOperationEntry
200 * @param tailOps MatchActionOperation set that the tail
201 * MatchActionOperations will be placed in
202 * @param headOps MatchActionOperation set that the head
203 * MatchActionOperations will be placed in
204 */
205 /*
206 private void generateInstallMatchActionOperations(
207 BatchOperationEntry<Operator, ?> e,
208 MatchActionOperations tailOps,
209 MatchActionOperations headOps) {
210
211 if (!(e.getTarget() instanceof Flow)) {
212 throw new IllegalStateException(
213 "The target is not Flow object: " + e.getTarget());
214 }
215
216 // Compile flows to match-actions
217 Flow flow = (Flow) e.getTarget();
218 List<MatchActionOperations> maOps = flow.compile(
219 e.getOperator(), matchActionService);
220 verifyNotNull(maOps, "Could not compile the flow: " + flow);
221 verify(maOps.size() == 2,
222 "The flow generates unspported match-action operations.");
223
224 // Map FlowId to MatchActionIds
225 for (MatchActionOperations maOp : maOps) {
226 for (MatchActionOperationEntry entry : maOp.getOperations()) {
227 flowMatchActionsMap.put(
228 KryoFactory.serialize(flow.getId()),
229 KryoFactory.serialize(entry.getTarget()));
230 }
231 }
232
233 // Merge match-action operations
234 for (MatchActionOperationEntry mae : maOps.get(0).getOperations()) {
235 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
236 tailOps.addOperation(mae);
237 }
238 for (MatchActionOperationEntry mae : maOps.get(1).getOperations()) {
239 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
240 headOps.addOperation(mae);
alshabib902d41b2014-10-07 16:52:05 -0700241 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700242 }
Brian O'Connorcb900f42014-10-07 21:55:33 -0700243 */
244 /**
245 * Generates MatchActionOperations for a REMOVE FlowBatchOperation.
246 *
247 * @param e Flow BatchOperationEntry
248 * @param removeOps MatchActionOperation set that the remove
249 * MatchActionOperations will be placed in
250 */
251 /*
252 private void generateRemoveMatchActionOperations(
253 BatchOperationEntry<Operator, ?> e,
254 MatchActionOperations removeOps) {
255
256 if (!(e.getTarget() instanceof FlowId)) {
257 throw new IllegalStateException(
258 "The target is not a FlowId object: " + e.getTarget());
259 }
260
261 // Compile flows to match-actions
262 FlowId flowId = (FlowId) e.getTarget();
263
264 for (byte[] matchActionIdBytes :
265 flowMatchActionsMap.remove(KryoFactory.serialize(flowId))) {
266 MatchActionId matchActionId = KryoFactory.deserialize(matchActionIdBytes);
267 removeOps.addOperation(new MatchActionOperationEntry(
268 MatchActionOperations.Operator.REMOVE, matchActionId));
269 }
270 }
271 */
Brian O'Connor66630c82014-10-02 21:08:19 -0700272}