blob: 461f6705c0be7fb7ed707dc4429bbf7810ac2775 [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
alshabib92c65ad2014-10-08 21:56:05 -070070 private 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) {
141 if (intent.constraints() == null) {
142 return null;
143 }
144 LinkResourceRequest.Builder builder =
145 DefaultLinkResourceRequest.builder(intent.id(), intent.path().links());
146 for (Constraint constraint : intent.constraints()) {
147 builder.addConstraint(constraint);
148 }
149 LinkResourceRequest request = builder.build();
150 return request.resources().isEmpty() ? null : resourceService.requestResources(request);
Ray Milkeycaa450b2014-10-29 15:54:24 -0700151 }
152
Brian O'Connorcb900f42014-10-07 21:55:33 -0700153 // TODO refactor below this line... ----------------------------
154
155 /**
156 * Generates the series of MatchActionOperations from the
157 * {@link FlowBatchOperation}.
158 * <p>
159 * FIXME: Currently supporting PacketPathFlow and SingleDstTreeFlow only.
160 * <p>
161 * FIXME: MatchActionOperations should have dependency field to the other
162 * match action operations, and this method should use this.
163 *
164 * @param op the {@link FlowBatchOperation} object
165 * @return the list of {@link MatchActionOperations} objects
166 */
167 /*
168 private List<MatchActionOperations>
169 generateMatchActionOperationsList(FlowBatchOperation op) {
170
171 // MatchAction operations at head (ingress) switches.
172 MatchActionOperations headOps = matchActionService.createOperationsList();
173
174 // MatchAction operations at rest of the switches.
175 MatchActionOperations tailOps = matchActionService.createOperationsList();
176
177 MatchActionOperations removeOps = matchActionService.createOperationsList();
178
179 for (BatchOperationEntry<Operator, ?> e : op.getOperations()) {
180
181 if (e.getOperator() == FlowBatchOperation.Operator.ADD) {
182 generateInstallMatchActionOperations(e, tailOps, headOps);
183 } else if (e.getOperator() == FlowBatchOperation.Operator.REMOVE) {
184 generateRemoveMatchActionOperations(e, removeOps);
185 } else {
186 throw new UnsupportedOperationException(
187 "FlowManager supports ADD and REMOVE operations only.");
188 }
189
190 }
191
192 return Arrays.asList(tailOps, headOps, removeOps);
193 }
194 */
195
196 /**
197 * Generates MatchActionOperations for an INSTALL FlowBatchOperation.
198 * <p/>
199 * FIXME: Currently only supports flows that generate exactly two match
200 * action operation sets.
201 *
202 * @param e Flow BatchOperationEntry
203 * @param tailOps MatchActionOperation set that the tail
204 * MatchActionOperations will be placed in
205 * @param headOps MatchActionOperation set that the head
206 * MatchActionOperations will be placed in
207 */
208 /*
209 private void generateInstallMatchActionOperations(
210 BatchOperationEntry<Operator, ?> e,
211 MatchActionOperations tailOps,
212 MatchActionOperations headOps) {
213
214 if (!(e.getTarget() instanceof Flow)) {
215 throw new IllegalStateException(
216 "The target is not Flow object: " + e.getTarget());
217 }
218
219 // Compile flows to match-actions
220 Flow flow = (Flow) e.getTarget();
221 List<MatchActionOperations> maOps = flow.compile(
222 e.getOperator(), matchActionService);
223 verifyNotNull(maOps, "Could not compile the flow: " + flow);
224 verify(maOps.size() == 2,
225 "The flow generates unspported match-action operations.");
226
227 // Map FlowId to MatchActionIds
228 for (MatchActionOperations maOp : maOps) {
229 for (MatchActionOperationEntry entry : maOp.getOperations()) {
230 flowMatchActionsMap.put(
231 KryoFactory.serialize(flow.getId()),
232 KryoFactory.serialize(entry.getTarget()));
233 }
234 }
235
236 // Merge match-action operations
237 for (MatchActionOperationEntry mae : maOps.get(0).getOperations()) {
238 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
239 tailOps.addOperation(mae);
240 }
241 for (MatchActionOperationEntry mae : maOps.get(1).getOperations()) {
242 verify(mae.getOperator() == MatchActionOperations.Operator.INSTALL);
243 headOps.addOperation(mae);
alshabib902d41b2014-10-07 16:52:05 -0700244 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700245 }
Brian O'Connorcb900f42014-10-07 21:55:33 -0700246 */
247 /**
248 * Generates MatchActionOperations for a REMOVE FlowBatchOperation.
249 *
250 * @param e Flow BatchOperationEntry
251 * @param removeOps MatchActionOperation set that the remove
252 * MatchActionOperations will be placed in
253 */
254 /*
255 private void generateRemoveMatchActionOperations(
256 BatchOperationEntry<Operator, ?> e,
257 MatchActionOperations removeOps) {
258
259 if (!(e.getTarget() instanceof FlowId)) {
260 throw new IllegalStateException(
261 "The target is not a FlowId object: " + e.getTarget());
262 }
263
264 // Compile flows to match-actions
265 FlowId flowId = (FlowId) e.getTarget();
266
267 for (byte[] matchActionIdBytes :
268 flowMatchActionsMap.remove(KryoFactory.serialize(flowId))) {
269 MatchActionId matchActionId = KryoFactory.deserialize(matchActionIdBytes);
270 removeOps.addOperation(new MatchActionOperationEntry(
271 MatchActionOperations.Operator.REMOVE, matchActionId));
272 }
273 }
274 */
Brian O'Connor66630c82014-10-02 21:08:19 -0700275}