Blame dcb/dcb_app.c

Packit Bot 867fae
// SPDX-License-Identifier: GPL-2.0+
Packit Bot 867fae
Packit Bot 867fae
#include <errno.h>
Packit Bot 867fae
#include <inttypes.h>
Packit Bot 867fae
#include <stdio.h>
Packit Bot 867fae
#include <libmnl/libmnl.h>
Packit Bot 867fae
#include <linux/dcbnl.h>
Packit Bot 867fae
Packit Bot 867fae
#include "dcb.h"
Packit Bot 867fae
#include "utils.h"
Packit Bot 867fae
#include "rt_names.h"
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_help_add(void)
Packit Bot 867fae
{
Packit Bot 867fae
	fprintf(stderr,
Packit Bot 867fae
		"Usage: dcb app { add | del | replace } dev STRING\n"
Packit Bot 867fae
		"           [ default-prio PRIO ]\n"
Packit Bot 867fae
		"           [ ethtype-prio ET:PRIO ]\n"
Packit Bot 867fae
		"           [ stream-port-prio PORT:PRIO ]\n"
Packit Bot 867fae
		"           [ dgram-port-prio PORT:PRIO ]\n"
Packit Bot 867fae
		"           [ port-prio PORT:PRIO ]\n"
Packit Bot 867fae
		"           [ dscp-prio INTEGER:PRIO ]\n"
Packit Bot 867fae
		"\n"
Packit Bot 867fae
		" where PRIO := { 0 .. 7 }\n"
Packit Bot 867fae
		"       ET := { 0x600 .. 0xffff }\n"
Packit Bot 867fae
		"       PORT := { 1 .. 65535 }\n"
Packit Bot 867fae
		"       DSCP := { 0 .. 63 }\n"
Packit Bot 867fae
		"\n"
Packit Bot 867fae
	);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_help_show_flush(void)
Packit Bot 867fae
{
Packit Bot 867fae
	fprintf(stderr,
Packit Bot 867fae
		"Usage: dcb app { show | flush } dev STRING\n"
Packit Bot 867fae
		"           [ default-prio ]\n"
Packit Bot 867fae
		"           [ ethtype-prio ]\n"
Packit Bot 867fae
		"           [ stream-port-prio ]\n"
Packit Bot 867fae
		"           [ dgram-port-prio ]\n"
Packit Bot 867fae
		"           [ port-prio ]\n"
Packit Bot 867fae
		"           [ dscp-prio ]\n"
Packit Bot 867fae
		"\n"
Packit Bot 867fae
	);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_help(void)
Packit Bot 867fae
{
Packit Bot 867fae
	fprintf(stderr,
Packit Bot 867fae
		"Usage: dcb app help\n"
Packit Bot 867fae
		"\n"
Packit Bot 867fae
	);
Packit Bot 867fae
	dcb_app_help_show_flush();
Packit Bot 867fae
	dcb_app_help_add();
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
struct dcb_app_table {
Packit Bot 867fae
	struct dcb_app *apps;
Packit Bot 867fae
	size_t n_apps;
Packit Bot 867fae
};
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_table_fini(struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	free(tab->apps);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_table_push(struct dcb_app_table *tab, struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app *apps = realloc(tab->apps, (tab->n_apps + 1) * sizeof(*tab->apps));
Packit Bot 867fae
Packit Bot 867fae
	if (apps == NULL) {
Packit Bot 867fae
		perror("Cannot allocate APP table");
Packit Bot 867fae
		return -ENOMEM;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	tab->apps = apps;
Packit Bot 867fae
	tab->apps[tab->n_apps++] = *app;
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_table_remove_existing(struct dcb_app_table *a,
Packit Bot 867fae
					  const struct dcb_app_table *b)
Packit Bot 867fae
{
Packit Bot 867fae
	size_t ia, ja;
Packit Bot 867fae
	size_t ib;
Packit Bot 867fae
Packit Bot 867fae
	for (ia = 0, ja = 0; ia < a->n_apps; ia++) {
Packit Bot 867fae
		struct dcb_app *aa = &a->apps[ia];
Packit Bot 867fae
		bool found = false;
Packit Bot 867fae
Packit Bot 867fae
		for (ib = 0; ib < b->n_apps; ib++) {
Packit Bot 867fae
			const struct dcb_app *ab = &b->apps[ib];
Packit Bot 867fae
Packit Bot 867fae
			if (aa->selector == ab->selector &&
Packit Bot 867fae
			    aa->protocol == ab->protocol &&
Packit Bot 867fae
			    aa->priority == ab->priority) {
Packit Bot 867fae
				found = true;
Packit Bot 867fae
				break;
Packit Bot 867fae
			}
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		if (!found)
Packit Bot 867fae
			a->apps[ja++] = *aa;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	a->n_apps = ja;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_table_remove_replaced(struct dcb_app_table *a,
Packit Bot 867fae
					  const struct dcb_app_table *b)
Packit Bot 867fae
{
Packit Bot 867fae
	size_t ia, ja;
Packit Bot 867fae
	size_t ib;
Packit Bot 867fae
Packit Bot 867fae
	for (ia = 0, ja = 0; ia < a->n_apps; ia++) {
Packit Bot 867fae
		struct dcb_app *aa = &a->apps[ia];
Packit Bot 867fae
		bool present = false;
Packit Bot 867fae
		bool found = false;
Packit Bot 867fae
Packit Bot 867fae
		for (ib = 0; ib < b->n_apps; ib++) {
Packit Bot 867fae
			const struct dcb_app *ab = &b->apps[ib];
Packit Bot 867fae
Packit Bot 867fae
			if (aa->selector == ab->selector &&
Packit Bot 867fae
			    aa->protocol == ab->protocol)
Packit Bot 867fae
				present = true;
Packit Bot 867fae
			else
Packit Bot 867fae
				continue;
Packit Bot 867fae
Packit Bot 867fae
			if (aa->priority == ab->priority) {
Packit Bot 867fae
				found = true;
Packit Bot 867fae
				break;
Packit Bot 867fae
			}
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		/* Entries that remain in A will be removed, so keep in the
Packit Bot 867fae
		 * table only APP entries whose sel/pid is mentioned in B,
Packit Bot 867fae
		 * but that do not have the full sel/pid/prio match.
Packit Bot 867fae
		 */
Packit Bot 867fae
		if (present && !found)
Packit Bot 867fae
			a->apps[ja++] = *aa;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	a->n_apps = ja;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_table_copy(struct dcb_app_table *a,
Packit Bot 867fae
			      const struct dcb_app_table *b)
Packit Bot 867fae
{
Packit Bot 867fae
	size_t i;
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	for (i = 0; i < b->n_apps; i++) {
Packit Bot 867fae
		ret = dcb_app_table_push(a, &b->apps[i]);
Packit Bot 867fae
		if (ret != 0)
Packit Bot 867fae
			return ret;
Packit Bot 867fae
	}
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_cmp(const struct dcb_app *a, const struct dcb_app *b)
Packit Bot 867fae
{
Packit Bot 867fae
	if (a->protocol < b->protocol)
Packit Bot 867fae
		return -1;
Packit Bot 867fae
	if (a->protocol > b->protocol)
Packit Bot 867fae
		return 1;
Packit Bot 867fae
	return a->priority - b->priority;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_cmp_cb(const void *a, const void *b)
Packit Bot 867fae
{
Packit Bot 867fae
	return dcb_app_cmp(a, b);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_table_sort(struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	qsort(tab->apps, tab->n_apps, sizeof(*tab->apps), dcb_app_cmp_cb);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
struct dcb_app_parse_mapping {
Packit Bot 867fae
	__u8 selector;
Packit Bot 867fae
	struct dcb_app_table *tab;
Packit Bot 867fae
	int err;
Packit Bot 867fae
};
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_parse_mapping_cb(__u32 key, __u64 value, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_parse_mapping *pm = data;
Packit Bot 867fae
	struct dcb_app app = {
Packit Bot 867fae
		.selector = pm->selector,
Packit Bot 867fae
		.priority = value,
Packit Bot 867fae
		.protocol = key,
Packit Bot 867fae
	};
Packit Bot 867fae
Packit Bot 867fae
	if (pm->err)
Packit Bot 867fae
		return;
Packit Bot 867fae
Packit Bot 867fae
	pm->err = dcb_app_table_push(pm->tab, &app);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_parse_mapping_ethtype_prio(__u32 key, char *value, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	__u8 prio;
Packit Bot 867fae
Packit Bot 867fae
	if (key < 0x600) {
Packit Bot 867fae
		fprintf(stderr, "Protocol IDs < 0x600 are reserved for EtherType\n");
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	if (get_u8(&prio, value, 0))
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
Packit Bot 867fae
	return dcb_parse_mapping("ETHTYPE", key, 0xffff,
Packit Bot 867fae
				 "PRIO", prio, IEEE_8021QAZ_MAX_TCS - 1,
Packit Bot 867fae
				 dcb_app_parse_mapping_cb, data);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_parse_dscp(__u32 *key, const char *arg)
Packit Bot 867fae
{
Packit Bot 867fae
	if (parse_mapping_num_all(key, arg) == 0)
Packit Bot 867fae
		return 0;
Packit Bot 867fae
Packit Bot 867fae
	if (rtnl_dsfield_a2n(key, arg) != 0)
Packit Bot 867fae
		return -1;
Packit Bot 867fae
Packit Bot 867fae
	if (*key & 0x03) {
Packit Bot 867fae
		fprintf(stderr, "The values `%s' uses non-DSCP bits.\n", arg);
Packit Bot 867fae
		return -1;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	/* Unshift the value to convert it from dsfield to DSCP. */
Packit Bot 867fae
	*key >>= 2;
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_parse_mapping_dscp_prio(__u32 key, char *value, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	__u8 prio;
Packit Bot 867fae
Packit Bot 867fae
	if (get_u8(&prio, value, 0))
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
Packit Bot 867fae
	return dcb_parse_mapping("DSCP", key, 63,
Packit Bot 867fae
				 "PRIO", prio, IEEE_8021QAZ_MAX_TCS - 1,
Packit Bot 867fae
				 dcb_app_parse_mapping_cb, data);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_parse_mapping_port_prio(__u32 key, char *value, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	__u8 prio;
Packit Bot 867fae
Packit Bot 867fae
	if (key == 0) {
Packit Bot 867fae
		fprintf(stderr, "Port ID of 0 is invalid\n");
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	if (get_u8(&prio, value, 0))
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
Packit Bot 867fae
	return dcb_parse_mapping("PORT", key, 0xffff,
Packit Bot 867fae
				 "PRIO", prio, IEEE_8021QAZ_MAX_TCS - 1,
Packit Bot 867fae
				 dcb_app_parse_mapping_cb, data);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_parse_default_prio(int *argcp, char ***argvp, struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	int argc = *argcp;
Packit Bot 867fae
	char **argv = *argvp;
Packit Bot 867fae
	int ret = 0;
Packit Bot 867fae
Packit Bot 867fae
	while (argc > 0) {
Packit Bot 867fae
		struct dcb_app app;
Packit Bot 867fae
		__u8 prio;
Packit Bot 867fae
Packit Bot 867fae
		if (get_u8(&prio, *argv, 0)) {
Packit Bot 867fae
			ret = 1;
Packit Bot 867fae
			break;
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		app = (struct dcb_app){
Packit Bot 867fae
			.selector = IEEE_8021QAZ_APP_SEL_ETHERTYPE,
Packit Bot 867fae
			.protocol = 0,
Packit Bot 867fae
			.priority = prio,
Packit Bot 867fae
		};
Packit Bot 867fae
		ret = dcb_app_table_push(tab, &app);
Packit Bot 867fae
		if (ret != 0)
Packit Bot 867fae
			break;
Packit Bot 867fae
Packit Bot 867fae
		argc--, argv++;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	*argcp = argc;
Packit Bot 867fae
	*argvp = argv;
Packit Bot 867fae
	return ret;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_ethtype(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_ETHERTYPE &&
Packit Bot 867fae
	       app->protocol != 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_default(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_ETHERTYPE &&
Packit Bot 867fae
	       app->protocol == 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_dscp(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_DSCP;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_stream_port(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_STREAM;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_dgram_port(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_DGRAM;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static bool dcb_app_is_port(const struct dcb_app *app)
Packit Bot 867fae
{
Packit Bot 867fae
	return app->selector == IEEE_8021QAZ_APP_SEL_ANY;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_print_key_dec(__u16 protocol)
Packit Bot 867fae
{
Packit Bot 867fae
	return print_uint(PRINT_ANY, NULL, "%d:", protocol);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_print_key_hex(__u16 protocol)
Packit Bot 867fae
{
Packit Bot 867fae
	return print_uint(PRINT_ANY, NULL, "%x:", protocol);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_print_key_dscp(__u16 protocol)
Packit Bot 867fae
{
Packit Bot 867fae
	const char *name = rtnl_dsfield_get_name(protocol << 2);
Packit Bot 867fae
Packit Bot 867fae
Packit Bot 867fae
	if (!is_json_context() && name != NULL)
Packit Bot 867fae
		return print_string(PRINT_FP, NULL, "%s:", name);
Packit Bot 867fae
	return print_uint(PRINT_ANY, NULL, "%d:", protocol);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_filtered(const struct dcb_app_table *tab,
Packit Bot 867fae
				   bool (*filter)(const struct dcb_app *),
Packit Bot 867fae
				   int (*print_key)(__u16 protocol),
Packit Bot 867fae
				   const char *json_name,
Packit Bot 867fae
				   const char *fp_name)
Packit Bot 867fae
{
Packit Bot 867fae
	bool first = true;
Packit Bot 867fae
	size_t i;
Packit Bot 867fae
Packit Bot 867fae
	for (i = 0; i < tab->n_apps; i++) {
Packit Bot 867fae
		struct dcb_app *app = &tab->apps[i];
Packit Bot 867fae
Packit Bot 867fae
		if (!filter(app))
Packit Bot 867fae
			continue;
Packit Bot 867fae
		if (first) {
Packit Bot 867fae
			open_json_array(PRINT_JSON, json_name);
Packit Bot 867fae
			print_string(PRINT_FP, NULL, "%s ", fp_name);
Packit Bot 867fae
			first = false;
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		open_json_array(PRINT_JSON, NULL);
Packit Bot 867fae
		print_key(app->protocol);
Packit Bot 867fae
		print_uint(PRINT_ANY, NULL, "%d ", app->priority);
Packit Bot 867fae
		close_json_array(PRINT_JSON, NULL);
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	if (!first) {
Packit Bot 867fae
		close_json_array(PRINT_JSON, json_name);
Packit Bot 867fae
		print_nl();
Packit Bot 867fae
	}
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_ethtype_prio(const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_filtered(tab, dcb_app_is_ethtype,  dcb_app_print_key_hex,
Packit Bot 867fae
			       "ethtype_prio", "ethtype-prio");
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_dscp_prio(const struct dcb *dcb,
Packit Bot 867fae
				    const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_filtered(tab, dcb_app_is_dscp,
Packit Bot 867fae
			       dcb->numeric ? dcb_app_print_key_dec
Packit Bot 867fae
					    : dcb_app_print_key_dscp,
Packit Bot 867fae
			       "dscp_prio", "dscp-prio");
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_stream_port_prio(const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_filtered(tab, dcb_app_is_stream_port, dcb_app_print_key_dec,
Packit Bot 867fae
			       "stream_port_prio", "stream-port-prio");
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_dgram_port_prio(const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_filtered(tab, dcb_app_is_dgram_port, dcb_app_print_key_dec,
Packit Bot 867fae
			       "dgram_port_prio", "dgram-port-prio");
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_port_prio(const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_filtered(tab, dcb_app_is_port, dcb_app_print_key_dec,
Packit Bot 867fae
			       "port_prio", "port-prio");
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print_default_prio(const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	bool first = true;
Packit Bot 867fae
	size_t i;
Packit Bot 867fae
Packit Bot 867fae
	for (i = 0; i < tab->n_apps; i++) {
Packit Bot 867fae
		if (!dcb_app_is_default(&tab->apps[i]))
Packit Bot 867fae
			continue;
Packit Bot 867fae
		if (first) {
Packit Bot 867fae
			open_json_array(PRINT_JSON, "default_prio");
Packit Bot 867fae
			print_string(PRINT_FP, NULL, "default-prio ", NULL);
Packit Bot 867fae
			first = false;
Packit Bot 867fae
		}
Packit Bot 867fae
		print_uint(PRINT_ANY, NULL, "%d ", tab->apps[i].priority);
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	if (!first) {
Packit Bot 867fae
		close_json_array(PRINT_JSON, "default_prio");
Packit Bot 867fae
		print_nl();
Packit Bot 867fae
	}
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static void dcb_app_print(const struct dcb *dcb, const struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	dcb_app_print_ethtype_prio(tab);
Packit Bot 867fae
	dcb_app_print_default_prio(tab);
Packit Bot 867fae
	dcb_app_print_dscp_prio(dcb, tab);
Packit Bot 867fae
	dcb_app_print_stream_port_prio(tab);
Packit Bot 867fae
	dcb_app_print_dgram_port_prio(tab);
Packit Bot 867fae
	dcb_app_print_port_prio(tab);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_get_table_attr_cb(const struct nlattr *attr, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table *tab = data;
Packit Bot 867fae
	struct dcb_app *app;
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	if (mnl_attr_get_type(attr) != DCB_ATTR_IEEE_APP) {
Packit Bot 867fae
		fprintf(stderr, "Unknown attribute in DCB_ATTR_IEEE_APP_TABLE: %d\n",
Packit Bot 867fae
			mnl_attr_get_type(attr));
Packit Bot 867fae
		return MNL_CB_OK;
Packit Bot 867fae
	}
Packit Bot 867fae
	if (mnl_attr_get_payload_len(attr) < sizeof(struct dcb_app)) {
Packit Bot 867fae
		fprintf(stderr, "DCB_ATTR_IEEE_APP payload expected to have size %zd, not %d\n",
Packit Bot 867fae
			sizeof(struct dcb_app), mnl_attr_get_payload_len(attr));
Packit Bot 867fae
		return MNL_CB_OK;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	app = mnl_attr_get_payload(attr);
Packit Bot 867fae
	ret = dcb_app_table_push(tab, app);
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return MNL_CB_ERROR;
Packit Bot 867fae
Packit Bot 867fae
	return MNL_CB_OK;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_get(struct dcb *dcb, const char *dev, struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	uint16_t payload_len;
Packit Bot 867fae
	void *payload;
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_get_attribute_va(dcb, dev, DCB_ATTR_IEEE_APP_TABLE, &payload, &payload_len);
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = mnl_attr_parse_payload(payload, payload_len, dcb_app_get_table_attr_cb, tab);
Packit Bot 867fae
	if (ret != MNL_CB_OK)
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
struct dcb_app_add_del {
Packit Bot 867fae
	const struct dcb_app_table *tab;
Packit Bot 867fae
	bool (*filter)(const struct dcb_app *app);
Packit Bot 867fae
};
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_add_del_cb(struct dcb *dcb, struct nlmsghdr *nlh, void *data)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_add_del *add_del = data;
Packit Bot 867fae
	struct nlattr *nest;
Packit Bot 867fae
	size_t i;
Packit Bot 867fae
Packit Bot 867fae
	nest = mnl_attr_nest_start(nlh, DCB_ATTR_IEEE_APP_TABLE);
Packit Bot 867fae
Packit Bot 867fae
	for (i = 0; i < add_del->tab->n_apps; i++) {
Packit Bot 867fae
		const struct dcb_app *app = &add_del->tab->apps[i];
Packit Bot 867fae
Packit Bot 867fae
		if (add_del->filter == NULL || add_del->filter(app))
Packit Bot 867fae
			mnl_attr_put(nlh, DCB_ATTR_IEEE_APP, sizeof(*app), app);
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	mnl_attr_nest_end(nlh, nest);
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_app_add_del(struct dcb *dcb, const char *dev, int command,
Packit Bot 867fae
			   const struct dcb_app_table *tab,
Packit Bot 867fae
			   bool (*filter)(const struct dcb_app *))
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_add_del add_del = {
Packit Bot 867fae
		.tab = tab,
Packit Bot 867fae
		.filter = filter,
Packit Bot 867fae
	};
Packit Bot 867fae
Packit Bot 867fae
	if (tab->n_apps == 0)
Packit Bot 867fae
		return 0;
Packit Bot 867fae
Packit Bot 867fae
	return dcb_set_attribute_va(dcb, command, dev, dcb_app_add_del_cb, &add_del);
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_parse_add_del(struct dcb *dcb, const char *dev,
Packit Bot 867fae
				     int argc, char **argv, struct dcb_app_table *tab)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_parse_mapping pm = {
Packit Bot 867fae
		.tab = tab,
Packit Bot 867fae
	};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	if (!argc) {
Packit Bot 867fae
		dcb_app_help_add();
Packit Bot 867fae
		return 0;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	do {
Packit Bot 867fae
		if (matches(*argv, "help") == 0) {
Packit Bot 867fae
			dcb_app_help_add();
Packit Bot 867fae
			return 0;
Packit Bot 867fae
		} else if (matches(*argv, "ethtype-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			pm.selector = IEEE_8021QAZ_APP_SEL_ETHERTYPE;
Packit Bot 867fae
			ret = parse_mapping(&argc, &argv, false,
Packit Bot 867fae
					    &dcb_app_parse_mapping_ethtype_prio,
Packit Bot 867fae
					    &pm);
Packit Bot 867fae
		} else if (matches(*argv, "default-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			ret = dcb_app_parse_default_prio(&argc, &argv, pm.tab);
Packit Bot 867fae
			if (ret != 0) {
Packit Bot 867fae
				fprintf(stderr, "Invalid default priority %s\n", *argv);
Packit Bot 867fae
				return ret;
Packit Bot 867fae
			}
Packit Bot 867fae
		} else if (matches(*argv, "dscp-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			pm.selector = IEEE_8021QAZ_APP_SEL_DSCP;
Packit Bot 867fae
			ret = parse_mapping_gen(&argc, &argv,
Packit Bot 867fae
						&dcb_app_parse_dscp,
Packit Bot 867fae
						&dcb_app_parse_mapping_dscp_prio,
Packit Bot 867fae
						&pm);
Packit Bot 867fae
		} else if (matches(*argv, "stream-port-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			pm.selector = IEEE_8021QAZ_APP_SEL_STREAM;
Packit Bot 867fae
			ret = parse_mapping(&argc, &argv, false,
Packit Bot 867fae
					    &dcb_app_parse_mapping_port_prio,
Packit Bot 867fae
					    &pm);
Packit Bot 867fae
		} else if (matches(*argv, "dgram-port-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			pm.selector = IEEE_8021QAZ_APP_SEL_DGRAM;
Packit Bot 867fae
			ret = parse_mapping(&argc, &argv, false,
Packit Bot 867fae
					    &dcb_app_parse_mapping_port_prio,
Packit Bot 867fae
					    &pm);
Packit Bot 867fae
		} else if (matches(*argv, "port-prio") == 0) {
Packit Bot 867fae
			NEXT_ARG();
Packit Bot 867fae
			pm.selector = IEEE_8021QAZ_APP_SEL_ANY;
Packit Bot 867fae
			ret = parse_mapping(&argc, &argv, false,
Packit Bot 867fae
					    &dcb_app_parse_mapping_port_prio,
Packit Bot 867fae
					    &pm);
Packit Bot 867fae
		} else {
Packit Bot 867fae
			fprintf(stderr, "What is \"%s\"?\n", *argv);
Packit Bot 867fae
			dcb_app_help_add();
Packit Bot 867fae
			return -EINVAL;
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		if (ret != 0) {
Packit Bot 867fae
			fprintf(stderr, "Invalid mapping %s\n", *argv);
Packit Bot 867fae
			return ret;
Packit Bot 867fae
		}
Packit Bot 867fae
		if (pm.err)
Packit Bot 867fae
			return pm.err;
Packit Bot 867fae
	} while (argc > 0);
Packit Bot 867fae
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_add(struct dcb *dcb, const char *dev, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table tab = {};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_cmd_app_parse_add_del(dcb, dev, argc, argv, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_SET, &tab, NULL);
Packit Bot 867fae
	dcb_app_table_fini(&tab;;
Packit Bot 867fae
	return ret;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_del(struct dcb *dcb, const char *dev, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table tab = {};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_cmd_app_parse_add_del(dcb, dev, argc, argv, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &tab, NULL);
Packit Bot 867fae
	dcb_app_table_fini(&tab;;
Packit Bot 867fae
	return ret;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_show(struct dcb *dcb, const char *dev, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table tab = {};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_get(dcb, dev, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	dcb_app_table_sort(&tab;;
Packit Bot 867fae
Packit Bot 867fae
	open_json_object(NULL);
Packit Bot 867fae
Packit Bot 867fae
	if (!argc) {
Packit Bot 867fae
		dcb_app_print(dcb, &tab;;
Packit Bot 867fae
		goto out;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	do {
Packit Bot 867fae
		if (matches(*argv, "help") == 0) {
Packit Bot 867fae
			dcb_app_help_show_flush();
Packit Bot 867fae
			goto out;
Packit Bot 867fae
		} else if (matches(*argv, "ethtype-prio") == 0) {
Packit Bot 867fae
			dcb_app_print_ethtype_prio(&tab;;
Packit Bot 867fae
		} else if (matches(*argv, "dscp-prio") == 0) {
Packit Bot 867fae
			dcb_app_print_dscp_prio(dcb, &tab;;
Packit Bot 867fae
		} else if (matches(*argv, "stream-port-prio") == 0) {
Packit Bot 867fae
			dcb_app_print_stream_port_prio(&tab;;
Packit Bot 867fae
		} else if (matches(*argv, "dgram-port-prio") == 0) {
Packit Bot 867fae
			dcb_app_print_dgram_port_prio(&tab;;
Packit Bot 867fae
		} else if (matches(*argv, "port-prio") == 0) {
Packit Bot 867fae
			dcb_app_print_port_prio(&tab;;
Packit Bot 867fae
		} else {
Packit Bot 867fae
			fprintf(stderr, "What is \"%s\"?\n", *argv);
Packit Bot 867fae
			dcb_app_help_show_flush();
Packit Bot 867fae
			ret = -EINVAL;
Packit Bot 867fae
			goto out;
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
	} while (argc > 0);
Packit Bot 867fae
Packit Bot 867fae
out:
Packit Bot 867fae
	close_json_object();
Packit Bot 867fae
	dcb_app_table_fini(&tab;;
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_flush(struct dcb *dcb, const char *dev, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table tab = {};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_get(dcb, dev, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	if (!argc) {
Packit Bot 867fae
		ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &tab, NULL);
Packit Bot 867fae
		goto out;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	do {
Packit Bot 867fae
		if (matches(*argv, "help") == 0) {
Packit Bot 867fae
			dcb_app_help_show_flush();
Packit Bot 867fae
			goto out;
Packit Bot 867fae
		} else if (matches(*argv, "ethtype-prio") == 0) {
Packit Bot 867fae
			ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &tab,
Packit Bot 867fae
					      &dcb_app_is_ethtype);
Packit Bot 867fae
			if (ret != 0)
Packit Bot 867fae
				goto out;
Packit Bot 867fae
		} else if (matches(*argv, "default-prio") == 0) {
Packit Bot 867fae
			ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &tab,
Packit Bot 867fae
					      &dcb_app_is_default);
Packit Bot 867fae
			if (ret != 0)
Packit Bot 867fae
				goto out;
Packit Bot 867fae
		} else if (matches(*argv, "dscp-prio") == 0) {
Packit Bot 867fae
			ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &tab,
Packit Bot 867fae
					      &dcb_app_is_dscp);
Packit Bot 867fae
			if (ret != 0)
Packit Bot 867fae
				goto out;
Packit Bot 867fae
		} else {
Packit Bot 867fae
			fprintf(stderr, "What is \"%s\"?\n", *argv);
Packit Bot 867fae
			dcb_app_help_show_flush();
Packit Bot 867fae
			ret = -EINVAL;
Packit Bot 867fae
			goto out;
Packit Bot 867fae
		}
Packit Bot 867fae
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
	} while (argc > 0);
Packit Bot 867fae
Packit Bot 867fae
out:
Packit Bot 867fae
	dcb_app_table_fini(&tab;;
Packit Bot 867fae
	return ret;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
static int dcb_cmd_app_replace(struct dcb *dcb, const char *dev, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	struct dcb_app_table orig = {};
Packit Bot 867fae
	struct dcb_app_table tab = {};
Packit Bot 867fae
	struct dcb_app_table new = {};
Packit Bot 867fae
	int ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_get(dcb, dev, &orig);
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		return ret;
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_cmd_app_parse_add_del(dcb, dev, argc, argv, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		goto out;
Packit Bot 867fae
Packit Bot 867fae
	/* Attempts to add an existing entry would be rejected, so drop
Packit Bot 867fae
	 * these entries from tab.
Packit Bot 867fae
	 */
Packit Bot 867fae
	ret = dcb_app_table_copy(&new, &tab;;
Packit Bot 867fae
	if (ret != 0)
Packit Bot 867fae
		goto out;
Packit Bot 867fae
	dcb_app_table_remove_existing(&new, &orig);
Packit Bot 867fae
Packit Bot 867fae
	ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_SET, &new, NULL);
Packit Bot 867fae
	if (ret != 0) {
Packit Bot 867fae
		fprintf(stderr, "Could not add new APP entries\n");
Packit Bot 867fae
		goto out;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
	/* Remove the obsolete entries. */
Packit Bot 867fae
	dcb_app_table_remove_replaced(&orig, &tab;;
Packit Bot 867fae
	ret = dcb_app_add_del(dcb, dev, DCB_CMD_IEEE_DEL, &orig, NULL);
Packit Bot 867fae
	if (ret != 0) {
Packit Bot 867fae
		fprintf(stderr, "Could not remove replaced APP entries\n");
Packit Bot 867fae
		goto out;
Packit Bot 867fae
	}
Packit Bot 867fae
Packit Bot 867fae
out:
Packit Bot 867fae
	dcb_app_table_fini(&new;;
Packit Bot 867fae
	dcb_app_table_fini(&tab;;
Packit Bot 867fae
	dcb_app_table_fini(&orig);
Packit Bot 867fae
	return 0;
Packit Bot 867fae
}
Packit Bot 867fae
Packit Bot 867fae
int dcb_cmd_app(struct dcb *dcb, int argc, char **argv)
Packit Bot 867fae
{
Packit Bot 867fae
	if (!argc || matches(*argv, "help") == 0) {
Packit Bot 867fae
		dcb_app_help();
Packit Bot 867fae
		return 0;
Packit Bot 867fae
	} else if (matches(*argv, "show") == 0) {
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
		return dcb_cmd_parse_dev(dcb, argc, argv,
Packit Bot 867fae
					 dcb_cmd_app_show, dcb_app_help_show_flush);
Packit Bot 867fae
	} else if (matches(*argv, "flush") == 0) {
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
		return dcb_cmd_parse_dev(dcb, argc, argv,
Packit Bot 867fae
					 dcb_cmd_app_flush, dcb_app_help_show_flush);
Packit Bot 867fae
	} else if (matches(*argv, "add") == 0) {
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
		return dcb_cmd_parse_dev(dcb, argc, argv,
Packit Bot 867fae
					 dcb_cmd_app_add, dcb_app_help_add);
Packit Bot 867fae
	} else if (matches(*argv, "del") == 0) {
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
		return dcb_cmd_parse_dev(dcb, argc, argv,
Packit Bot 867fae
					 dcb_cmd_app_del, dcb_app_help_add);
Packit Bot 867fae
	} else if (matches(*argv, "replace") == 0) {
Packit Bot 867fae
		NEXT_ARG_FWD();
Packit Bot 867fae
		return dcb_cmd_parse_dev(dcb, argc, argv,
Packit Bot 867fae
					 dcb_cmd_app_replace, dcb_app_help_add);
Packit Bot 867fae
	} else {
Packit Bot 867fae
		fprintf(stderr, "What is \"%s\"?\n", *argv);
Packit Bot 867fae
		dcb_app_help();
Packit Bot 867fae
		return -EINVAL;
Packit Bot 867fae
	}
Packit Bot 867fae
}