Blame ip/iptoken.c

Packit Service 3880ab
/*
Packit Service 3880ab
 * iptoken.c    "ip token"
Packit Service 3880ab
 *
Packit Service 3880ab
 *              This program is free software; you can redistribute it and/or
Packit Service 3880ab
 *              modify it under the terms of the GNU General Public License
Packit Service 3880ab
 *              as published by the Free Software Foundation; either version
Packit Service 3880ab
 *              2 of the License, or (at your option) any later version.
Packit Service 3880ab
 *
Packit Service 3880ab
 * Authors:     Daniel Borkmann, <borkmann@redhat.com>
Packit Service 3880ab
 */
Packit Service 3880ab
Packit Service 3880ab
#include <stdio.h>
Packit Service 3880ab
#include <stdlib.h>
Packit Service 3880ab
#include <stdbool.h>
Packit Service 3880ab
#include <unistd.h>
Packit Service 3880ab
#include <fcntl.h>
Packit Service 3880ab
#include <string.h>
Packit Service 3880ab
#include <sys/socket.h>
Packit Service 3880ab
#include <netinet/in.h>
Packit Service 3880ab
#include <netinet/ip.h>
Packit Service 3880ab
#include <arpa/inet.h>
Packit Service 3880ab
#include <linux/types.h>
Packit Service 3880ab
#include <linux/if.h>
Packit Service 3880ab
Packit Service 3880ab
#include "rt_names.h"
Packit Service 3880ab
#include "utils.h"
Packit Service 3880ab
#include "ip_common.h"
Packit Service 3880ab
#include "json_print.h"
Packit Service 3880ab
Packit Service 3880ab
extern struct rtnl_handle rth;
Packit Service 3880ab
Packit Service 3880ab
struct rtnl_dump_args {
Packit Service 3880ab
	FILE *fp;
Packit Service 3880ab
	int ifindex;
Packit Service 3880ab
};
Packit Service 3880ab
Packit Service 3880ab
static void usage(void) __attribute__((noreturn));
Packit Service 3880ab
Packit Service 3880ab
static void usage(void)
Packit Service 3880ab
{
Packit Service 3880ab
	fprintf(stderr, "Usage: ip token [ list | set | del | get ] [ TOKEN ] [ dev DEV ]\n");
Packit Service 3880ab
	exit(-1);
Packit Service 3880ab
}
Packit Service 3880ab
Packit Service 3880ab
static int print_token(struct nlmsghdr *n, void *arg)
Packit Service 3880ab
{
Packit Service 3880ab
	struct rtnl_dump_args *args = arg;
Packit Service 3880ab
	FILE *fp = args->fp;
Packit Service 3880ab
	int ifindex = args->ifindex;
Packit Service 3880ab
	struct ifinfomsg *ifi = NLMSG_DATA(n);
Packit Service 3880ab
	int len = n->nlmsg_len;
Packit Service 3880ab
	struct rtattr *tb[IFLA_MAX + 1];
Packit Service 3880ab
	struct rtattr *ltb[IFLA_INET6_MAX + 1];
Packit Service 3880ab
Packit Service 3880ab
	if (n->nlmsg_type != RTM_NEWLINK)
Packit Service 3880ab
		return -1;
Packit Service 3880ab
Packit Service 3880ab
	len -= NLMSG_LENGTH(sizeof(*ifi));
Packit Service 3880ab
	if (len < 0)
Packit Service 3880ab
		return -1;
Packit Service 3880ab
Packit Service 3880ab
	if (ifi->ifi_family != AF_INET6)
Packit Service 3880ab
		return 0;
Packit Service 3880ab
	if (ifi->ifi_index == 0)
Packit Service 3880ab
		return 0;
Packit Service 3880ab
	if (ifindex > 0 && ifi->ifi_index != ifindex)
Packit Service 3880ab
		return 0;
Packit Service 3880ab
	if (ifi->ifi_flags & (IFF_LOOPBACK | IFF_NOARP))
Packit Service 3880ab
		return 0;
Packit Service 3880ab
Packit Service 3880ab
	parse_rtattr(tb, IFLA_MAX, IFLA_RTA(ifi), len);
Packit Service 3880ab
	if (!tb[IFLA_PROTINFO])
Packit Service 3880ab
		return -1;
Packit Service 3880ab
Packit Service 3880ab
	parse_rtattr_nested(ltb, IFLA_INET6_MAX, tb[IFLA_PROTINFO]);
Packit Service 3880ab
	if (!ltb[IFLA_INET6_TOKEN]) {
Packit Service 3880ab
		fprintf(stderr, "Seems there's no support for IPv6 token!\n");
Packit Service 3880ab
		return -1;
Packit Service 3880ab
	}
Packit Service 3880ab
Packit Service 3880ab
	open_json_object(NULL);
Packit Service 3880ab
	print_string(PRINT_FP, NULL, "token ", NULL);
Packit Service 3880ab
	print_color_string(PRINT_ANY,
Packit Service 3880ab
			   ifa_family_color(ifi->ifi_family),
Packit Service 3880ab
			   "token", "%s",
Packit Service 3880ab
			   format_host_rta(ifi->ifi_family, ltb[IFLA_INET6_TOKEN]));
Packit Service 3880ab
	print_string(PRINT_FP, NULL, " dev ", NULL);
Packit Service 3880ab
	print_color_string(PRINT_ANY, COLOR_IFNAME,
Packit Service 3880ab
			   "ifname", "%s\n",
Packit Service 3880ab
			   ll_index_to_name(ifi->ifi_index));
Packit Service 3880ab
	close_json_object();
Packit Service 3880ab
	fflush(fp);
Packit Service 3880ab
Packit Service 3880ab
	return 0;
Packit Service 3880ab
}
Packit Service 3880ab
Packit Service 3880ab
static int iptoken_list(int argc, char **argv)
Packit Service 3880ab
{
Packit Service 3880ab
	int af = AF_INET6;
Packit Service 3880ab
	struct rtnl_dump_args da = { .fp = stdout };
Packit Service 3880ab
Packit Service 3880ab
	while (argc > 0) {
Packit Service 3880ab
		if (strcmp(*argv, "dev") == 0) {
Packit Service 3880ab
			NEXT_ARG();
Packit Service 3880ab
			if ((da.ifindex = ll_name_to_index(*argv)) == 0)
Packit Service 3880ab
				invarg("dev is invalid\n", *argv);
Packit Service 3880ab
			break;
Packit Service 3880ab
		}
Packit Service 3880ab
		argc--; argv++;
Packit Service 3880ab
	}
Packit Service 3880ab
Packit Service 3880ab
	if (rtnl_linkdump_req(&rth, af) < 0) {
Packit Service 3880ab
		perror("Cannot send dump request");
Packit Service 3880ab
		return -1;
Packit Service 3880ab
	}
Packit Service 3880ab
Packit Service 3880ab
	new_json_obj(json);
Packit Service 3880ab
	if (rtnl_dump_filter(&rth, print_token, &da) < 0) {
Packit Service 3880ab
		delete_json_obj();
Packit Service 3880ab
		fprintf(stderr, "Dump terminated\n");
Packit Service 3880ab
		return -1;
Packit Service 3880ab
	}
Packit Service 3880ab
	delete_json_obj();
Packit Service 3880ab
Packit Service 3880ab
	return 0;
Packit Service 3880ab
}
Packit Service 3880ab
Packit Service 3880ab
static int iptoken_set(int argc, char **argv, bool delete)
Packit Service 3880ab
{
Packit Service 3880ab
	struct {
Packit Service 3880ab
		struct nlmsghdr n;
Packit Service 3880ab
		struct ifinfomsg ifi;
Packit Service 3880ab
		char buf[512];
Packit Service 3880ab
	} req = {
Packit Service 3880ab
		.n.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifinfomsg)),
Packit Service 3880ab
		.n.nlmsg_flags = NLM_F_REQUEST,
Packit Service 3880ab
		.n.nlmsg_type = RTM_SETLINK,
Packit Service 3880ab
		.ifi.ifi_family = AF_INET6,
Packit Service 3880ab
	};
Packit Service 3880ab
	struct rtattr *afs, *afs6;
Packit Service 3880ab
	bool have_token = delete, have_dev = false;
Packit Service 3880ab
	inet_prefix addr = { .bytelen = 16, };
Packit Service 3880ab
Packit Service 3880ab
	while (argc > 0) {
Packit Service 3880ab
		if (strcmp(*argv, "dev") == 0) {
Packit Service 3880ab
			NEXT_ARG();
Packit Service 3880ab
			if (!have_dev) {
Packit Service 3880ab
				if ((req.ifi.ifi_index =
Packit Service 3880ab
				     ll_name_to_index(*argv)) == 0)
Packit Service 3880ab
					invarg("dev is invalid\n", *argv);
Packit Service 3880ab
				have_dev = true;
Packit Service 3880ab
			}
Packit Service 3880ab
		} else {
Packit Service 3880ab
			if (matches(*argv, "help") == 0)
Packit Service 3880ab
				usage();
Packit Service 3880ab
			if (!have_token) {
Packit Service 3880ab
				get_prefix(&addr, *argv, req.ifi.ifi_family);
Packit Service 3880ab
				have_token = true;
Packit Service 3880ab
			}
Packit Service 3880ab
		}
Packit Service 3880ab
		argc--; argv++;
Packit Service 3880ab
	}
Packit Service 3880ab
Packit Service 3880ab
	if (!have_token) {
Packit Service 3880ab
		fprintf(stderr, "Not enough information: token is required.\n");
Packit Service 3880ab
		return -1;
Packit Service 3880ab
	}
Packit Service 3880ab
	if (!have_dev) {
Packit Service 3880ab
		fprintf(stderr, "Not enough information: \"dev\" argument is required.\n");
Packit Service 3880ab
		return -1;
Packit Service 3880ab
	}
Packit Service 3880ab
Packit Service 3880ab
	afs = addattr_nest(&req.n, sizeof(req), IFLA_AF_SPEC);
Packit Service 3880ab
	afs6 = addattr_nest(&req.n, sizeof(req), AF_INET6);
Packit Service 3880ab
	addattr_l(&req.n, sizeof(req), IFLA_INET6_TOKEN,
Packit Service 3880ab
		  &addr.data, addr.bytelen);
Packit Service 3880ab
	addattr_nest_end(&req.n, afs6);
Packit Service 3880ab
	addattr_nest_end(&req.n, afs);
Packit Service 3880ab
Packit Service 3880ab
	if (rtnl_talk(&rth, &req.n, NULL) < 0)
Packit Service 3880ab
		return -2;
Packit Service 3880ab
Packit Service 3880ab
	return 0;
Packit Service 3880ab
}
Packit Service 3880ab
Packit Service 3880ab
int do_iptoken(int argc, char **argv)
Packit Service 3880ab
{
Packit Service 3880ab
	ll_init_map(&rth);
Packit Service 3880ab
Packit Service 3880ab
	if (argc < 1) {
Packit Service 3880ab
		return iptoken_list(0, NULL);
Packit Service 3880ab
	} else if (matches(argv[0], "list") == 0 ||
Packit Service 3880ab
		   matches(argv[0], "lst") == 0 ||
Packit Service 3880ab
		   matches(argv[0], "show") == 0) {
Packit Service 3880ab
		return iptoken_list(argc - 1, argv + 1);
Packit Service 3880ab
	} else if (matches(argv[0], "set") == 0 ||
Packit Service 3880ab
		   matches(argv[0], "add") == 0) {
Packit Service 3880ab
		return iptoken_set(argc - 1, argv + 1, false);
Packit Service 3880ab
	} else if (matches(argv[0], "delete") == 0) {
Packit Service 3880ab
		return iptoken_set(argc - 1, argv + 1, true);
Packit Service 3880ab
	} else if (matches(argv[0], "get") == 0) {
Packit Service 3880ab
		return iptoken_list(argc - 1, argv + 1);
Packit Service 3880ab
	} else if (matches(argv[0], "help") == 0)
Packit Service 3880ab
		usage();
Packit Service 3880ab
Packit Service 3880ab
	fprintf(stderr, "Command \"%s\" is unknown, try \"ip token help\".\n", *argv);
Packit Service 3880ab
	exit(-1);
Packit Service 3880ab
}