Skip to content

Commit

Permalink
Add support for switch network instance with multiple ports
Browse files Browse the repository at this point in the history
This allows to:
- bridge multiple switches and add redundant links. STP is used to avoid
  bridge loops.
- connect end-devices into the same L2 segment as applications running
  on the edge node. VLAN access ports are now supported not only for
  application VIFs, but also for physical device network ports.
  BPDU guards prevent end-devices to participate in STP.

Signed-off-by: Milan Lenco <[email protected]>
  • Loading branch information
milan-zededa authored and eriknordmark committed Aug 29, 2024
1 parent acb62b6 commit 7a6981b
Show file tree
Hide file tree
Showing 22 changed files with 1,219 additions and 297 deletions.
56 changes: 45 additions & 11 deletions docs/APP-CONNECTIVITY.md
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,7 @@ table. If flow logging is disabled, packet is allowed or dropped immediately usi
actions.
If the packet is allowed and the destination is external, packet will be transmitted out through
one of the network ports used by the network instance.
Note that if NI is switch (L2 only), packet will be just forwarded through the (single) port,
Note that if NI is switch (L2 only), packet will be just forwarded through one of the ports,
not routed.
If the packet is allowed and the destination is another app (inside the same NI), packet's dst
MAC address will be set to the dst app MAC, and it will be forwarded again through the same bridge
Expand All @@ -115,7 +115,7 @@ actions.
If the packet is allowed, it will match the connected route of the NI bridge and gets forwarded
via the bridge and the VIF into the application.
Note that in case of a switch NI (L2 only), D-NAT and routing operations are not performed
and inbound packets are simply forwarded from the port, through the bridge and VIF into
and inbound packets are simply forwarded from the input port, through the bridge and VIF into
the application (still the ACLs do apply).
The diagram below depicts all these inbound packet-flow stages:

Expand Down Expand Up @@ -219,7 +219,7 @@ More information about metadata server can be found in [ECO-METADATA.md](ECO-MET
### Switch Network Instance

Switch Network Instance is a simple L2-only bridge between connected applications and
(optionally) a device network port. Traffic is only forwarded by the host network stack.
(optionally) one or more network ports. Traffic is only forwarded by the host network stack.
This allows applications to directly access external endpoints and vice-versa.
Switch network can be configured without port (i.e. as air-gapped), in which case it is merely
a bridge between application VIFs.
Expand Down Expand Up @@ -255,17 +255,50 @@ Predefined shared labels are:
* `uplink`: assigned to every management port
* `freeuplink`: assigned to every management port with zero cost

Please note that switch network instance is currently limited to one port at most.
However, the plan is to support attaching multiple ports to a single bridge and run
Spanning Tree Protocol to avoid bridge loops and the broadcast storm that results
from them.

Network instance (both Local and Switch) can be configured without any port. In this case,
the network is "air-gapped", meaning that it is not reachable from outside and, likewise,
it will not provide external connectivity to the applications. Air-gap NIs are used only
to connect applications running on the same edge device.

### Network Instance IP Routing
### Multi-port Switch Network Instance

A switch network instance with multiple ports enables the following:

* Bridging multiple switches while adding redundant links. STP is used to prevent bridge loops.
* Connecting end-devices (e.g. sensors) to the same Layer 2 segment as applications running
on the edge node.

The user can use a shared label to select multiple network ports for a switch network
instance. For a port to be eligible for bridging with other ports, it must be configured
as app-shared and with DHCP passthrough - meaning it should not have a static IP address
or run a DHCP client.

EVE automatically runs the traditional IEEE 802.1D Spanning Tree Protocol (STP) for bridges
with multiple ports to prevent bridge loops and the broadcast storms that would result from
them. Users can enable BPDU guard on ports intended to connect end-devices, which are not
expected to participate in the STP algorithm. Application VIFs always have BPDU guard enabled.

### VLAN-aware Switch Network Instance

By default, a Switch Network Instance does not apply any special handling to VLANs.
It operates like a traditional Layer 2 switch, forwarding traffic based solely on MAC
addresses without considering VLAN tags. This means that the bridge forwards both tagged
and untagged frames without altering or interpreting the VLAN tags. As a result, any VLAN-tagged
traffic will pass through the bridge unchanged, and no VLAN filtering or segregation will
occur unless explicitly configured.

To enable VLAN filtering and make the Switch Network Instance VLAN-aware, the user must
designate at least one application VIF or NI port as a VLAN Access Port for a specific
VLAN ID. Multiple applications connected to the same Switch Network Instance can either
access the same VLAN - allowing direct communication with one another - or be separated
into different virtual network segments. In the EVE API, access VLANs are assigned to NI
ports within the NetworkInstanceConfig, while VLAN configurations for application interfaces
are applied separately via the AppInstanceConfig.

VIFs and NI ports that are not designated as VLAN access ports are configured by EVE
as trunk ports, allowing all VLANs with at least one access port.

### Local Network Instance IP Routing

Local Network Instance with multiple ports will have link-local and connected routes
from all the ports present in its routing table. Additionally, user may configure
Expand Down Expand Up @@ -307,7 +340,7 @@ network signal strength.

#### Network Instance Default IP Route

A typical use case for multi-path routing is the default route. NI with multiple
A typical use case for multi-path routing is the default route. Local NI with multiple
ports that have gateway IP defined cannot install default route for each of them
into the NI routing table. Load-balancing is not supported and using different metrics
(default behaviour in Linux with multiple default routes) would result in only
Expand All @@ -328,7 +361,7 @@ Most of the edge deployments will deploy applications that will have connectivit
WAN (Internet) and LAN (e.g. shop floor, machine floor).
There may be a single WAN port and multiple LAN ports.
User has the option to either create a single local network instance with all those ports
assigned and use the [IP routing capabilities of the network instance](#network-instance-ip-routing),
assigned and use the [IP routing capabilities of the network instance](#local-network-instance-ip-routing),
or create a separate instance for every port and use DHCP-based propagation of IP routes
into applications. The latter option, described below, is more difficult to configure
but gives the application full control over IP routing.
Expand Down Expand Up @@ -407,6 +440,7 @@ For container applications running inside an EVE-created shim-VM, EVE initialize
of interfaces during the shim-VM boot. MTUs of all interfaces are passed to the VM via kernel
boot arguments (/proc/cmdline). The init script parses out these values and applies them
to application interfaces (excluding direct assignments).

Furthermore, interfaces connected to local network instances will have their MTUs
automatically updated using DHCP if there is a change in the MTU configuration. To update
the MTU of interfaces connected to switch network instances, user may run an external
Expand Down
4 changes: 2 additions & 2 deletions docs/DEVICE-CONNECTIVITY.md
Original file line number Diff line number Diff line change
Expand Up @@ -30,8 +30,8 @@ are user-defined, simpler human-readable strings, which are only required to be
the edge-node, not globally.
Additionally, (predefined or user-defined) "shared" labels are not unique and the same label
can be assigned to multiple ports. This is currently used to reference and attach multiple
ports into a local network instance. Also, IP route with multiple possible paths may reference
a group of suitable output ports using an adapter shared label.
ports into a local or a switch network instance. Also, IP route with multiple possible paths may
reference a group of suitable output ports using an adapter shared label.

### Physical network ports

Expand Down
14 changes: 8 additions & 6 deletions docs/NETWORKING.md
Original file line number Diff line number Diff line change
Expand Up @@ -82,11 +82,12 @@ EVE provides 2 types of network instances:
- one or more network ports can be attached to network instance to provide
external connectivity
- **Switch**:
- simple L2-only switch between connected applications and a network port
(currently limited to one port at most)
- simple L2-only switch between connected applications and network ports
- traffic is only forwarded
- does not run DHCP server (but can be used in combination with external DHCP server)
- allows applications to directly connect to external network segments
- allows applications to directly connect to external network segments and, conversely,
permits external endpoints to directly access the applications, without routing or NAT
in between

Both types of network instances can be air-gapped, i.e., without any network port attached
and thus without external connectivity.
Expand Down Expand Up @@ -115,9 +116,10 @@ distributions, and could be confusing even for experienced Linux users, for exam
- Main IP routing table (`ip route show table main`) is not used, instead every (IP-enabled)
network adapter and every local network instance has its own routing table, which is selected
using IP rules by the source IP address of a given packet (see `ip rule list`).
- Network adapter for ethernet port is implemented as a bridge which enslaves the port and takes
its interface name and MAC address - this way we can then put application interfaces under this
bridge to implement switch network instance
- (L3-enabled) Network adapter for ethernet port is implemented as a bridge which enslaves
the port and takes its interface name and MAC address - this way we can then use the bridge
as an IP endpoint and at the same time put application interfaces under this bridge
to implement (single-port) switch network instance
- If flow logging is enabled, ACLs are implemented by iptables `mangle` table for the most
part (not `filter`). Traffic flow that should not be allowed is marked with a "DROP mark"
and routed into a dummy `blackhole` interface. The purpose of this is to ensure that conntrack
Expand Down
25 changes: 25 additions & 0 deletions pkg/pillar/cmd/zedagent/parseconfig.go
Original file line number Diff line number Diff line change
Expand Up @@ -450,6 +450,19 @@ func parseStaticRoute(route *zconfig.IPRoute, config *types.NetworkInstanceConfi
return nil
}

func parseVlanAccessPort(accessPort *zconfig.VlanAccessPort,
config *types.NetworkInstanceConfig) error {
vlanID := accessPort.GetVlanId()
if vlanID < minVlanID || vlanID > maxVlanID {
return fmt.Errorf("VLAN ID out of range: %d", vlanID)
}
config.VlanAccessPorts = append(config.VlanAccessPorts, types.VlanAccessPort{
VlanID: uint16(vlanID),
PortLabel: accessPort.GetAccessPort(),
})
return nil
}

func publishNetworkInstanceConfig(ctx *getconfigContext,
networkInstances []*zconfig.NetworkInstanceConfig) {

Expand Down Expand Up @@ -479,6 +492,9 @@ func publishNetworkInstanceConfig(ctx *getconfigContext,
PortLabel: apiConfigEntry.Port.GetName(),
PropagateConnRoutes: apiConfigEntry.PropagateConnectedRoutes,
EnableFlowlog: !apiConfigEntry.DisableFlowlog,
STPConfig: types.STPConfig{
PortsWithBpduGuard: apiConfigEntry.GetStp().GetPortsWithBpduGuard(),
},
}
uuidStr := networkInstanceConfig.UUID.String()
log.Functionf("publishNetworkInstanceConfig: processing %s %s type %d activate %v",
Expand All @@ -497,6 +513,15 @@ func publishNetworkInstanceConfig(ctx *getconfigContext,
// Let's relax the requirement until cloud side update the right IpType
networkInstanceConfig.IpType = types.AddressTypeNone
}
for _, accessPort := range apiConfigEntry.VlanAccessPorts {
err := parseVlanAccessPort(accessPort, &networkInstanceConfig)
if err != nil {
errStr := fmt.Sprintf("Invalid VLAN access port config: %v", err)
log.Errorf("publishNetworkInstanceConfig (%s): %s", uuidStr, errStr)
networkInstanceConfig.SetErrorNow(errStr)
// Proceed to send error back to controller
}
}
}

// other than switch-type(l2)
Expand Down
Loading

0 comments on commit 7a6981b

Please sign in to comment.