ifupdown-ng/cmd/ifupdown.c

340 lines
7.4 KiB
C
Raw Normal View History

2020-07-23 16:44:45 +00:00
/*
* cmd/ifupdown.c
* Purpose: bring interfaces up or down
*
* Copyright (c) 2020 Ariadne Conill <ariadne@dereferenced.org>
* Copyright (c) 2020 Maximilian Wilhelm <max@sdn.clinic>
2020-07-23 16:44:45 +00:00
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* This software is provided 'as is' and without any warranty, express or
* implied. In no event shall the authors be liable for any damages arising
* from the use of this software.
*/
#define _GNU_SOURCE
#include <fnmatch.h>
#include <stdio.h>
#include <string.h>
#include <getopt.h>
#include <fcntl.h>
#include <errno.h>
#include <unistd.h>
2020-07-23 16:44:45 +00:00
#include "libifupdown/libifupdown.h"
2020-07-24 09:07:19 +00:00
#include "cmd/multicall.h"
2020-07-23 16:44:45 +00:00
static bool up;
bool
is_ifdown()
{
if (strstr(argv0, "ifdown") != NULL)
return true;
return false;
}
int
acquire_state_lock(const char *state_path, const char *lifname)
{
if (exec_opts.mock || exec_opts.no_lock)
return -1;
char lockpath[4096] = {};
snprintf(lockpath, sizeof lockpath, "%s.%s.lock", state_path, lifname);
int fd = open(lockpath, O_CREAT | O_WRONLY | O_TRUNC);
if (fd < 0)
{
if (exec_opts.verbose)
fprintf(stderr, "%s: while opening lockfile %s: %s\n", argv0, lockpath, strerror(errno));
return -2;
}
int flags = fcntl(fd, F_GETFD);
if (flags < 0)
{
close(fd);
if (exec_opts.verbose)
fprintf(stderr, "%s: while getting flags for lockfile: %s\n", argv0, strerror(errno));
return -2;
}
flags |= FD_CLOEXEC;
if (fcntl(fd, F_SETFD, flags) == -1)
{
close(fd);
if (exec_opts.verbose)
fprintf(stderr, "%s: while setting lockfile close-on-exec: %s\n", argv0, strerror(errno));
return -2;
}
struct flock fl = {
.l_type = F_WRLCK,
.l_whence = SEEK_SET
};
if (exec_opts.verbose)
fprintf(stderr, "%s: acquiring lock on %s\n", argv0, lockpath);
if (fcntl(fd, F_SETLK, &fl) == -1)
{
close(fd);
if (exec_opts.verbose)
fprintf(stderr, "%s: while locking lockfile: %s\n", argv0, strerror(errno));
return -2;
}
return fd;
}
bool
skip_interface(struct lif_interface *iface, const char *ifname)
{
if (iface->is_template)
{
fprintf(stderr, "%s: cannot change state on %s (template interface)\n", argv0, ifname);
return false;
}
if (iface->has_config_error)
{
if (exec_opts.force)
{
fprintf(stderr, "%s: (de)configuring interface %s despite config errors\n", argv0, ifname);
return false;
}
else
{
fprintf(stderr, "%s: skipping interface %s due to config errors\n", argv0, ifname);
return true;
}
}
if (exec_opts.force)
return false;
if (up && iface->refcount > 0)
{
if (exec_opts.verbose)
fprintf(stderr, "%s: skipping auto interface %s (already configured), use --force to force configuration\n",
argv0, ifname);
return true;
}
if (!up && iface->refcount == 0)
{
if (exec_opts.verbose)
fprintf(stderr, "%s: skipping auto interface %s (already deconfigured), use --force to force deconfiguration\n",
argv0, ifname);
return true;
}
return false;
}
2020-07-23 16:44:45 +00:00
bool
2020-07-24 09:42:34 +00:00
change_interface(struct lif_interface *iface, struct lif_dict *collection, struct lif_dict *state, const char *ifname)
2020-07-23 16:44:45 +00:00
{
int lockfd = acquire_state_lock(exec_opts.state_file, ifname);
if (lockfd == -2)
{
fprintf(stderr, "%s: could not acquire exclusive lock for %s: %s\n", argv0, ifname, strerror(errno));
return false;
}
if (skip_interface(iface, ifname))
{
if (lockfd != -1)
close(lockfd);
return true;
}
2020-07-23 16:57:46 +00:00
if (exec_opts.verbose)
{
fprintf(stderr, "%s: changing state of interface %s to '%s'\n",
2020-07-23 16:57:46 +00:00
argv0, ifname, up ? "up" : "down");
}
2020-07-24 09:42:34 +00:00
if (!lif_lifecycle_run(&exec_opts, iface, collection, state, ifname, up))
2020-07-23 16:44:45 +00:00
{
2020-07-23 16:57:46 +00:00
fprintf(stderr, "%s: failed to change interface %s state to '%s'\n",
2020-07-23 16:44:45 +00:00
argv0, ifname, up ? "up" : "down");
if (lockfd != -1)
close(lockfd);
2020-07-23 16:44:45 +00:00
return false;
}
if (lockfd != -1)
close(lockfd);
2020-07-23 16:44:45 +00:00
return true;
}
2020-07-23 16:57:46 +00:00
bool
change_auto_interfaces(struct lif_dict *collection, struct lif_dict *state, struct match_options *opts)
{
struct lif_node *iter;
LIF_DICT_FOREACH(iter, collection)
{
struct lif_dict_entry *entry = iter->data;
struct lif_interface *iface = entry->data;
if (opts->is_auto && !iface->is_auto)
continue;
if (opts->exclude_pattern != NULL &&
!fnmatch(opts->exclude_pattern, iface->ifname, 0))
continue;
if (opts->include_pattern != NULL &&
fnmatch(opts->include_pattern, iface->ifname, 0))
continue;
2020-07-24 09:42:34 +00:00
if (!change_interface(iface, collection, state, iface->ifname))
2020-07-23 16:57:46 +00:00
return false;
}
return true;
}
int
update_state_file_and_exit(int rc, struct lif_dict *state)
{
if (exec_opts.mock)
{
exit(rc);
return rc;
}
if (!lif_state_write_path(state, exec_opts.state_file))
{
fprintf(stderr, "%s: could not update %s\n", argv0, exec_opts.state_file);
exit(EXIT_FAILURE);
return EXIT_FAILURE;
}
exit(rc);
return rc;
}
2020-07-23 16:44:45 +00:00
int
2020-07-24 09:07:19 +00:00
ifupdown_main(int argc, char *argv[])
2020-07-23 16:44:45 +00:00
{
up = !is_ifdown();
struct lif_dict state = {};
struct lif_dict collection = {};
struct lif_interface_file_parse_state parse_state = {
.collection = &collection,
};
2020-07-23 16:44:45 +00:00
lif_interface_collection_init(&collection);
if (!lif_state_read_path(&state, exec_opts.state_file))
2020-07-23 16:44:45 +00:00
{
fprintf(stderr, "%s: could not parse %s\n", argv0, exec_opts.state_file);
2020-07-23 16:44:45 +00:00
return EXIT_FAILURE;
}
if (!lif_interface_file_parse(&parse_state, exec_opts.interfaces_file))
2020-07-23 16:44:45 +00:00
{
fprintf(stderr, "%s: could not parse %s\n", argv0, exec_opts.interfaces_file);
return EXIT_FAILURE;
}
if (lif_lifecycle_count_rdepends(&exec_opts, &collection) == -1)
{
fprintf(stderr, "%s: could not validate dependency tree\n", argv0);
return EXIT_FAILURE;
}
if(!lif_compat_apply(&collection))
{
fprintf(stderr, "%s: failed to apply compatibility glue\n", argv0);
return EXIT_FAILURE;
}
if (!lif_state_sync(&state, &collection))
{
fprintf(stderr, "%s: could not sync state\n", argv0);
2020-07-23 16:44:45 +00:00
return EXIT_FAILURE;
}
2020-07-23 16:57:46 +00:00
if (match_opts.is_auto)
{
if (!change_auto_interfaces(&collection, &state, &match_opts))
return update_state_file_and_exit(EXIT_FAILURE, &state);
2020-07-23 16:57:46 +00:00
return update_state_file_and_exit(EXIT_SUCCESS, &state);
2020-07-23 16:57:46 +00:00
}
else if (optind >= argc)
2020-08-12 05:14:55 +00:00
generic_usage(self_applet, EXIT_FAILURE);
2020-07-23 16:44:45 +00:00
int idx = optind;
for (; idx < argc; idx++)
{
char lifbuf[4096];
strlcpy(lifbuf, argv[idx], sizeof lifbuf);
char *ifname = lifbuf;
char *lifname = lifbuf;
char *p;
if ((p = strchr(lifbuf, '=')) != NULL)
{
*p++ = '\0';
lifname = p;
}
struct lif_interface *iface = lif_state_lookup(&state, &collection, argv[idx]);
if (iface == NULL)
2020-07-23 16:44:45 +00:00
{
struct lif_dict_entry *entry = lif_dict_find(&collection, lifname);
if (entry == NULL)
{
fprintf(stderr, "%s: unknown interface %s\n", argv0, argv[idx]);
return update_state_file_and_exit(EXIT_FAILURE, &state);
}
iface = entry->data;
2020-07-23 16:44:45 +00:00
}
2020-07-24 09:42:34 +00:00
if (!change_interface(iface, &collection, &state, ifname))
return update_state_file_and_exit(EXIT_FAILURE, &state);
2020-07-23 17:01:05 +00:00
}
2020-07-23 16:44:45 +00:00
return update_state_file_and_exit(EXIT_SUCCESS, &state);
2020-07-23 16:44:45 +00:00
}
2020-07-24 09:07:19 +00:00
struct if_applet ifup_applet = {
.name = "ifup",
2020-08-12 05:14:55 +00:00
.desc = "bring interfaces up",
2020-07-24 09:07:19 +00:00
.main = ifupdown_main,
2020-08-12 05:14:55 +00:00
.usage = "ifup [options] <interfaces>",
.manpage = "8 ifup",
2020-08-12 05:14:55 +00:00
.groups = { &global_option_group, &match_option_group, &exec_option_group, },
2020-07-24 09:07:19 +00:00
};
struct if_applet ifdown_applet = {
.name = "ifdown",
2020-08-12 05:14:55 +00:00
.desc = "take interfaces down",
2020-07-24 09:07:19 +00:00
.main = ifupdown_main,
2020-08-12 05:14:55 +00:00
.usage = "ifdown [options] <interfaces>",
.manpage = "8 ifdown",
2020-08-12 05:14:55 +00:00
.groups = { &global_option_group, &match_option_group, &exec_option_group, },
2020-07-24 09:07:19 +00:00
};