/* * libifupdown/lifecycle.c * Purpose: management of interface lifecycle (bring up, takedown, reload) * * Copyright (c) 2020 Ariadne Conill * * 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. */ #include #include #include "libifupdown/environment.h" #include "libifupdown/execute.h" #include "libifupdown/interface.h" #include "libifupdown/lifecycle.h" #include "libifupdown/state.h" static bool handle_commands_for_phase(const struct lif_execute_opts *opts, char *const envp[], struct lif_interface *iface, const char *lifname, const char *phase) { struct lif_node *iter; (void) lifname; LIF_DICT_FOREACH(iter, &iface->vars) { struct lif_dict_entry *entry = iter->data; if (strcmp(entry->key, phase)) continue; const char *cmd = entry->data; if (!lif_execute_fmt(opts, envp, "%s", cmd)) return false; } return true; } static bool handle_address(const struct lif_execute_opts *opts, struct lif_address *addr, const char *cmd, const char *lifname) { char addrbuf[4096]; if (!lif_address_unparse(addr, addrbuf, sizeof addrbuf, true)) return false; return lif_execute_fmt(opts, NULL, "/sbin/ip -%d addr %s %s dev %s", addr->domain == AF_INET ? 4 : 6, cmd, addrbuf, lifname); } static bool handle_gateway(const struct lif_execute_opts *opts, const char *gateway, const char *cmd) { int ipver = strchr(gateway, ':') ? 6 : 4; return lif_execute_fmt(opts, NULL, "/sbin/ip -%d route %s default via %s", ipver, cmd, gateway); } static bool handle_pre_up(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *lifname) { (void) opts; (void) iface; (void) lifname; return true; } static bool handle_up(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *lifname) { struct lif_node *iter; if (!lif_execute_fmt(opts, NULL, "/sbin/ip link set up dev %s", lifname)) return false; if (iface->is_loopback) return true; LIF_DICT_FOREACH(iter, &iface->vars) { struct lif_dict_entry *entry = iter->data; if (!strcmp(entry->key, "address")) { struct lif_address *addr = entry->data; if (!handle_address(opts, addr, "add", lifname)) return false; } else if (!strcmp(entry->key, "gateway")) { if (!handle_gateway(opts, entry->data, "add")) return false; } } if (iface->is_dhcp) { /* XXX: determine which dhcp client we should use */ if (!lif_execute_fmt(opts, NULL, "/sbin/udhcpc -b -R -p /var/run/udhcpc.%s.pid -i %s", lifname, lifname)) return false; } return true; } static bool handle_down(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *lifname) { struct lif_node *iter; if (iface->is_loopback) goto skip_addresses; LIF_DICT_FOREACH(iter, &iface->vars) { struct lif_dict_entry *entry = iter->data; if (!strcmp(entry->key, "address")) { struct lif_address *addr = entry->data; if (!handle_address(opts, addr, "del", lifname)) return false; } else if (!strcmp(entry->key, "gateway")) { if (!handle_gateway(opts, entry->data, "del")) return false; } } if (iface->is_dhcp) { /* XXX: determine which dhcp client we should use */ if (!lif_execute_fmt(opts, NULL, "/bin/kill $(cat /var/run/udhcpc.%s.pid)", lifname)) return false; } skip_addresses: if (!lif_execute_fmt(opts, NULL, "/sbin/ip link set down dev %s", lifname)) return false; return true; } static bool handle_post_down(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *lifname) { (void) opts; (void) iface; (void) lifname; return true; } bool lif_lifecycle_run_phase(const struct lif_execute_opts *opts, struct lif_interface *iface, const char *phase, const char *lifname, bool up) { char **envp = NULL; lif_environment_push(&envp, "IFACE", iface->ifname); lif_environment_push(&envp, "PHASE", phase); /* try to provide $METHOD for ifupdown1 scripts if we can */ if (iface->is_loopback) lif_environment_push(&envp, "METHOD", "loopback"); else if (iface->is_dhcp) lif_environment_push(&envp, "METHOD", "dhcp"); /* same for $MODE */ if (up) lif_environment_push(&envp, "MODE", "start"); else lif_environment_push(&envp, "MODE", "stop"); struct lif_node *iter; bool did_address = false, did_gateway = false; LIF_DICT_FOREACH(iter, &iface->vars) { struct lif_dict_entry *entry = iter->data; if (!strcmp(entry->key, "address")) { if (did_address) continue; struct lif_address *addr = entry->data; char addrbuf[4096]; if (!lif_address_unparse(addr, addrbuf, sizeof addrbuf, true)) continue; lif_environment_push(&envp, "IF_ADDRESS", addrbuf); did_address = true; continue; } else if (!strcmp(entry->key, "gateway")) { if (did_gateway) continue; did_gateway = true; } char envkey[4096] = "IF_"; strlcat(envkey, entry->key, sizeof envkey); char *ep = envkey + 2; while (*ep++) { *ep = toupper(*ep); } lif_environment_push(&envp, envkey, (const char *) entry->data); } if (!strcmp(phase, "pre-up")) { if (!handle_pre_up(opts, iface, lifname)) goto on_error; } else if (!strcmp(phase, "up")) { if (!handle_up(opts, iface, lifname)) goto on_error; } else if (!strcmp(phase, "down")) { if (!handle_down(opts, iface, lifname)) goto on_error; } else if (!strcmp(phase, "post-down")) { if (!handle_post_down(opts, iface, lifname)) goto on_error; } handle_commands_for_phase(opts, envp, iface, lifname, phase); /* we should do error handling here, but ifupdown1 doesn't */ lif_execute_fmt(opts, envp, "/bin/run-parts /etc/network/if-%s.d", phase); lif_environment_free(&envp); return true; on_error: lif_environment_free(&envp); return false; } bool lif_lifecycle_run(const struct lif_execute_opts *opts, struct lif_interface *iface, struct lif_dict *state, const char *lifname, bool up) { if (lifname == NULL) lifname = iface->ifname; /* XXX: actually handle dependents here */ if (up) { /* XXX: we should try to recover (take the iface down) if bringing it up fails. * but, right now neither debian ifupdown or busybox ifupdown do any recovery, * so we wont right now. */ if (!lif_lifecycle_run_phase(opts, iface, "pre-up", lifname, up)) return false; if (!lif_lifecycle_run_phase(opts, iface, "up", lifname, up)) return false; if (!lif_lifecycle_run_phase(opts, iface, "post-up", lifname, up)) return false; lif_state_upsert(state, lifname, iface); } else { if (!lif_lifecycle_run_phase(opts, iface, "pre-down", lifname, up)) return false; if (!lif_lifecycle_run_phase(opts, iface, "down", lifname, up)) return false; if (!lif_lifecycle_run_phase(opts, iface, "post-down", lifname, up)) return false; lif_state_delete(state, lifname); } return true; }