Blame ip/ipseg6.c

Packit d3f73b
/*
Packit d3f73b
 * seg6.c "ip sr/seg6"
Packit d3f73b
 *
Packit d3f73b
 *	  This program is free software; you can redistribute it and/or
Packit d3f73b
 *	  modify it under the terms of the GNU General Public License
Packit d3f73b
 *	  version 2 as published by the Free Software Foundation;
Packit d3f73b
 *
Packit d3f73b
 * Author: David Lebrun <david.lebrun@uclouvain.be>
Packit d3f73b
 */
Packit d3f73b
Packit d3f73b
#include <stdio.h>
Packit d3f73b
#include <stdlib.h>
Packit d3f73b
#include <string.h>
Packit d3f73b
#include <unistd.h>
Packit d3f73b
#include <errno.h>
Packit d3f73b
#include <sys/types.h>
Packit d3f73b
#include <sys/socket.h>
Packit d3f73b
#include <arpa/inet.h>
Packit d3f73b
#include <sys/ioctl.h>
Packit d3f73b
#include <linux/if.h>
Packit d3f73b
Packit d3f73b
#include <linux/genetlink.h>
Packit d3f73b
#include <linux/seg6_genl.h>
Packit d3f73b
#include <linux/seg6_hmac.h>
Packit d3f73b
Packit d3f73b
#include "utils.h"
Packit d3f73b
#include "ip_common.h"
Packit d3f73b
#include "libgenl.h"
Packit d3f73b
#include "json_print.h"
Packit d3f73b
Packit d3f73b
#define HMAC_KEY_PROMPT "Enter secret for HMAC key ID (blank to delete): "
Packit d3f73b
Packit d3f73b
static void usage(void)
Packit d3f73b
{
Packit d3f73b
	fprintf(stderr,
Packit d3f73b
		"Usage: ip sr { COMMAND | help }\n"
Packit d3f73b
		"	   ip sr hmac show\n"
Packit d3f73b
		"	   ip sr hmac set KEYID ALGO\n"
Packit d3f73b
		"	   ip sr tunsrc show\n"
Packit d3f73b
		"	   ip sr tunsrc set ADDRESS\n"
Packit d3f73b
		"where  ALGO := { sha1 | sha256 }\n");
Packit d3f73b
	exit(-1);
Packit d3f73b
}
Packit d3f73b
Packit d3f73b
static struct rtnl_handle grth = { .fd = -1 };
Packit d3f73b
static int genl_family = -1;
Packit d3f73b
Packit d3f73b
#define SEG6_REQUEST(_req, _bufsiz, _cmd, _flags) \
Packit d3f73b
	GENL_REQUEST(_req, _bufsiz, genl_family, 0, \
Packit d3f73b
				SEG6_GENL_VERSION, _cmd, _flags)
Packit d3f73b
Packit d3f73b
static struct {
Packit d3f73b
	unsigned int cmd;
Packit d3f73b
	inet_prefix addr;
Packit d3f73b
	__u32 keyid;
Packit d3f73b
	const char *pass;
Packit d3f73b
	__u8 alg_id;
Packit d3f73b
} opts;
Packit d3f73b
Packit d3f73b
static void print_dumphmac(struct rtattr *attrs[])
Packit d3f73b
{
Packit d3f73b
	char secret[64];
Packit d3f73b
	char *algstr;
Packit d3f73b
	__u8 slen = rta_getattr_u8(attrs[SEG6_ATTR_SECRETLEN]);
Packit d3f73b
	__u8 alg_id = rta_getattr_u8(attrs[SEG6_ATTR_ALGID]);
Packit d3f73b
Packit d3f73b
	memset(secret, 0, 64);
Packit d3f73b
Packit d3f73b
	if (slen > 63) {
Packit d3f73b
		fprintf(stderr, "HMAC secret length %d > 63, truncated\n", slen);
Packit d3f73b
		slen = 63;
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	memcpy(secret, RTA_DATA(attrs[SEG6_ATTR_SECRET]), slen);
Packit d3f73b
Packit d3f73b
	switch (alg_id) {
Packit d3f73b
	case SEG6_HMAC_ALGO_SHA1:
Packit d3f73b
		algstr = "sha1";
Packit d3f73b
		break;
Packit d3f73b
	case SEG6_HMAC_ALGO_SHA256:
Packit d3f73b
		algstr = "sha256";
Packit d3f73b
		break;
Packit d3f73b
	default:
Packit d3f73b
		algstr = "<unknown>";
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	print_uint(PRINT_ANY, "hmac", "hmac %u ",
Packit d3f73b
		   rta_getattr_u32(attrs[SEG6_ATTR_HMACKEYID]));
Packit d3f73b
	print_string(PRINT_ANY, "algo", "algo %s ", algstr);
Packit d3f73b
	print_string(PRINT_ANY, "secret", "secret \"%s\"\n", secret);
Packit d3f73b
}
Packit d3f73b
Packit d3f73b
static void print_tunsrc(struct rtattr *attrs[])
Packit d3f73b
{
Packit d3f73b
	const char *dst
Packit d3f73b
		= rt_addr_n2a(AF_INET6, 16,
Packit d3f73b
			      RTA_DATA(attrs[SEG6_ATTR_DST]));
Packit d3f73b
Packit d3f73b
	print_string(PRINT_ANY, "tunsrc",
Packit d3f73b
		     "tunsrc addr %s\n", dst);
Packit d3f73b
}
Packit d3f73b
Packit d3f73b
static int process_msg(struct nlmsghdr *n, void *arg)
Packit d3f73b
{
Packit d3f73b
	struct rtattr *attrs[SEG6_ATTR_MAX + 1];
Packit d3f73b
	struct genlmsghdr *ghdr;
Packit d3f73b
	int len = n->nlmsg_len;
Packit d3f73b
Packit d3f73b
	if (n->nlmsg_type != genl_family)
Packit d3f73b
		return -1;
Packit d3f73b
Packit d3f73b
	len -= NLMSG_LENGTH(GENL_HDRLEN);
Packit d3f73b
	if (len < 0)
Packit d3f73b
		return -1;
Packit d3f73b
Packit d3f73b
	ghdr = NLMSG_DATA(n);
Packit d3f73b
Packit d3f73b
	parse_rtattr(attrs, SEG6_ATTR_MAX, (void *)ghdr + GENL_HDRLEN, len);
Packit d3f73b
Packit d3f73b
	open_json_object(NULL);
Packit d3f73b
	switch (ghdr->cmd) {
Packit d3f73b
	case SEG6_CMD_DUMPHMAC:
Packit d3f73b
		print_dumphmac(attrs);
Packit d3f73b
		break;
Packit d3f73b
Packit d3f73b
	case SEG6_CMD_GET_TUNSRC:
Packit d3f73b
		print_tunsrc(attrs);
Packit d3f73b
		break;
Packit d3f73b
	}
Packit d3f73b
	close_json_object();
Packit d3f73b
Packit d3f73b
	return 0;
Packit d3f73b
}
Packit d3f73b
Packit d3f73b
static int seg6_do_cmd(void)
Packit d3f73b
{
Packit d3f73b
	SEG6_REQUEST(req, 1024, opts.cmd, NLM_F_REQUEST);
Packit d3f73b
	struct nlmsghdr *answer;
Packit d3f73b
	int repl = 0, dump = 0;
Packit d3f73b
Packit d3f73b
	if (genl_family < 0) {
Packit d3f73b
		if (rtnl_open_byproto(&grth, 0, NETLINK_GENERIC) < 0) {
Packit d3f73b
			fprintf(stderr, "Cannot open generic netlink socket\n");
Packit d3f73b
			exit(1);
Packit d3f73b
		}
Packit d3f73b
		genl_family = genl_resolve_family(&grth, SEG6_GENL_NAME);
Packit d3f73b
		if (genl_family < 0)
Packit d3f73b
			exit(1);
Packit d3f73b
		req.n.nlmsg_type = genl_family;
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	switch (opts.cmd) {
Packit d3f73b
	case SEG6_CMD_SETHMAC:
Packit d3f73b
	{
Packit d3f73b
		addattr32(&req.n, sizeof(req), SEG6_ATTR_HMACKEYID, opts.keyid);
Packit d3f73b
		addattr8(&req.n, sizeof(req), SEG6_ATTR_SECRETLEN,
Packit d3f73b
			 strlen(opts.pass));
Packit d3f73b
		addattr8(&req.n, sizeof(req), SEG6_ATTR_ALGID, opts.alg_id);
Packit d3f73b
		if (strlen(opts.pass))
Packit d3f73b
			addattr_l(&req.n, sizeof(req), SEG6_ATTR_SECRET,
Packit d3f73b
				  opts.pass, strlen(opts.pass));
Packit d3f73b
		break;
Packit d3f73b
	}
Packit d3f73b
	case SEG6_CMD_SET_TUNSRC:
Packit d3f73b
		addattr_l(&req.n, sizeof(req), SEG6_ATTR_DST, opts.addr.data,
Packit d3f73b
			  sizeof(struct in6_addr));
Packit d3f73b
		break;
Packit d3f73b
	case SEG6_CMD_DUMPHMAC:
Packit d3f73b
		dump = 1;
Packit d3f73b
		break;
Packit d3f73b
	case SEG6_CMD_GET_TUNSRC:
Packit d3f73b
		repl = 1;
Packit d3f73b
		break;
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	if (!repl && !dump) {
Packit d3f73b
		if (rtnl_talk(&grth, &req.n, NULL) < 0)
Packit d3f73b
			return -1;
Packit d3f73b
	} else if (repl) {
Packit d3f73b
		if (rtnl_talk(&grth, &req.n, &answer) < 0)
Packit d3f73b
			return -2;
Packit d3f73b
		new_json_obj(json);
Packit d3f73b
		if (process_msg(answer, stdout) < 0) {
Packit d3f73b
			fprintf(stderr, "Error parsing reply\n");
Packit d3f73b
			exit(1);
Packit d3f73b
		}
Packit d3f73b
		delete_json_obj();
Packit d3f73b
		free(answer);
Packit d3f73b
	} else {
Packit d3f73b
		req.n.nlmsg_flags |= NLM_F_DUMP;
Packit d3f73b
		req.n.nlmsg_seq = grth.dump = ++grth.seq;
Packit d3f73b
		if (rtnl_send(&grth, &req, req.n.nlmsg_len) < 0) {
Packit d3f73b
			perror("Failed to send dump request");
Packit d3f73b
			exit(1);
Packit d3f73b
		}
Packit d3f73b
Packit d3f73b
		new_json_obj(json);
Packit d3f73b
		if (rtnl_dump_filter(&grth, process_msg, stdout) < 0) {
Packit d3f73b
			fprintf(stderr, "Dump terminated\n");
Packit d3f73b
			exit(1);
Packit d3f73b
		}
Packit d3f73b
		delete_json_obj();
Packit d3f73b
		fflush(stdout);
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	return 0;
Packit d3f73b
}
Packit d3f73b
Packit d3f73b
int do_seg6(int argc, char **argv)
Packit d3f73b
{
Packit d3f73b
	if (argc < 1 || matches(*argv, "help") == 0)
Packit d3f73b
		usage();
Packit d3f73b
Packit d3f73b
	memset(&opts, 0, sizeof(opts));
Packit d3f73b
Packit d3f73b
	if (matches(*argv, "hmac") == 0) {
Packit d3f73b
		NEXT_ARG();
Packit d3f73b
		if (matches(*argv, "show") == 0) {
Packit d3f73b
			opts.cmd = SEG6_CMD_DUMPHMAC;
Packit d3f73b
		} else if (matches(*argv, "set") == 0) {
Packit d3f73b
			NEXT_ARG();
Packit d3f73b
			if (get_u32(&opts.keyid, *argv, 0) || opts.keyid == 0)
Packit d3f73b
				invarg("hmac KEYID value is invalid", *argv);
Packit d3f73b
			NEXT_ARG();
Packit d3f73b
			if (strcmp(*argv, "sha1") == 0) {
Packit d3f73b
				opts.alg_id = SEG6_HMAC_ALGO_SHA1;
Packit d3f73b
			} else if (strcmp(*argv, "sha256") == 0) {
Packit d3f73b
				opts.alg_id = SEG6_HMAC_ALGO_SHA256;
Packit d3f73b
			} else {
Packit d3f73b
				invarg("hmac ALGO value is invalid", *argv);
Packit d3f73b
			}
Packit d3f73b
			opts.cmd = SEG6_CMD_SETHMAC;
Packit d3f73b
			opts.pass = getpass(HMAC_KEY_PROMPT);
Packit d3f73b
		} else {
Packit d3f73b
			invarg("unknown", *argv);
Packit d3f73b
		}
Packit d3f73b
	} else if (matches(*argv, "tunsrc") == 0) {
Packit d3f73b
		NEXT_ARG();
Packit d3f73b
		if (matches(*argv, "show") == 0) {
Packit d3f73b
			opts.cmd = SEG6_CMD_GET_TUNSRC;
Packit d3f73b
		} else if (matches(*argv, "set") == 0) {
Packit d3f73b
			NEXT_ARG();
Packit d3f73b
			opts.cmd = SEG6_CMD_SET_TUNSRC;
Packit d3f73b
			get_addr(&opts.addr, *argv, AF_INET6);
Packit d3f73b
		} else {
Packit d3f73b
			invarg("unknown", *argv);
Packit d3f73b
		}
Packit d3f73b
	} else {
Packit d3f73b
		invarg("unknown", *argv);
Packit d3f73b
	}
Packit d3f73b
Packit d3f73b
	return seg6_do_cmd();
Packit d3f73b
}