blob: 7a92b4fa78a9879edcf703fa469c6d4b78435ea5 [file] [log] [blame]
danielc8620b12015-11-30 15:50:59 +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 */
Hyunsun Moon21d17ba2015-12-04 16:16:25 -080016package org.onosproject.driver.pipeline;
danielc8620b12015-11-30 15:50:59 +090017
18import org.onlab.osgi.ServiceDirectory;
19import org.onlab.packet.Ethernet;
20import org.onosproject.core.ApplicationId;
21import org.onosproject.core.CoreService;
22import org.onosproject.net.DeviceId;
23import org.onosproject.net.behaviour.Pipeliner;
24import org.onosproject.net.behaviour.PipelinerContext;
25import org.onosproject.net.flow.DefaultFlowRule;
26import org.onosproject.net.flow.DefaultTrafficSelector;
27import org.onosproject.net.flow.DefaultTrafficTreatment;
28import org.onosproject.net.flow.FlowRule;
29import org.onosproject.net.flow.FlowRuleOperations;
30import org.onosproject.net.flow.FlowRuleOperationsContext;
31import org.onosproject.net.flow.FlowRuleService;
32import org.onosproject.net.flow.TrafficSelector;
33import org.onosproject.net.flow.TrafficTreatment;
34import org.onosproject.net.flow.criteria.Criterion;
35import org.onosproject.net.flow.criteria.EthTypeCriterion;
36import org.onosproject.net.flow.criteria.UdpPortCriterion;
37import org.onosproject.net.flowobjective.FilteringObjective;
38import org.onosproject.net.flowobjective.FlowObjectiveStore;
39import org.onosproject.net.flowobjective.ForwardingObjective;
40import org.onosproject.net.flowobjective.NextObjective;
41import org.onosproject.net.flowobjective.Objective;
42import org.onosproject.net.flowobjective.ObjectiveError;
43import org.slf4j.Logger;
44
45import java.util.Collection;
46import java.util.Collections;
47
48import static org.slf4j.LoggerFactory.getLogger;
49
50/**
51 * Driver for OpenstackSwitching.
52 */
53public class OpenstackPipeline extends DefaultSingleTablePipeline
54 implements Pipeliner {
55
56 private final Logger log = getLogger(getClass());
57 private CoreService coreService;
58 private ServiceDirectory serviceDirectory;
59 protected FlowObjectiveStore flowObjectiveStore;
60 protected DeviceId deviceId;
61 protected ApplicationId appId;
62 protected FlowRuleService flowRuleService;
63
64 protected static final int VNI_TABLE = 0;
65 protected static final int FORWARDING_TABLE = 1;
66
67 private static final int DROP_PRIORITY = 0;
68 private static final int TIME_OUT = 0;
69 private static final int DHCP_SERVER_PORT = 67;
70 private static final int DHCP_CLIENT_PORT = 68;
71
72
73 @Override
74 public void init(DeviceId deviceId, PipelinerContext context) {
75 super.init(deviceId, context);
76 this.serviceDirectory = context.directory();
77 this.deviceId = deviceId;
78
79 coreService = serviceDirectory.get(CoreService.class);
80 flowRuleService = serviceDirectory.get(FlowRuleService.class);
81 flowObjectiveStore = context.store();
82
83 appId = coreService.registerApplication(
84 "org.onosproject.driver.OpenstackPipeline");
85
86 initializePipeline();
87 }
88
89 @Override
90 public void filter(FilteringObjective filteringObjective) {
91 super.filter(filteringObjective);
92 }
93
94 @Override
95 public void next(NextObjective nextObjective) {
96 super.next(nextObjective);
97 }
98
99 @Override
100 public void forward(ForwardingObjective forwardingObjective) {
101 Collection<FlowRule> rules;
102 FlowRuleOperations.Builder flowOpsBuilder = FlowRuleOperations.builder();
103
104 rules = processForward(forwardingObjective);
105
106 switch (forwardingObjective.op()) {
107 case ADD:
108 rules.stream()
109 .filter(rule -> rule != null)
110 .forEach(flowOpsBuilder::add);
111 break;
112 case REMOVE:
113 rules.stream()
114 .filter(rule -> rule != null)
115 .forEach(flowOpsBuilder::remove);
116 break;
117 default:
118 fail(forwardingObjective, ObjectiveError.UNKNOWN);
119 log.warn("Unknown forwarding type {}");
120 }
121
122 flowRuleService.apply(flowOpsBuilder.build(new FlowRuleOperationsContext() {
123 @Override
124 public void onSuccess(FlowRuleOperations ops) {
125 pass(forwardingObjective);
126 }
127
128 @Override
129 public void onError(FlowRuleOperations ops) {
130 fail(forwardingObjective, ObjectiveError.FLOWINSTALLATIONFAILED);
131 }
132 }));
133 }
134
135 private void initializePipeline() {
136 processVNITable(true);
137 processForwardingTable(true);
138 }
139
140 private void processVNITable(boolean install) {
141 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
142 TrafficTreatment.Builder treatment = DefaultTrafficTreatment.builder();
143
144 treatment.transition(FORWARDING_TABLE);
145
146 FlowRule flowRule = DefaultFlowRule.builder()
147 .forDevice(deviceId)
148 .withSelector(selector.build())
149 .withTreatment(treatment.build())
150 .withPriority(DROP_PRIORITY)
151 .fromApp(appId)
152 .makePermanent()
153 .forTable(VNI_TABLE)
154 .build();
155
156 applyRules(install, flowRule);
157 }
158
159 private void processForwardingTable(boolean install) {
160 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
161 TrafficTreatment.Builder treatment = DefaultTrafficTreatment.builder();
162
163 treatment.drop();
164
165 FlowRule flowRule = DefaultFlowRule.builder()
166 .forDevice(deviceId)
167 .withSelector(selector.build())
168 .withTreatment(treatment.build())
169 .withPriority(DROP_PRIORITY)
170 .fromApp(appId)
171 .makePermanent()
172 .forTable(FORWARDING_TABLE)
173 .build();
174
175 applyRules(install, flowRule);
176 }
177
178 private void applyRules(boolean install, FlowRule flowRule) {
179 FlowRuleOperations.Builder flowOpsBuilder = FlowRuleOperations.builder();
180
181 flowOpsBuilder = install ? flowOpsBuilder.add(flowRule) : flowOpsBuilder.remove(flowRule);
182
183 flowRuleService.apply(flowOpsBuilder.build(new FlowRuleOperationsContext() {
184 @Override
185 public void onSuccess(FlowRuleOperations ops) {
186 log.debug("Provisioned vni or forwarding table");
187 }
188
189 @Override
190 public void onError(FlowRuleOperations ops) {
191 log.debug("Failed to privision vni or forwarding table");
192 }
193 }));
194 }
195
196 private Collection<FlowRule> processForward(ForwardingObjective forwardingObjective) {
197 switch (forwardingObjective.flag()) {
198 case SPECIFIC:
199 return processSpecific(forwardingObjective);
200 case VERSATILE:
201 return processVersatile(forwardingObjective);
202 default:
203 fail(forwardingObjective, ObjectiveError.UNKNOWN);
204 log.warn("Unknown forwarding flag {}", forwardingObjective.flag());
205 }
206 return Collections.emptySet();
207 }
208
209 private Collection<FlowRule> processVersatile(ForwardingObjective forwardingObjective) {
210 log.debug("Processing versatile forwarding objective");
211
212 FlowRule.Builder ruleBuilder = DefaultFlowRule.builder()
213 .forDevice(deviceId)
214 .withSelector(forwardingObjective.selector())
215 .withTreatment(forwardingObjective.treatment())
216 .withPriority(forwardingObjective.priority())
217 .fromApp(forwardingObjective.appId());
218
219 if (forwardingObjective.permanent()) {
220 ruleBuilder.makePermanent();
221 } else {
222 ruleBuilder.makeTemporary(TIME_OUT);
223 }
224
225 //ARP & DHCP Rule
226 EthTypeCriterion ethCriterion =
227 (EthTypeCriterion) forwardingObjective.selector().getCriterion(Criterion.Type.ETH_TYPE);
228 UdpPortCriterion udpPortCriterion = (UdpPortCriterion) forwardingObjective
229 .selector().getCriterion(Criterion.Type.UDP_DST);
230 if (ethCriterion != null) {
231 if (ethCriterion.ethType().toShort() == Ethernet.TYPE_ARP ||
232 ethCriterion.ethType().toShort() == Ethernet.TYPE_LLDP) {
233 ruleBuilder.forTable(VNI_TABLE);
234 return Collections.singletonList(ruleBuilder.build());
235 } else if (udpPortCriterion != null && udpPortCriterion.udpPort().toInt() == DHCP_SERVER_PORT) {
236 ruleBuilder.forTable(VNI_TABLE);
237 return Collections.singletonList(ruleBuilder.build());
238 }
239 }
240 return Collections.emptySet();
241 }
242
243 private Collection<FlowRule> processSpecific(ForwardingObjective forwardingObjective) {
244 log.debug("Processing specific forwarding objective");
245
246 FlowRule.Builder ruleBuilder = DefaultFlowRule.builder()
247 .forDevice(deviceId)
248 .withSelector(forwardingObjective.selector())
249 .withTreatment(forwardingObjective.treatment())
250 .withPriority(forwardingObjective.priority())
251 .fromApp(forwardingObjective.appId());
252
253 if (forwardingObjective.permanent()) {
254 ruleBuilder.makePermanent();
255 } else {
256 ruleBuilder.makeTemporary(TIME_OUT);
257 }
258
259 //VNI Table Rule
260 if (forwardingObjective.selector().getCriterion(Criterion.Type.IN_PORT) != null) {
261 TrafficTreatment.Builder tBuilder = DefaultTrafficTreatment.builder();
262 forwardingObjective.treatment().allInstructions().forEach(tBuilder::add);
263 tBuilder.transition(FORWARDING_TABLE);
264 ruleBuilder.withTreatment(tBuilder.build());
265 ruleBuilder.forTable(VNI_TABLE);
266 } else {
267 ruleBuilder.forTable(FORWARDING_TABLE);
268 }
269
270 return Collections.singletonList(ruleBuilder.build());
271 }
272
273
274 private void pass(Objective obj) {
275 if (obj.context().isPresent()) {
276 obj.context().get().onSuccess(obj);
277 }
278 }
279
280 private void fail(Objective obj, ObjectiveError error) {
281 if (obj.context().isPresent()) {
282 obj.context().get().onError(obj, error);
283 }
284 }
285}
286