summaryrefslogtreecommitdiff
path: root/src/initrd/nmi-dt-reader.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/initrd/nmi-dt-reader.c')
-rw-r--r--src/initrd/nmi-dt-reader.c411
1 files changed, 411 insertions, 0 deletions
diff --git a/src/initrd/nmi-dt-reader.c b/src/initrd/nmi-dt-reader.c
new file mode 100644
index 0000000000..c2431bf6b8
--- /dev/null
+++ b/src/initrd/nmi-dt-reader.c
@@ -0,0 +1,411 @@
+/* NetworkManager initrd configuration generator
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301 USA.
+ *
+ * Copyright 2019 Red Hat, Inc.
+ */
+
+#include "nm-default.h"
+
+#include "nm-initrd-generator.h"
+
+#include <arpa/inet.h>
+
+#include "nm-core-internal.h"
+
+/*****************************************************************************/
+
+#define _NMLOG(level, domain, ...) \
+ nm_log ((level), (domain), NULL, NULL, \
+ "dt-reader: " _NM_UTILS_MACRO_FIRST (__VA_ARGS__) \
+ _NM_UTILS_MACRO_REST (__VA_ARGS__))
+
+/*****************************************************************************/
+
+static gboolean
+dt_get_property (const char *base,
+ const char *dev,
+ const char *prop,
+ char **contents,
+ size_t *length)
+{
+ gs_free char *filename = g_build_filename (base, dev, prop, NULL);
+ gs_free_error GError *error = NULL;
+
+ if (!g_file_test (filename, G_FILE_TEST_EXISTS))
+ return FALSE;
+
+ if (!contents)
+ return TRUE;
+
+ if (!g_file_get_contents (filename, contents, length, &error)) {
+ _LOGW (LOGD_CORE, "%s: Can not read the %s property: %s",
+ dev, prop, error->message);
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static NMIPAddress *
+dt_get_ipaddr_property (const char *base,
+ const char *dev,
+ const char *prop,
+ int *family)
+{
+ NMIPAddress *addr;
+ gs_free char *buf = NULL;
+ size_t len;
+ gs_free_error GError *error = NULL;
+
+ if (!dt_get_property (base, dev, prop, &buf, &len))
+ return NULL;
+
+ switch (len) {
+ case 4:
+ if (*family == AF_UNSPEC)
+ *family = AF_INET;
+ break;
+ case 16:
+ if (*family == AF_UNSPEC)
+ *family = AF_INET6;
+ break;
+ default:
+ break;
+ }
+
+ if (*family == AF_UNSPEC) {
+ _LOGW (LOGD_CORE, "%s: Address %s has unrecognized length (%zd)",
+ dev, prop, len);
+ return NULL;
+ }
+
+ addr = nm_ip_address_new_binary (*family, buf, 0, &error);
+ if (!addr) {
+ _LOGW (LOGD_CORE, "%s: Address %s is malformed: %s",
+ dev, prop, error->message);
+ }
+
+ return addr;
+}
+
+static char *
+dt_get_hwaddr_property (const char *base,
+ const char *dev,
+ const char *prop)
+{
+ gs_free guint8 *buf = NULL;
+ size_t len;
+
+ if (!dt_get_property (base, dev, prop, (char **) &buf, &len))
+ return NULL;
+
+ if (len != ETH_ALEN) {
+ _LOGW (LOGD_CORE, "%s: MAC address %s has unrecognized length (%zd)",
+ dev, prop, len);
+ return NULL;
+ }
+
+ return g_strdup_printf ("%02x:%02x:%02x:%02x:%02x:%02x",
+ buf[0], buf[1], buf[2],
+ buf[3], buf[4], buf[4]);
+}
+
+static NMIPAddress *
+str_addr (const char *str, int *family)
+{
+ struct in_addr inp;
+
+ if (*family == AF_UNSPEC)
+ *family = guess_ip_address_family (str);
+
+ if (*family == AF_UNSPEC) {
+ _LOGW (LOGD_CORE, "Malformed IP address: '%s'", str);
+ return NULL;
+ }
+
+ if (*family == AF_INET && inet_aton (str, &inp)) {
+ /* For IPv4, we need to be more tolerant than
+ * nm_ip_address_new(), to recognize things like
+ * the extra zeroes in "255.255.255.000" */
+ return nm_ip_address_new_binary (*family, &inp, 0, NULL);
+ }
+
+ return nm_ip_address_new (*family, str, 0, NULL);
+}
+
+NMConnection *
+nmi_dt_reader_parse (const char *sysfs_dir)
+{
+ NMConnection *connection;
+ gs_free char *base = NULL;
+ gs_free char *bootpath = NULL;
+ gs_strfreev char **tokens = NULL;
+ char *path = NULL;
+ gboolean bootp = FALSE;
+ const char *s_ipaddr = NULL;
+ const char *s_netmask = NULL;
+ const char *s_gateway = NULL;
+ NMIPAddress *ipaddr = NULL;
+ NMIPAddress *netmask = NULL;
+ NMIPAddress *gateway = NULL;
+ const char *duplex = NULL;
+ gs_free char *hwaddr = NULL;
+ gs_free char *local_hwaddr = NULL;
+ gs_free char *hostname = NULL;
+ guint32 speed = 0;
+ int prefix = -1;
+ NMSettingIPConfig *s_ip = NULL;
+ NMSetting *s_ip4 = NULL;
+ NMSetting *s_ip6 = NULL;
+ NMSetting *s_wired = NULL;
+ int family = AF_UNSPEC;
+ int i = 0;
+ char *c;
+ gs_free_error GError *error = NULL;
+
+ base = g_build_filename (sysfs_dir, "firmware", "devicetree",
+ "base", NULL);
+
+ if (!dt_get_property (base, "chosen", "bootpath", &bootpath, NULL))
+ return NULL;
+
+ c = strchr (bootpath, ':');
+ if (c) {
+ *c = '\0';
+ path = c + 1;
+ } else {
+ path = "";
+ }
+
+ dt_get_property (base, "chosen", "client-name", &hostname, NULL);
+
+ local_hwaddr = dt_get_hwaddr_property (base, bootpath, "local-mac-address");
+ hwaddr = dt_get_hwaddr_property (base, bootpath, "mac-address");
+ if (g_strcmp0 (local_hwaddr, hwaddr) == 0)
+ g_clear_pointer (&local_hwaddr, g_free);
+
+ tokens = g_strsplit (path, ",", 0);
+
+ /*
+ * Ethernet device settings. Defined by "Open Firmware,
+ * Recommended Practice: Device Support Extensions, Version 1.0 [1]
+ * [1] https://www.devicetree.org/open-firmware/practice/devicex/dse1_0a.ps
+ */
+
+ for (i = 0; tokens[i]; i++) {
+ /* Skip these. They have magical meaning for OpenFirmware. */
+ if ( strcmp (tokens[i], "nfs") == 0
+ || strcmp (tokens[i], "last") == 0)
+ continue;
+ if (strcmp (tokens[i], "promiscuous") == 0) {
+ /* Ignore. */
+ continue;
+ }
+
+ if (g_str_has_prefix (tokens[i], "speed=")) {
+ speed = _nm_utils_ascii_str_to_int64 (tokens[i] + 6,
+ 10, 0, G_MAXUINT32, 0);
+ continue;
+ }
+
+ if (g_str_has_prefix (tokens[i], "duplex=auto")) {
+ continue;
+ } else if ( g_str_has_prefix (tokens[i], "duplex=half")
+ || g_str_has_prefix (tokens[i], "duplex=full")) {
+ duplex = tokens[i] + 7;
+ continue;
+ }
+
+ break;
+ }
+
+ /*
+ * Network boot configuration. Defined by "Open Firmware,
+ * Recommended Practice: TFTP Booting Extension, Version 1.0 [1]
+ * [1] https://www.devicetree.org/open-firmware/practice/obp-tftp/tftp1_0.pdf
+ */
+
+ for (; tokens[i]; i++) {
+ if ( strcmp (tokens[i], "bootp") == 0
+ || strcmp (tokens[i], "dhcp") == 0
+ || strcmp (tokens[i], "rarp") == 0) {
+ bootp = TRUE;
+ continue;
+ }
+ break;
+ }
+
+ /* s-iaddr, or perhaps a raw absolute filename */
+ if (tokens[i] && tokens[i][0] != '/')
+ i++;
+
+ /* filename */
+ if (tokens[i])
+ i++;
+
+ /* c-iaddr */
+ if (tokens[i]) {
+ s_ipaddr = tokens[i];
+ i++;
+ }
+
+ /* g-iaddr */
+ if (tokens[i]) {
+ s_gateway = tokens[i];
+ i++;
+ }
+
+ if (tokens[i] && ( strchr (tokens[i], '.')
+ || strchr (tokens[i], ':'))) {
+ /* yaboot claims the mask can be specified here,
+ * though it doesn't support it. */
+ s_netmask = tokens[i];
+ i++;
+ }
+
+ /* bootp-retries */
+ if (tokens[i])
+ i++;
+
+ /* tftp-retries */
+ if (tokens[i])
+ i++;
+
+ if (tokens[i]) {
+ /* yaboot accepts a mask here */
+ s_netmask = tokens[i];
+ i++;
+ }
+
+ connection = nm_simple_connection_new ();
+
+ nm_connection_add_setting (connection,
+ g_object_new (NM_TYPE_SETTING_CONNECTION,
+ NM_SETTING_CONNECTION_TYPE, NM_SETTING_WIRED_SETTING_NAME,
+ NM_SETTING_CONNECTION_ID, "OpenFirmware Connection",
+ NULL));
+
+ s_ip4 = nm_setting_ip4_config_new ();
+ nm_connection_add_setting (connection, s_ip4);
+
+ s_ip6 = nm_setting_ip6_config_new ();
+ nm_connection_add_setting (connection, s_ip6);
+
+ if (!bootp && dt_get_property (base, "chosen", "bootp-response", NULL, NULL))
+ bootp = TRUE;
+
+ if (!bootp) {
+ netmask = dt_get_ipaddr_property (base, "chosen", "netmask-ip", &family);
+ gateway = dt_get_ipaddr_property (base, "chosen", "gateway-ip", &family);
+ if (gateway)
+ s_gateway = nm_ip_address_get_address (gateway);
+ ipaddr = dt_get_ipaddr_property (base, "chosen", "client-ip", &family);
+
+ if (family == AF_UNSPEC) {
+ g_warn_if_fail (netmask == NULL);
+ g_warn_if_fail (ipaddr == NULL);
+ g_warn_if_fail (gateway == NULL);
+
+ netmask = str_addr (s_netmask, &family);
+ ipaddr = str_addr (s_ipaddr, &family);
+
+ prefix = _nm_utils_ascii_str_to_int64 (s_netmask, 10, 0, 128, -1);
+ }
+
+ if (prefix == -1 && family == AF_INET && netmask) {
+ guint32 netmask_v4;
+
+ nm_ip_address_get_address_binary (netmask, &netmask_v4);
+ prefix = nm_utils_ip4_netmask_to_prefix (netmask_v4);
+ }
+
+ if (prefix == -1)
+ _LOGW (LOGD_CORE, "Unable to determine the network prefix");
+ else
+ nm_ip_address_set_prefix (ipaddr, prefix);
+
+ if (netmask)
+ nm_ip_address_unref (netmask);
+ if (gateway)
+ nm_ip_address_unref (gateway);
+ }
+
+ if (!ipaddr) {
+ family = AF_UNSPEC;
+ bootp = TRUE;
+ }
+
+ if (bootp) {
+ g_object_set (s_ip4,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP4_CONFIG_METHOD_AUTO,
+ NM_SETTING_IP_CONFIG_DHCP_HOSTNAME, hostname,
+ NULL);
+ g_object_set (s_ip6,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP6_CONFIG_METHOD_AUTO,
+ NM_SETTING_IP_CONFIG_DHCP_HOSTNAME, hostname,
+ NULL);
+ } else {
+ switch (family) {
+ case AF_INET:
+ s_ip = (NMSettingIPConfig *) s_ip4;
+ g_object_set (s_ip4,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP4_CONFIG_METHOD_MANUAL,
+ NULL);
+ g_object_set (s_ip6,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP6_CONFIG_METHOD_DISABLED,
+ NULL);
+ break;
+ case AF_INET6:
+ s_ip = (NMSettingIPConfig *) s_ip6;
+ g_object_set (s_ip4,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP4_CONFIG_METHOD_DISABLED,
+ NULL);
+ g_object_set (s_ip6,
+ NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP6_CONFIG_METHOD_MANUAL,
+ NULL);
+ break;
+ default:
+ g_return_val_if_reached (NULL);
+ }
+
+ nm_setting_ip_config_add_address (s_ip, ipaddr);
+ g_object_set (s_ip, NM_SETTING_IP_CONFIG_GATEWAY, s_gateway, NULL);
+ }
+
+ if (ipaddr)
+ nm_ip_address_unref (ipaddr);
+
+ if (duplex || speed || hwaddr || local_hwaddr) {
+ s_wired = nm_setting_wired_new ();
+ nm_connection_add_setting (connection, s_wired);
+
+ g_object_set (s_wired,
+ NM_SETTING_WIRED_SPEED, speed,
+ NM_SETTING_WIRED_DUPLEX, duplex,
+ NM_SETTING_WIRED_MAC_ADDRESS, hwaddr,
+ NM_SETTING_WIRED_CLONED_MAC_ADDRESS, local_hwaddr,
+ NULL);
+ }
+
+ if (!nm_connection_normalize (connection, NULL, NULL, &error)) {
+ _LOGW (LOGD_CORE, "Generated an invalid connection: %s",
+ error->message);
+ g_clear_pointer (&connection, g_object_unref);
+ }
+
+ return connection;
+}