Commit 5a2d0201 authored by Nikolay Aleksandrov's avatar Nikolay Aleksandrov Committed by Stephen Hemminger

bridge: vlan: add support to filter by vlan id

Add the optional keyword "vid" to bridge vlan show so the user can
request filtering by a specific vlan id. Currently the filtering is
implemented only in user-space. The argument name has been chosen to
match the add/del one - "vid". This filtering can be used also with the
"-compressvlans" option to see in which range is a vlan (if in any).
Also this will be used to show only specific per-vlan statistics later
when support is added to the kernel for it.

Examples:
$ bridge vlan show vid 450
port	vlan ids
eth2	 450

$ bridge -c vlan show vid 450
port	vlan ids
eth2	 400-500

$ bridge vlan show vid 1
port	vlan ids
eth1	 1 PVID Egress Untagged
eth2	 1 PVID
br0	 1 PVID Egress Untagged
Signed-off-by: default avatarNikolay Aleksandrov <nikolay@cumulusnetworks.com>
parent 24687d67
...@@ -13,13 +13,13 @@ ...@@ -13,13 +13,13 @@
#include "br_common.h" #include "br_common.h"
#include "utils.h" #include "utils.h"
static unsigned int filter_index; static unsigned int filter_index, filter_vlan;
static void usage(void) static void usage(void)
{ {
fprintf(stderr, "Usage: bridge vlan { add | del } vid VLAN_ID dev DEV [ pvid] [ untagged ]\n"); fprintf(stderr, "Usage: bridge vlan { add | del } vid VLAN_ID dev DEV [ pvid] [ untagged ]\n");
fprintf(stderr, " [ self ] [ master ]\n"); fprintf(stderr, " [ self ] [ master ]\n");
fprintf(stderr, " bridge vlan { show } [ dev DEV ]\n"); fprintf(stderr, " bridge vlan { show } [ dev DEV ] [ vid VLAN_ID ]\n");
exit(-1); exit(-1);
} }
...@@ -138,6 +138,26 @@ static int vlan_modify(int cmd, int argc, char **argv) ...@@ -138,6 +138,26 @@ static int vlan_modify(int cmd, int argc, char **argv)
return 0; return 0;
} }
/* In order to use this function for both filtering and non-filtering cases
* we need to make it a tristate:
* return -1 - if filtering we've gone over so don't continue
* return 0 - skip entry and continue (applies to range start or to entries
* which are less than filter_vlan)
* return 1 - print the entry and continue
*/
static int filter_vlan_check(struct bridge_vlan_info *vinfo)
{
/* if we're filtering we should stop on the first greater entry */
if (filter_vlan && vinfo->vid > filter_vlan &&
!(vinfo->flags & BRIDGE_VLAN_INFO_RANGE_END))
return -1;
if ((vinfo->flags & BRIDGE_VLAN_INFO_RANGE_BEGIN) ||
vinfo->vid < filter_vlan)
return 0;
return 1;
}
static int print_vlan(const struct sockaddr_nl *who, static int print_vlan(const struct sockaddr_nl *who,
struct nlmsghdr *n, struct nlmsghdr *n,
void *arg) void *arg)
...@@ -169,26 +189,40 @@ static int print_vlan(const struct sockaddr_nl *who, ...@@ -169,26 +189,40 @@ static int print_vlan(const struct sockaddr_nl *who,
/* if AF_SPEC isn't there, vlan table is not preset for this port */ /* if AF_SPEC isn't there, vlan table is not preset for this port */
if (!tb[IFLA_AF_SPEC]) { if (!tb[IFLA_AF_SPEC]) {
fprintf(fp, "%s\tNone\n", ll_index_to_name(ifm->ifi_index)); if (!filter_vlan)
fprintf(fp, "%s\tNone\n",
ll_index_to_name(ifm->ifi_index));
return 0; return 0;
} else { } else {
struct rtattr *i, *list = tb[IFLA_AF_SPEC]; struct rtattr *i, *list = tb[IFLA_AF_SPEC];
int rem = RTA_PAYLOAD(list); int rem = RTA_PAYLOAD(list);
__u16 last_vid_start = 0;
fprintf(fp, "%s", ll_index_to_name(ifm->ifi_index)); if (!filter_vlan)
fprintf(fp, "%s", ll_index_to_name(ifm->ifi_index));
for (i = RTA_DATA(list); RTA_OK(i, rem); i = RTA_NEXT(i, rem)) { for (i = RTA_DATA(list); RTA_OK(i, rem); i = RTA_NEXT(i, rem)) {
struct bridge_vlan_info *vinfo; struct bridge_vlan_info *vinfo;
int vcheck_ret;
if (i->rta_type != IFLA_BRIDGE_VLAN_INFO) if (i->rta_type != IFLA_BRIDGE_VLAN_INFO)
continue; continue;
vinfo = RTA_DATA(i); vinfo = RTA_DATA(i);
if (vinfo->flags & BRIDGE_VLAN_INFO_RANGE_END)
fprintf(fp, "-%hu", vinfo->vid); if (!(vinfo->flags & BRIDGE_VLAN_INFO_RANGE_END))
else last_vid_start = vinfo->vid;
fprintf(fp, "\t %hu", vinfo->vid); vcheck_ret = filter_vlan_check(vinfo);
if (vinfo->flags & BRIDGE_VLAN_INFO_RANGE_BEGIN) if (vcheck_ret == -1)
break;
else if (vcheck_ret == 0)
continue; continue;
if (filter_vlan)
fprintf(fp, "%s",
ll_index_to_name(ifm->ifi_index));
fprintf(fp, "\t %hu", last_vid_start);
if (last_vid_start != vinfo->vid)
fprintf(fp, "-%hu", vinfo->vid);
if (vinfo->flags & BRIDGE_VLAN_INFO_PVID) if (vinfo->flags & BRIDGE_VLAN_INFO_PVID)
fprintf(fp, " PVID"); fprintf(fp, " PVID");
if (vinfo->flags & BRIDGE_VLAN_INFO_UNTAGGED) if (vinfo->flags & BRIDGE_VLAN_INFO_UNTAGGED)
...@@ -196,7 +230,8 @@ static int print_vlan(const struct sockaddr_nl *who, ...@@ -196,7 +230,8 @@ static int print_vlan(const struct sockaddr_nl *who,
fprintf(fp, "\n"); fprintf(fp, "\n");
} }
} }
fprintf(fp, "\n"); if (!filter_vlan)
fprintf(fp, "\n");
fflush(fp); fflush(fp);
return 0; return 0;
} }
...@@ -211,6 +246,11 @@ static int vlan_show(int argc, char **argv) ...@@ -211,6 +246,11 @@ static int vlan_show(int argc, char **argv)
if (filter_dev) if (filter_dev)
duparg("dev", *argv); duparg("dev", *argv);
filter_dev = *argv; filter_dev = *argv;
} else if (strcmp(*argv, "vid") == 0) {
NEXT_ARG();
if (filter_vlan)
duparg("vid", *argv);
filter_vlan = atoi(*argv);
} }
argc--; argv++; argc--; argv++;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment