Files
NetworkManager/src/platform/tests/platform.c
2013-05-18 04:38:17 +02:00

693 lines
19 KiB
C

/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
/* nm-platform.c - Handle runtime kernel networking configuration
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Copyright (C) 2013 Red Hat, Inc.
*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <arpa/inet.h>
#include "nm-platform.h"
#include "nm-linux-platform.h"
#include "nm-fake-platform.h"
#define error(...) fprintf (stderr, __VA_ARGS__)
typedef gboolean boolean_t;
typedef int decimal_t;
typedef const char *string_t;
#define print_boolean(value) printf ("%s\n", value ? "yes" : "no")
#define print_decimal(value) printf ("%d\n", value)
#define print_string(value) printf ("%s\n", value)
static gboolean
do_sysctl_set (char **argv)
{
return nm_platform_sysctl_set (argv[0], argv[1]);
}
static gboolean
do_sysctl_get (char **argv)
{
auto_g_free char *value = nm_platform_sysctl_get (argv[0]);
printf ("%s\n", value);
return !!value;
}
static int
parse_ifindex (const char *str)
{
char *endptr;
int ifindex = 0;
ifindex = strtol (str, &endptr, 10);
if (*endptr) {
ifindex = nm_platform_link_get_ifindex (str);
}
return ifindex;
}
static gboolean
do_link_get_all (char **argv)
{
GArray *links;
NMPlatformLink *device;
int i;
links = nm_platform_link_get_all ();
for (i = 0; i < links->len; i++) {
device = &g_array_index (links, NMPlatformLink, i);
printf ("%d: %s type %d\n", device->ifindex, device->name, device->type);
}
g_array_unref (links);
return TRUE;
}
static gboolean
do_dummy_add (char **argv)
{
return nm_platform_dummy_add (argv[0]);
}
static gboolean
do_bridge_add (char **argv)
{
return nm_platform_bridge_add (argv[0]);
}
static gboolean
do_bond_add (char **argv)
{
return nm_platform_bond_add (argv[0]);
}
static gboolean
do_team_add (char **argv)
{
return nm_platform_team_add (argv[0]);
}
static gboolean
do_vlan_add (char **argv)
{
const char *name = *argv++;
int parent = parse_ifindex (*argv++);
int vlanid = strtol (*argv++, NULL, 10);
guint32 vlan_flags = strtol (*argv++, NULL, 10);
return nm_platform_vlan_add (name, parent, vlanid, vlan_flags);
}
static gboolean
do_link_exists (char **argv)
{
gboolean value = nm_platform_link_exists (argv[0]);
print_boolean (value);
return TRUE;
}
#define LINK_CMD(cmdname) \
static gboolean \
do_link_##cmdname (char **argv) \
{ \
int ifindex = parse_ifindex (argv[0]); \
return ifindex ? nm_platform_link_##cmdname (ifindex) : FALSE; \
}
#define LINK_CMD_GET_FULL(cmdname, type, cond) \
static gboolean \
do_link_##cmdname (char **argv) \
{ \
int ifindex = parse_ifindex (argv[0]); \
if (ifindex) { \
type##_t value = nm_platform_link_##cmdname (ifindex); \
if (cond) { \
print_##type (value); \
return TRUE; \
} \
} \
return FALSE; \
}
#define LINK_CMD_GET(cmdname, type) LINK_CMD_GET_FULL (cmdname, type, TRUE);
LINK_CMD (delete)
/* do_link_delete_by_ifname:
*
* We don't need this as we allow ifname instead of ifindex anyway.
*/
static gboolean
do_link_get_ifindex (char **argv)
{
int ifindex = nm_platform_link_get_ifindex (argv[0]);
if (ifindex)
printf ("%d\n", ifindex);
return !!ifindex;
}
LINK_CMD_GET_FULL (get_name, string, value)
LINK_CMD_GET_FULL (get_type, decimal, value > 0)
LINK_CMD (set_up)
LINK_CMD (set_down)
LINK_CMD (set_arp)
LINK_CMD (set_noarp)
LINK_CMD_GET (is_up, boolean)
LINK_CMD_GET (is_connected, boolean)
LINK_CMD_GET (uses_arp, boolean)
static gboolean
do_link_set_address (char **argv)
{
int ifindex = parse_ifindex (*argv++);
char *hex = *argv++;
int hexlen = strlen (hex);
char address[hexlen/2];
char *endptr;
int i;
g_assert (!(hexlen % 2));
for (i = 0; i < sizeof (address); i++) {
char digit[3];
digit[0] = hex[2*i];
digit[1] = hex[2*i+1];
digit[2] = '\0';
address[i] = strtoul (digit, &endptr, 16);
g_assert (!*endptr);
}
return nm_platform_link_set_address (ifindex, address, sizeof (address));
}
static gboolean
do_link_get_address (char **argv)
{
int ifindex = parse_ifindex (*argv++);
const char *address;
size_t length;
int i;
address = nm_platform_link_get_address (ifindex, &length);
if (!address || length <= 0)
return FALSE;
for (i = 0; i < length; i++)
printf ("%02x", address[i]);
printf ("\n");
return TRUE;
}
LINK_CMD_GET (supports_carrier_detect, boolean)
LINK_CMD_GET (supports_vlans, boolean)
static gboolean
do_link_enslave (char **argv)
{
int master = parse_ifindex (*argv++);
int slave = parse_ifindex (*argv++);
return nm_platform_link_enslave (master, slave);
}
static gboolean
do_link_release (char **argv)
{
int master = parse_ifindex (*argv++);
int slave = parse_ifindex (*argv++);
return nm_platform_link_release (master, slave);
}
LINK_CMD_GET (get_master, decimal)
static gboolean
do_master_set_option (char **argv)
{
int ifindex = parse_ifindex (*argv++);
const char *option = *argv++;
const char *value = *argv++;
return nm_platform_master_set_option (ifindex, option, value);
}
static gboolean
do_master_get_option (char **argv)
{
int ifindex = parse_ifindex (*argv++);
const char *option = *argv++;
auto_g_free char *value = nm_platform_master_get_option (ifindex, option);
printf ("%s\n", value);
return !!value;
}
static gboolean
do_slave_set_option (char **argv)
{
int ifindex = parse_ifindex (*argv++);
const char *option = *argv++;
const char *value = *argv++;
return nm_platform_slave_set_option (ifindex, option, value);
}
static gboolean
do_slave_get_option (char **argv)
{
int ifindex = parse_ifindex (*argv++);
const char *option = *argv++;
auto_g_free char *value = nm_platform_slave_get_option (ifindex, option);
printf ("%s\n", value);
return !!value;
}
static gboolean
do_vlan_get_info (char **argv)
{
int ifindex = parse_ifindex (*argv++);
int parent;
int vlanid;
if (!nm_platform_vlan_get_info (ifindex, &parent, &vlanid))
return FALSE;
printf ("%d %d\n", parent, vlanid);
return TRUE;
}
static gboolean
do_vlan_set_ingress_map (char **argv)
{
int ifindex = parse_ifindex (*argv++);
int from = strtol (*argv++, NULL, 10);
int to = strtol (*argv++, NULL, 10);
return nm_platform_vlan_set_ingress_map (ifindex, from, to);
}
static gboolean
do_vlan_set_egress_map (char **argv)
{
int ifindex = parse_ifindex (*argv++);
int from = strtol (*argv++, NULL, 10);
int to = strtol (*argv++, NULL, 10);
return nm_platform_vlan_set_egress_map (ifindex, from, to);
}
static gboolean
do_ip4_address_get_all (char **argv)
{
int ifindex = parse_ifindex (argv[0]);
GArray *addresses;
NMPlatformIP4Address *address;
char addrstr[INET_ADDRSTRLEN];
int i;
if (ifindex) {
addresses = nm_platform_ip4_address_get_all (ifindex);
for (i = 0; i < addresses->len; i++) {
address = &g_array_index (addresses, NMPlatformIP4Address, i);
inet_ntop (AF_INET, &address->address, addrstr, sizeof (addrstr));
printf ("%s/%d\n", addrstr, address->plen);
}
g_array_unref (addresses);
}
return !!ifindex;
}
static gboolean
do_ip6_address_get_all (char **argv)
{
int ifindex = parse_ifindex (argv[0]);
GArray *addresses;
NMPlatformIP6Address *address;
char addrstr[INET6_ADDRSTRLEN];
int i;
if (ifindex) {
addresses = nm_platform_ip6_address_get_all (ifindex);
for (i = 0; i < addresses->len; i++) {
address = &g_array_index (addresses, NMPlatformIP6Address, i);
inet_ntop (AF_INET6, &address->address, addrstr, sizeof (addrstr));
printf ("%s/%d\n", addrstr, address->plen);
}
g_array_unref (addresses);
}
return !!ifindex;
}
static gboolean
parse_ip_address (int family, char *str, gpointer address, int *plen)
{
char *endptr;
if (plen) {
char *ptr = strchr (str, '/');
if (ptr) {
*ptr++ = '\0';
*plen = strtol (ptr, &endptr, 10);
if (*endptr)
ptr = NULL;
}
if (!ptr) {
error ("Bad format of IP address, expected address/plen.\n");
return FALSE;
}
}
if (inet_pton (family, str, address))
return TRUE;
error ("Bad format of IP address, expected address%s.\n", plen ? "/plen" : "");
return FALSE;
}
typedef in_addr_t ip4_t;
typedef struct in6_addr ip6_t;
#define parse_ip4_address(s, a, p) parse_ip_address (AF_INET, s, a, p)
#define parse_ip6_address(s, a, p) parse_ip_address (AF_INET6, s, a, p)
#define ADDR_CMD_FULL(v, cmdname, print) \
static gboolean \
do_##v##_address_##cmdname (char **argv) \
{ \
int ifindex = parse_ifindex (argv[0]); \
v##_t address; \
int plen; \
if (ifindex && parse_##v##_address (argv[1], &address, &plen)) { \
gboolean value = nm_platform_##v##_address_##cmdname (ifindex, address, plen); \
if (print) { \
print_boolean (value); \
return TRUE; \
} else \
return value; \
} else \
return FALSE; \
}
#define ADDR_CMD(cmdname) ADDR_CMD_FULL (ip4, cmdname, FALSE) ADDR_CMD_FULL (ip6, cmdname, FALSE)
#define ADDR_CMD_PRINT(cmdname) ADDR_CMD_FULL (ip4, cmdname, TRUE) ADDR_CMD_FULL (ip6, cmdname, TRUE)
ADDR_CMD (add)
ADDR_CMD (delete)
ADDR_CMD_PRINT (exists)
static gboolean
do_ip4_route_get_all (char **argv)
{
int ifindex = parse_ifindex (argv[0]);
GArray *routes;
NMPlatformIP4Route *route;
char networkstr[INET_ADDRSTRLEN], gatewaystr[INET_ADDRSTRLEN];
int i;
if (ifindex) {
routes = nm_platform_ip4_route_get_all (ifindex);
for (i = 0; i < routes->len; i++) {
route = &g_array_index (routes, NMPlatformIP4Route, i);
inet_ntop (AF_INET, &route->network, networkstr, sizeof (networkstr));
inet_ntop (AF_INET, &route->gateway, gatewaystr, sizeof (gatewaystr));
printf ("%s/%d via %s metric %d\n",
networkstr, route->plen, gatewaystr, route->metric);
}
g_array_unref (routes);
}
return !!ifindex;
}
static gboolean
do_ip6_route_get_all (char **argv)
{
int ifindex = parse_ifindex (argv[0]);
GArray *routes;
NMPlatformIP6Route *route;
char networkstr[INET6_ADDRSTRLEN], gatewaystr[INET6_ADDRSTRLEN];
int i;
if (ifindex) {
routes = nm_platform_ip6_route_get_all (ifindex);
for (i = 0; i < routes->len; i++) {
route = &g_array_index (routes, NMPlatformIP6Route, i);
inet_ntop (AF_INET6, &route->network, networkstr, sizeof (networkstr));
inet_ntop (AF_INET6, &route->gateway, gatewaystr, sizeof (gatewaystr));
printf ("%s/%d via %s metric %d\n",
networkstr, route->plen, gatewaystr, route->metric);
}
g_array_unref (routes);
}
return !!ifindex;
}
static gboolean
do_ip4_route_add (char **argv)
{
int ifindex = parse_ifindex (*argv++);
in_addr_t network, gateway;
int plen, metric, mss;
parse_ip4_address (*argv++, &network, &plen);
parse_ip4_address (*argv++, &gateway, NULL);
metric = strtol (*argv++, NULL, 10);
mss = strtol (*argv++, NULL, 10);
return nm_platform_ip4_route_add (ifindex, network, plen, gateway, metric, mss);
}
static gboolean
do_ip6_route_add (char **argv)
{
int ifindex = parse_ifindex (*argv++);
struct in6_addr network, gateway;
int plen, metric, mss;
parse_ip6_address (*argv++, &network, &plen);
parse_ip6_address (*argv++, &gateway, NULL);
metric = strtol (*argv++, NULL, 10);
mss = strtol (*argv++, NULL, 10);
return nm_platform_ip6_route_add (ifindex, network, plen, gateway, metric, mss);
}
static gboolean
do_ip4_route_delete (char **argv)
{
int ifindex = parse_ifindex (*argv++);
in_addr_t network;
int plen, metric;
parse_ip4_address (*argv++, &network, &plen);
metric = strtol (*argv++, NULL, 10);
return nm_platform_ip4_route_delete (ifindex, network, plen, metric);
}
static gboolean
do_ip6_route_delete (char **argv)
{
int ifindex = parse_ifindex (*argv++);
struct in6_addr network;
int plen, metric;
parse_ip6_address (*argv++, &network, &plen);
metric = strtol (*argv++, NULL, 10);
return nm_platform_ip6_route_delete (ifindex, network, plen, metric);
}
static gboolean
do_ip4_route_exists (char **argv)
{
int ifindex = parse_ifindex (*argv++);
in_addr_t network;
int plen, metric;
parse_ip4_address (*argv++, &network, &plen);
metric = strtol (*argv++, NULL, 10);
print_boolean (nm_platform_ip4_route_exists (ifindex, network, plen, metric));
return TRUE;
}
static gboolean
do_ip6_route_exists (char **argv)
{
int ifindex = parse_ifindex (*argv++);
struct in6_addr network;
int plen, metric;
parse_ip6_address (*argv++, &network, &plen);
metric = strtol (*argv++, NULL, 10);
print_boolean (nm_platform_ip6_route_exists (ifindex, network, plen, metric));
return TRUE;
}
typedef struct {
const char *name;
const char *help;
int (*handler) (char **argv);
int argc;
const char *arghelp;
} command_t;
static const command_t commands[] = {
{ "sysctl-set", "get /proc/sys or /sys value", do_sysctl_set, 2, "<path> <value>" },
{ "sysctl-get", "get /proc/sys or /sys value", do_sysctl_get, 1, "<value>" },
{ "link-get-all", "print all links", do_link_get_all, 0, "" },
{ "dummy-add", "add dummy interface", do_dummy_add, 1, "<ifname>" },
{ "bridge-add", "add bridge interface", do_bridge_add, 1, "<ifname>" },
{ "bond-add", "add bond interface", do_bond_add, 1, "<ifname>" },
{ "team-add", "add team interface", do_team_add, 1, "<ifname>" },
{ "vlan-add", "add vlan interface", do_vlan_add, 4, "<ifname> <parent> <vlanid> <vlanflags>" },
{ "link-exists", "check ifname for existance", do_link_exists, 1, "<ifname>" },
{ "link-delete", "delete interface", do_link_delete, 1, "<ifname/ifindex>" },
{ "link-get-ifindex>", "get interface index", do_link_get_ifindex, 1, "<ifname>" },
{ "link-get-name", "get interface name", do_link_get_name, 1, "<ifindex>" },
{ "link-get-type", "get interface type", do_link_get_type, 1, "<ifname/ifindex>" },
{ "link-set-up", "set interface up", do_link_set_up, 1, "<ifname/ifindex>" },
{ "link-set-down", "set interface down", do_link_set_down, 1, "<ifname/ifindex>" },
{ "link-set-arp", "activate interface arp", do_link_set_arp, 1, "<ifname/ifindex>" },
{ "link-set-noarp", "deactivate interface arp", do_link_set_noarp, 1, "<ifname/ifindex>" },
{ "link-is-up", "check if interface is up", do_link_is_up, 1, "<ifname/ifindex>" },
{ "link-is-connected", "check interface carrier", do_link_is_connected, 1, "<ifname/ifindex>" },
{ "link-uses-arp", "check whether interface uses arp", do_link_uses_arp, 1, "<ifname/ifindex>" },
{ "link-get-address", "print link address", do_link_get_address, 1, "<ifname/ifindex>" },
{ "link-set-address", "set link address", do_link_set_address, 2, "<ifname/ifindex> <hex>" },
{ "link-supports-carrier-detect", "check whether interface supports carrier detect",
do_link_supports_carrier_detect, 1, "<ifname/ifindex>" },
{ "link-supports-vlans", "check whether interface supports VLANs",
do_link_supports_vlans, 1, "<ifname/ifindex>" },
{ "link-enslave", "enslave slave interface with master", do_link_enslave, 2, "<master> <slave>" },
{ "link-release", "release save interface from master", do_link_release, 2, "<master> <slave>" },
{ "link-get-master", "print master interface of a slave", do_link_get_master, 1, "<ifname/ifindex>" },
{ "link-master-set-option", "set master option", do_master_set_option, 3,
"<ifname/ifindex> <option> <value>" },
{ "link-master-get-option", "get master option", do_master_get_option, 2,
"<ifname/ifindex> <option>" },
{ "link-slave-set-option", "set slave option", do_slave_set_option, 3,
"<ifname/ifindex> <option>" },
{ "link-slave-get-option", "get slave option", do_slave_get_option, 2,
"<ifname/ifindex> <option>" },
{ "vlan-get-info", "get vlan info", do_vlan_get_info, 1, "<ifname/ifindex>" },
{ "vlan-set-ingress-map", "set vlan ingress map", do_vlan_set_ingress_map, 3,
"<ifname/ifindex> <from> <to>" },
{ "vlan-set-egress-map", "set vlan egress map", do_vlan_set_egress_map, 3,
"<ifname/ifindex> <from> <to>" },
{ "ip4-address-get-all", "print all IPv4 addresses", do_ip4_address_get_all, 1, "<ifname/ifindex>" },
{ "ip6-address-get-all", "print all IPv6 addresses", do_ip6_address_get_all, 1, "<ifname/ifindex>" },
{ "ip4-address-add", "add IPv4 address", do_ip4_address_add, 2, "<ifname/ifindex> <address>/<plen>" },
{ "ip6-address-add", "add IPv6 address", do_ip6_address_add, 2, "<ifname/ifindex> <address>/<plen>" },
{ "ip4-address-delete", "delete IPv4 address", do_ip4_address_delete, 2,
"<ifname/ifindex> <address>/<plen>" },
{ "ip6-address-delete", "delete IPv6 address", do_ip6_address_delete, 2,
"<ifname/ifindex> <address>/<plen>" },
{ "ip4-address-exists", "check for existence of IPv4 address", do_ip4_address_exists, 2,
"<ifname/ifindex> <address>/<plen>" },
{ "ip6-address-exists", "check for existence of IPv6 address", do_ip6_address_exists, 2,
"<ifname/ifindex> <address>/<plen>" },
{ "ip4-route-get-all", "print all IPv4 routes", do_ip4_route_get_all, 1, "<ifname/ifindex>" },
{ "ip6-route-get-all", "print all IPv6 routes", do_ip6_route_get_all, 1, "<ifname/ifindex>" },
{ "ip4-route-add", "add IPv4 route", do_ip4_route_add, 5,
"<ifname/ifindex> <network>/<plen> <gateway> <metric> <mss>" },
{ "ip6-route-add", "add IPv6 route", do_ip6_route_add, 5,
"<ifname/ifindex> <network>/<plen> <gateway> <metric> <mss>" },
{ "ip4-route-delete", "delete IPv4 route", do_ip4_route_delete, 3,
"<ifname/ifindex> <network>/<plen> <metric>" },
{ "ip6-route-delete", "delete IPv6 route", do_ip6_route_delete, 3,
"<ifname/ifindex> <network>/<plen> <metric>" },
{ "ip4-route-exists", "check for existence of IPv4 route", do_ip4_route_exists, 3,
"<ifname/ifindex> <network>/<plen> <metric>" },
{ "ip6-route-exists", "check for existence of IPv6 route", do_ip6_route_exists, 3,
"<ifname/ifindex> <network>/<plen> <metric>" },
{ NULL, NULL, NULL, 0, NULL },
};
int
main (int argc, char **argv)
{
const char *arg0 = *argv++;
const command_t *command = NULL;
gboolean status = TRUE;
int error;
g_type_init ();
if (*argv && !g_strcmp0 (argv[1], "--fake")) {
nm_fake_platform_setup ();
} else
nm_linux_platform_setup ();
if (*argv)
for (command = commands; command->name; command++)
if (g_str_has_prefix (command->name, *argv))
break;
if (command && command->name) {
argv++;
if (g_strv_length (argv) == command->argc)
status = command->handler (argv);
else {
error ("Wrong number of arguments to '%s' (expected %d).\n\nUsage: %s %s %s\n-- %s\n",
command->name, command->argc,
arg0, command->name, command->arghelp, command->help);
return EXIT_FAILURE;
}
} else {
error ("Usage: %s COMMAND\n\n", arg0);
error ("COMMAND\n");
for (command = commands; command->name; command++)
error (" %s %s\n -- %s\n", command->name, command->arghelp, command->help);
error ("\n");
}
error = nm_platform_get_error ();
if (error) {
const char *msg = nm_platform_get_error_msg ();
error ("nm-platform: %s\n", msg);
}
return !!error;
}