/* * Copyright (C) 2016 Canonical, Ltd. * Author: Martin Pitt * * 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; version 3. * * 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, see . */ #include #include #include #include #include #include #include #include #include #include "util.h" #include "parse.h" #include "networkd.h" #include "nm.h" #include "openvswitch.h" #include "sriov.h" static gchar* rootdir; static gchar** files; static gboolean any_networkd; static gboolean any_sriov; static gchar* mapping_iface; static GOptionEntry options[] = { {"root-dir", 'r', 0, G_OPTION_ARG_FILENAME, &rootdir, "Search for and generate configuration files in this root directory instead of /"}, {G_OPTION_REMAINING, 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &files, "Read configuration from this/these file(s) instead of /etc/netplan/*.yaml", "[config file ..]"}, {"mapping", 0, 0, G_OPTION_ARG_STRING, &mapping_iface, "Only show the device to backend mapping for the specified interface."}, {NULL} }; static void reload_udevd(void) { const gchar *argv[] = { "/bin/udevadm", "control", "--reload", NULL }; g_spawn_sync(NULL, (gchar**)argv, NULL, G_SPAWN_STDERR_TO_DEV_NULL, NULL, NULL, NULL, NULL, NULL, NULL); }; // LCOV_EXCL_START /* covered via 'cloud-init' integration test */ static gboolean check_called_just_in_time() { const gchar *argv[] = { "/bin/systemctl", "is-system-running", NULL }; gchar *output = NULL; g_spawn_sync(NULL, (gchar**)argv, NULL, G_SPAWN_STDERR_TO_DEV_NULL, NULL, NULL, &output, NULL, NULL, NULL); if (output != NULL && strstr(output, "initializing") != NULL) { g_free(output); const gchar *argv2[] = { "/bin/systemctl", "is-active", "network.target", NULL }; gint exit_code = 0; g_spawn_sync(NULL, (gchar**)argv2, NULL, G_SPAWN_STDERR_TO_DEV_NULL, NULL, NULL, NULL, NULL, &exit_code, NULL); /* return TRUE, if network.target is not yet active */ return !g_spawn_check_exit_status(exit_code, NULL); } g_free(output); return FALSE; }; static void start_unit_jit(gchar *unit) { const gchar *argv[] = { "/bin/systemctl", "start", "--no-block", "--no-ask-password", unit, NULL }; g_spawn_sync(NULL, (gchar**)argv, NULL, G_SPAWN_DEFAULT, NULL, NULL, NULL, NULL, NULL, NULL); }; // LCOV_EXCL_STOP static void nd_iterator_list(gpointer value, gpointer user_data) { NetplanNetDefinition* def = (NetplanNetDefinition*) value; if (write_networkd_conf(def, (const char*) user_data)) any_networkd = TRUE; write_ovs_conf(def, (const char*) user_data); write_nm_conf(def, (const char*) user_data); if (def->sriov_explicit_vf_count < G_MAXUINT || def->sriov_link) any_sriov = TRUE; } static int find_interface(gchar* interface) { GPtrArray *found; GFileInfo *info; GFile *driver_file; gchar *driver_path; gchar *driver = NULL; gpointer key, value; GHashTableIter iter; int ret = EXIT_FAILURE; found = g_ptr_array_new (); /* Try to get the driver name for the interface... */ driver_path = g_strdup_printf("/sys/class/net/%s/device/driver", interface); driver_file = g_file_new_for_path (driver_path); info = g_file_query_info (driver_file, G_FILE_ATTRIBUTE_STANDARD_SYMLINK_TARGET, 0, NULL, NULL); if (info != NULL) { /* testing for driver matching is done via autopkgtest */ // LCOV_EXCL_START driver = g_path_get_basename (g_file_info_get_symlink_target (info)); g_object_unref (info); // LCOV_EXCL_STOP } g_object_unref (driver_file); g_free (driver_path); g_hash_table_iter_init (&iter, netdefs); while (g_hash_table_iter_next (&iter, &key, &value)) { NetplanNetDefinition *nd = (NetplanNetDefinition *) value; if (!g_strcmp0(nd->set_name, interface)) g_ptr_array_add (found, (gpointer) nd); else if (!g_strcmp0(nd->id, interface)) g_ptr_array_add (found, (gpointer) nd); else if (!g_strcmp0(nd->match.original_name, interface)) g_ptr_array_add (found, (gpointer) nd); } if (found->len == 0 && driver != NULL) { /* testing for driver matching is done via autopkgtest */ // LCOV_EXCL_START g_hash_table_iter_init (&iter, netdefs); while (g_hash_table_iter_next (&iter, &key, &value)) { NetplanNetDefinition *nd = (NetplanNetDefinition *) value; if (!g_strcmp0(nd->match.driver, driver)) g_ptr_array_add (found, (gpointer) nd); } // LCOV_EXCL_STOP } if (driver) g_free (driver); // LCOV_EXCL_LINE if (found->len != 1) { goto exit_find; } else { const NetplanNetDefinition *nd = (NetplanNetDefinition *)g_ptr_array_index (found, 0); g_printf("id=%s, backend=%s, set_name=%s, match_name=%s, match_mac=%s, match_driver=%s\n", nd->id, netplan_backend_to_name[nd->backend], nd->set_name, nd->match.original_name, nd->match.mac, nd->match.driver); } ret = EXIT_SUCCESS; exit_find: g_ptr_array_free (found, TRUE); return ret; } int main(int argc, char** argv) { GError* error = NULL; GOptionContext* opt_context; /* are we being called as systemd generator? */ gboolean called_as_generator = (strstr(argv[0], "systemd/system-generators/") != NULL); g_autofree char* generator_run_stamp = NULL; glob_t gl; /* Parse CLI options */ opt_context = g_option_context_new(NULL); if (called_as_generator) g_option_context_set_help_enabled(opt_context, FALSE); g_option_context_set_summary(opt_context, "Generate backend network configuration from netplan YAML definition."); g_option_context_set_description(opt_context, "This program reads the specified netplan YAML definition file(s)\n" "or, if none are given, /etc/netplan/*.yaml.\n" "It then generates the corresponding systemd-networkd, NetworkManager,\n" "and udev configuration files in /run."); g_option_context_add_main_entries(opt_context, options, NULL); if (!g_option_context_parse(opt_context, &argc, &argv, &error)) { g_fprintf(stderr, "failed to parse options: %s\n", error->message); return 1; } if (called_as_generator) { if (files == NULL || g_strv_length(files) != 3 || files[0] == NULL) { g_fprintf(stderr, "%s can not be called directly, use 'netplan generate'.", argv[0]); return 1; } generator_run_stamp = g_build_path(G_DIR_SEPARATOR_S, files[0], "netplan.stamp", NULL); if (g_access(generator_run_stamp, F_OK) == 0) { g_fprintf(stderr, "netplan generate already ran, remove %s to force re-run\n", generator_run_stamp); return 0; } } /* Read all input files */ if (files && !called_as_generator) { for (gchar** f = files; f && *f; ++f) process_input_file(*f); } else if (!process_yaml_hierarchy(rootdir)) return 1; // LCOV_EXCL_LINE netdefs = netplan_finish_parse(&error); if (error) { g_fprintf(stderr, "%s\n", error->message); exit(1); } /* Clean up generated config from previous runs */ cleanup_networkd_conf(rootdir); cleanup_nm_conf(rootdir); cleanup_ovs_conf(rootdir); cleanup_sriov_conf(rootdir); if (mapping_iface && netdefs) return find_interface(mapping_iface); /* Generate backend specific configuration files from merged data. */ write_ovs_conf_finish(rootdir); // OVS cleanup unit is always written if (netdefs) { g_debug("Generating output files.."); g_list_foreach (netdefs_ordered, nd_iterator_list, rootdir); write_nm_conf_finish(rootdir); if (any_sriov) write_sriov_conf_finish(rootdir); /* We may have written .rules & .link files, thus we must * invalidate udevd cache of its config as by default it only * invalidates cache at most every 3 seconds. Not sure if this * should live in `generate' or `apply', but it is confusing * when udevd ignores just-in-time created rules files. */ reload_udevd(); } /* Disable /usr/lib/NetworkManager/conf.d/10-globally-managed-devices.conf * (which restricts NM to wifi and wwan) if global renderer is NM */ if (netplan_get_global_backend() == NETPLAN_BACKEND_NM) g_string_free_to_file(g_string_new(NULL), rootdir, "/run/NetworkManager/conf.d/10-globally-managed-devices.conf", NULL); if (called_as_generator) { /* Ensure networkd starts if we have any configuration for it */ if (any_networkd) enable_networkd(files[0]); /* Leave a stamp file so that we don't regenerate the configuration * multiple times and userspace can wait for it to finish */ FILE* f = fopen(generator_run_stamp, "w"); g_assert(f != NULL); fclose(f); } else if (check_called_just_in_time()) { /* netplan-feature: generate-just-in-time */ /* When booting with cloud-init, network configuration * might be provided just-in-time. Specifically after * system-generators were executed, but before * network.target is started. In such case, auxiliary * units that netplan enables have not been included in * the initial boot transaction. Detect such scenario and * add all netplan units to the initial boot transaction. */ // LCOV_EXCL_START /* covered via 'cloud-init' integration test */ if (any_networkd) { start_unit_jit("systemd-networkd.socket"); start_unit_jit("systemd-networkd-wait-online.service"); start_unit_jit("systemd-networkd.service"); } g_autofree char* glob_run = g_strjoin(NULL, rootdir ?: "", G_DIR_SEPARATOR_S, "run/systemd/system/netplan-*.service", NULL); if (!glob(glob_run, 0, NULL, &gl)) { for (size_t i = 0; i < gl.gl_pathc; ++i) { gchar *unit_name = g_path_get_basename(gl.gl_pathv[i]); start_unit_jit(unit_name); g_free(unit_name); } } // LCOV_EXCL_STOP } return 0; }