blob: 0e8abca4f9dbd39c89b575256893024709df9379 [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
Brian O'Connorfa81eae2014-10-30 13:20:05 -0700134 @Override
135 public List<FlowRuleBatchOperation> replace(PathIntent oldIntent, PathIntent newIntent) {
136 // FIXME: implement this
137 List<FlowRuleBatchOperation> batches = Lists.newArrayList();
138 batches.addAll(uninstall(oldIntent));
139 batches.addAll(install(newIntent));
140 return batches;
141 }
142
Thomas Vachuskaedc944c2014-11-04 15:42:25 -0800143 /**
144 * Allocate resources required for an intent.
145 *
146 * @param intent intent to allocate resource for
147 * @return allocated resources if any are required, null otherwise
148 */
149 private LinkResourceAllocations allocateResources(PathIntent intent) {
Thomas Vachuskaedc944c2014-11-04 15:42:25 -0800150 LinkResourceRequest.Builder builder =
151 DefaultLinkResourceRequest.builder(intent.id(), intent.path().links());
152 for (Constraint constraint : intent.constraints()) {
153 builder.addConstraint(constraint);
154 }
155 LinkResourceRequest request = builder.build();
156 return request.resources().isEmpty() ? null : resourceService.requestResources(request);
Ray Milkeycaa450b2014-10-29 15:54:24 -0700157 }
158
Brian O'Connorfa81eae2014-10-30 13:20:05 -0700159 // FIXME refactor below this line... ----------------------------
Brian O'Connorcb900f42014-10-07 21:55:33 -0700160
161 /**
162 * Generates the series of MatchActionOperations from the
163 * {@link FlowBatchOperation}.
164 * <p>
165 * FIXME: Currently supporting PacketPathFlow and SingleDstTreeFlow only.
166 * <p>
167 * FIXME: MatchActionOperations should have dependency field to the other
168 * match action operations, and this method should use this.
169 *
170 * @param op the {@link FlowBatchOperation} object
171 * @return the list of {@link MatchActionOperations} objects
172 */
173 /*
174 private List<MatchActionOperations>
175 generateMatchActionOperationsList(FlowBatchOperation op) {
176
177 // MatchAction operations at head (ingress) switches.
178 MatchActionOperations headOps = matchActionService.createOperationsList();
179
180 // MatchAction operations at rest of the switches.
181 MatchActionOperations tailOps = matchActionService.createOperationsList();
182
183 MatchActionOperations removeOps = matchActionService.createOperationsList();
184
185 for (BatchOperationEntry<Operator, ?> e : op.getOperations()) {
186
187 if (e.getOperator() == FlowBatchOperation.Operator.ADD) {
188 generateInstallMatchActionOperations(e, tailOps, headOps);
189 } else if (e.getOperator() == FlowBatchOperation.Operator.REMOVE) {
190 generateRemoveMatchActionOperations(e, removeOps);
191 } else {
192 throw new UnsupportedOperationException(
193 "FlowManager supports ADD and REMOVE operations only.");
194 }
195
196 }
197
198 return Arrays.asList(tailOps, headOps, removeOps);
199 }
200 */
201
202 /**
203 * Generates MatchActionOperations for an INSTALL FlowBatchOperation.
204 * <p/>
205 * FIXME: Currently only supports flows that generate exactly two match
206 * action operation sets.
207 *
208 * @param e Flow BatchOperationEntry
209 * @param tailOps MatchActionOperation set that the tail
210 * MatchActionOperations will be placed in
211 * @param headOps MatchActionOperation set that the head
212 * MatchActionOperations will be placed in
213 */
214 /*
215 private void generateInstallMatchActionOperations(
216 BatchOperationEntry<Operator, ?> e,
217 MatchActionOperations tailOps,
218 MatchActionOperations headOps) {
219
220 if (!(e.getTarget() instanceof Flow)) {
221 throw new IllegalStateException(
222 "The target is not Flow object: " + e.getTarget());
223 }
224
225 // Compile flows to match-actions
226 Flow flow = (Flow) e.getTarget();
227 List<MatchActionOperations> maOps = flow.compile(
228 e.getOperator(), matchActionService);
229 verifyNotNull(maOps, "Could not compile the flow: " + flow);
230 verify(maOps.size() == 2,
231 "The flow generates unspported match-action operations.");
232
233 // Map FlowId to MatchActionIds
234 for (MatchActionOperations maOp : maOps) {
235 for (MatchActionOperationEntry entry : maOp.getOperations()) {
236 flowMatchActionsMap.put(
237 KryoFactory.serialize(flow.getId()),
238 KryoFactory.serialize(entry.getTarget()));
239 }
240 }
241
242 // Merge match-action operations
243 for (MatchActionOperationEntry mae : maOps.get(0).getOperations()) {
244 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
245 tailOps.addOperation(mae);
246 }
247 for (MatchActionOperationEntry mae : maOps.get(1).getOperations()) {
248 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
249 headOps.addOperation(mae);
alshabib902d41b2014-10-07 16:52:05 -0700250 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700251 }
Brian O'Connorcb900f42014-10-07 21:55:33 -0700252 */
253 /**
254 * Generates MatchActionOperations for a REMOVE FlowBatchOperation.
255 *
256 * @param e Flow BatchOperationEntry
257 * @param removeOps MatchActionOperation set that the remove
258 * MatchActionOperations will be placed in
259 */
260 /*
261 private void generateRemoveMatchActionOperations(
262 BatchOperationEntry<Operator, ?> e,
263 MatchActionOperations removeOps) {
264
265 if (!(e.getTarget() instanceof FlowId)) {
266 throw new IllegalStateException(
267 "The target is not a FlowId object: " + e.getTarget());
268 }
269
270 // Compile flows to match-actions
271 FlowId flowId = (FlowId) e.getTarget();
272
273 for (byte[] matchActionIdBytes :
274 flowMatchActionsMap.remove(KryoFactory.serialize(flowId))) {
275 MatchActionId matchActionId = KryoFactory.deserialize(matchActionIdBytes);
276 removeOps.addOperation(new MatchActionOperationEntry(
277 MatchActionOperations.Operator.REMOVE, matchActionId));
278 }
279 }
280 */
Brian O'Connor66630c82014-10-02 21:08:19 -0700281}