blob: 081f68f63c59263f1e7b4a01b87abe5f0410a4d0 [file] [log] [blame]
/*
* Copyright 2017-present Open Networking Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.onosproject.pipelines.fabric.pipeliner;
import com.google.common.collect.ImmutableList;
import org.junit.Before;
import org.junit.Test;
import org.onosproject.net.flow.DefaultFlowRule;
import org.onosproject.net.flow.DefaultTrafficSelector;
import org.onosproject.net.flow.DefaultTrafficTreatment;
import org.onosproject.net.flow.FlowRule;
import org.onosproject.net.flow.TrafficSelector;
import org.onosproject.net.flow.TrafficTreatment;
import org.onosproject.net.flow.criteria.PiCriterion;
import org.onosproject.net.flowobjective.DefaultNextObjective;
import org.onosproject.net.flowobjective.NextObjective;
import org.onosproject.net.group.DefaultGroupBucket;
import org.onosproject.net.group.DefaultGroupDescription;
import org.onosproject.net.group.DefaultGroupKey;
import org.onosproject.net.group.GroupBucket;
import org.onosproject.net.group.GroupBuckets;
import org.onosproject.net.group.GroupDescription;
import org.onosproject.net.group.GroupKey;
import org.onosproject.net.pi.runtime.PiAction;
import org.onosproject.net.pi.runtime.PiActionGroupId;
import org.onosproject.net.pi.runtime.PiActionParam;
import org.onosproject.net.pi.runtime.PiGroupKey;
import org.onosproject.pipelines.fabric.FabricConstants;
import java.util.List;
import java.util.stream.Collectors;
import static org.junit.Assert.assertEquals;
/**
* Test cases for fabric.p4 pipeline next control block.
*/
public class FabricNextPipelinerTest extends FabricPipelinerTest {
private NextObjectiveTranslator translatorHashed;
private NextObjectiveTranslator translatorSimple;
private FlowRule vlanMetaFlowRule;
@Before
public void setup() {
super.doSetup();
translatorHashed = new NextObjectiveTranslator(DEVICE_ID, capabilitiesHashed);
translatorSimple = new NextObjectiveTranslator(DEVICE_ID, capabilitiesSimple);
PiCriterion nextIdCriterion = PiCriterion.builder()
.matchExact(FabricConstants.HDR_NEXT_ID, NEXT_ID_1)
.build();
TrafficSelector selector = DefaultTrafficSelector.builder()
.matchPi(nextIdCriterion)
.build();
PiAction piAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_SET_VLAN)
.withParameter(new PiActionParam(FabricConstants.VLAN_ID, VLAN_100.toShort()))
.build();
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.piTableAction(piAction)
.build();
vlanMetaFlowRule = DefaultFlowRule.builder()
.withSelector(selector)
.withTreatment(treatment)
.forTable(FabricConstants.FABRIC_INGRESS_NEXT_NEXT_VLAN)
.makePermanent()
// FIXME: currently next objective doesn't support priority, ignore this
.withPriority(0)
.forDevice(DEVICE_ID)
.fromApp(APP_ID)
.build();
}
/**
* Test program output rule for Simple table.
*/
@Test
public void testSimpleOutput() throws FabricPipelinerException {
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.setOutput(PORT_1)
.build();
PiAction piAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_OUTPUT_SIMPLE)
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
testSimple(treatment, piAction);
}
/**
* Test program set vlan and output rule for Simple table.
*/
@Test
public void testSimpleOutputWithVlanTranslation() throws FabricPipelinerException {
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.setVlanId(VLAN_100)
.setOutput(PORT_1)
.build();
PiAction piAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_OUTPUT_SIMPLE)
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
testSimple(treatment, piAction);
}
/**
* Test program set mac and output rule for Simple table.
*/
@Test
public void testSimpleOutputWithMacTranslation() throws FabricPipelinerException {
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.setEthSrc(ROUTER_MAC)
.setEthDst(HOST_MAC)
.setOutput(PORT_1)
.build();
PiAction piAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_ROUTING_SIMPLE)
.withParameter(new PiActionParam(
FabricConstants.SMAC, ROUTER_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.DMAC, HOST_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
testSimple(treatment, piAction);
}
/**
* Test program set mac, set vlan, and output rule for Simple table.
*/
@Test
public void testSimpleOutputWithVlanAndMacTranslation() throws FabricPipelinerException {
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.setEthSrc(ROUTER_MAC)
.setEthDst(HOST_MAC)
.setVlanId(VLAN_100)
.setOutput(PORT_1)
.build();
PiAction piAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_ROUTING_SIMPLE)
.withParameter(new PiActionParam(
FabricConstants.SMAC, ROUTER_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.DMAC, HOST_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
testSimple(treatment, piAction);
}
private void testSimple(TrafficTreatment treatment, PiAction piAction) throws FabricPipelinerException {
NextObjective nextObjective = DefaultNextObjective.builder()
.withId(NEXT_ID_1)
.withPriority(PRIORITY)
.withMeta(VLAN_META)
.addTreatment(treatment)
.withType(NextObjective.Type.SIMPLE)
.makePermanent()
.fromApp(APP_ID)
.add();
ObjectiveTranslation actualTranslation = translatorSimple.translate(nextObjective);
// Simple table
PiCriterion nextIdCriterion = PiCriterion.builder()
.matchExact(FabricConstants.HDR_NEXT_ID, NEXT_ID_1)
.build();
TrafficSelector nextIdSelector = DefaultTrafficSelector.builder()
.matchPi(nextIdCriterion)
.build();
FlowRule expectedFlowRule = DefaultFlowRule.builder()
.forDevice(DEVICE_ID)
.fromApp(APP_ID)
.makePermanent()
// FIXME: currently next objective doesn't support priority, ignore this
.withPriority(0)
.forTable(FabricConstants.FABRIC_INGRESS_NEXT_SIMPLE)
.withSelector(nextIdSelector)
.withTreatment(DefaultTrafficTreatment.builder()
.piTableAction(piAction).build())
.build();
ObjectiveTranslation expectedTranslation = ObjectiveTranslation.builder()
.addFlowRule(vlanMetaFlowRule)
.addFlowRule(expectedFlowRule)
.build();
assertEquals(expectedTranslation, actualTranslation);
}
/**
* Test program ecmp output group for Hashed table.
*/
@Test
public void testHashedOutput() throws Exception {
PiAction piAction1 = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_ROUTING_HASHED)
.withParameter(new PiActionParam(
FabricConstants.SMAC, ROUTER_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.DMAC, HOST_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
PiAction piAction2 = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_ROUTING_HASHED)
.withParameter(new PiActionParam(
FabricConstants.SMAC, ROUTER_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.DMAC, HOST_MAC.toBytes()))
.withParameter(new PiActionParam(
FabricConstants.PORT_NUM, PORT_1.toLong()))
.build();
TrafficTreatment treatment1 = DefaultTrafficTreatment.builder()
.piTableAction(piAction1)
.build();
TrafficTreatment treatment2 = DefaultTrafficTreatment.builder()
.piTableAction(piAction2)
.build();
NextObjective nextObjective = DefaultNextObjective.builder()
.withId(NEXT_ID_1)
.withPriority(PRIORITY)
.withMeta(VLAN_META)
.addTreatment(treatment1)
.addTreatment(treatment2)
.withType(NextObjective.Type.HASHED)
.makePermanent()
.fromApp(APP_ID)
.add();
ObjectiveTranslation actualTranslation = translatorHashed.doTranslate(nextObjective);
// Expected hashed table flow rule.
PiCriterion nextIdCriterion = PiCriterion.builder()
.matchExact(FabricConstants.HDR_NEXT_ID, NEXT_ID_1)
.build();
TrafficSelector nextIdSelector = DefaultTrafficSelector.builder()
.matchPi(nextIdCriterion)
.build();
PiActionGroupId actionGroupId = PiActionGroupId.of(NEXT_ID_1);
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.piTableAction(actionGroupId)
.build();
FlowRule expectedFlowRule = DefaultFlowRule.builder()
.forDevice(DEVICE_ID)
.fromApp(APP_ID)
.makePermanent()
// FIXME: currently next objective doesn't support priority, ignore this
.withPriority(0)
.forTable(FabricConstants.FABRIC_INGRESS_NEXT_HASHED)
.withSelector(nextIdSelector)
.withTreatment(treatment)
.build();
// Expected group
List<TrafficTreatment> treatments = ImmutableList.of(treatment1, treatment2);
List<GroupBucket> buckets = treatments.stream()
.map(DefaultGroupBucket::createSelectGroupBucket)
.collect(Collectors.toList());
GroupBuckets groupBuckets = new GroupBuckets(buckets);
PiGroupKey groupKey = new PiGroupKey(FabricConstants.FABRIC_INGRESS_NEXT_HASHED,
FabricConstants.FABRIC_INGRESS_NEXT_HASHED_SELECTOR,
NEXT_ID_1);
GroupDescription expectedGroup = new DefaultGroupDescription(
DEVICE_ID,
GroupDescription.Type.SELECT,
groupBuckets,
groupKey,
NEXT_ID_1,
APP_ID
);
ObjectiveTranslation expectedTranslation = ObjectiveTranslation.builder()
.addFlowRule(expectedFlowRule)
.addFlowRule(vlanMetaFlowRule)
.addGroup(expectedGroup)
.build();
assertEquals(expectedTranslation, actualTranslation);
}
/**
* Test program output group for Broadcast table.
*/
@Test
public void testBroadcastOutput() throws FabricPipelinerException {
TrafficTreatment treatment1 = DefaultTrafficTreatment.builder()
.setOutput(PORT_1)
.build();
TrafficTreatment treatment2 = DefaultTrafficTreatment.builder()
.popVlan()
.setOutput(PORT_2)
.build();
NextObjective nextObjective = DefaultNextObjective.builder()
.withId(NEXT_ID_1)
.withPriority(PRIORITY)
.addTreatment(treatment1)
.addTreatment(treatment2)
.withMeta(VLAN_META)
.withType(NextObjective.Type.BROADCAST)
.makePermanent()
.fromApp(APP_ID)
.add();
ObjectiveTranslation actualTranslation = translatorHashed.doTranslate(nextObjective);
// Should generate 3 flows:
// - Multicast table flow that matches on next-id and set multicast group (1)
// - Egress VLAN pop handling for treatment2 (0)
// - Next VLAN flow (2)
// And 2 groups:
// - Multicast group
// Expected multicast table flow rule.
PiCriterion nextIdCriterion = PiCriterion.builder()
.matchExact(FabricConstants.HDR_NEXT_ID, NEXT_ID_1)
.build();
TrafficSelector nextIdSelector = DefaultTrafficSelector.builder()
.matchPi(nextIdCriterion)
.build();
PiAction setMcGroupAction = PiAction.builder()
.withId(FabricConstants.FABRIC_INGRESS_NEXT_SET_MCAST_GROUP_ID)
.withParameter(new PiActionParam(
FabricConstants.GROUP_ID, NEXT_ID_1))
.build();
TrafficTreatment treatment = DefaultTrafficTreatment.builder()
.piTableAction(setMcGroupAction)
.build();
FlowRule expectedHashedFlowRule = DefaultFlowRule.builder()
.forDevice(DEVICE_ID)
.fromApp(APP_ID)
.makePermanent()
.withPriority(nextObjective.priority())
.forTable(FabricConstants.FABRIC_INGRESS_NEXT_MULTICAST)
.withSelector(nextIdSelector)
.withTreatment(treatment)
.build();
// Expected egress VLAN POP flow rule.
PiCriterion egressVlanTableMatch = PiCriterion.builder()
.matchExact(FabricConstants.HDR_EG_PORT, PORT_2.toLong())
.build();
TrafficSelector selectorForEgressVlan = DefaultTrafficSelector.builder()
.matchPi(egressVlanTableMatch)
.matchVlanId(VLAN_100)
.build();
PiAction piActionForEgressVlan = PiAction.builder()
.withId(FabricConstants.FABRIC_EGRESS_EGRESS_NEXT_POP_VLAN)
.build();
TrafficTreatment treatmentForEgressVlan = DefaultTrafficTreatment.builder()
.piTableAction(piActionForEgressVlan)
.build();
FlowRule expectedEgressVlanRule = DefaultFlowRule.builder()
.withSelector(selectorForEgressVlan)
.withTreatment(treatmentForEgressVlan)
.forTable(FabricConstants.FABRIC_EGRESS_EGRESS_NEXT_EGRESS_VLAN)
.makePermanent()
.withPriority(nextObjective.priority())
.forDevice(DEVICE_ID)
.fromApp(APP_ID)
.build();
// Expected ALL group.
TrafficTreatment allGroupTreatment1 = DefaultTrafficTreatment.builder()
.setOutput(PORT_1)
.build();
TrafficTreatment allGroupTreatment2 = DefaultTrafficTreatment.builder()
.setOutput(PORT_2)
.build();
List<TrafficTreatment> allTreatments = ImmutableList.of(
allGroupTreatment1, allGroupTreatment2);
List<GroupBucket> allBuckets = allTreatments.stream()
.map(DefaultGroupBucket::createAllGroupBucket)
.collect(Collectors.toList());
// FIXME: remove when we implement proper clone to CPU behavior
allBuckets.add(DefaultGroupBucket.createAllGroupBucket(
DefaultTrafficTreatment.builder().punt().build()));
GroupBuckets allGroupBuckets = new GroupBuckets(allBuckets);
GroupKey allGroupKey = new DefaultGroupKey(FabricPipeliner.KRYO.serialize(NEXT_ID_1));
GroupDescription expectedAllGroup = new DefaultGroupDescription(
DEVICE_ID,
GroupDescription.Type.ALL,
allGroupBuckets,
allGroupKey,
NEXT_ID_1,
APP_ID
);
ObjectiveTranslation expectedTranslation = ObjectiveTranslation.builder()
.addFlowRule(expectedHashedFlowRule)
.addFlowRule(vlanMetaFlowRule)
.addFlowRule(expectedEgressVlanRule)
.addGroup(expectedAllGroup)
.build();
assertEquals(expectedTranslation, actualTranslation);
}
}