summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBeniamino Galvani <bgalvani@redhat.com>2019-03-19 17:33:40 +0100
committerBeniamino Galvani <bgalvani@redhat.com>2019-03-26 17:19:39 +0100
commitc66d17d4fc3fd06e4e9853ff0929c3844fbed1b6 (patch)
tree8eda2eb034bacc40293e87f373bf9b42d0ba99df
parentcf71860183dcabd47846ff7f07e0ae28ddab9123 (diff)
downloadNetworkManager-c66d17d4fc3fd06e4e9853ff0929c3844fbed1b6.tar.gz
keyfile: support bridge vlans
-rw-r--r--libnm-core/nm-keyfile.c93
-rw-r--r--libnm-core/tests/test-keyfile.c89
2 files changed, 182 insertions, 0 deletions
diff --git a/libnm-core/nm-keyfile.c b/libnm-core/nm-keyfile.c
index a4b0064396..0d833752ef 100644
--- a/libnm-core/nm-keyfile.c
+++ b/libnm-core/nm-keyfile.c
@@ -1519,6 +1519,57 @@ team_config_parser (KeyfileReaderInfo *info, NMSetting *setting, const char *key
}
static void
+bridge_vlan_parser (KeyfileReaderInfo *info, NMSetting *setting, const char *key)
+{
+ const char *setting_name = nm_setting_get_name (setting);
+ gs_unref_ptrarray GPtrArray *vlans = NULL;
+ gs_strfreev char **keys = NULL;
+ gsize n_keys = 0;
+ int i;
+
+ keys = nm_keyfile_plugin_kf_get_keys (info->keyfile, setting_name, &n_keys, NULL);
+ if (n_keys == 0)
+ return;
+
+ vlans = g_ptr_array_new_with_free_func ((GDestroyNotify) nm_bridge_vlan_unref);
+
+ for (i = 0; i < n_keys; i++) {
+ NMBridgeVlan *vlan;
+ const char *index;
+ gs_free char *vlan_rest = NULL;
+ gs_free char *vlan_str = NULL;
+ gs_free_error GError *err = NULL;
+
+ if (!g_str_has_prefix (keys[i], "vlan."))
+ continue;
+
+ index = keys[i] + NM_STRLEN("vlan.");
+
+ if (index[0] == '\0')
+ continue;
+ if (index[0] == '0' && index[1] != '\0')
+ continue;
+ if (!NM_STRCHAR_ALL (index, ch, g_ascii_isdigit (ch)))
+ continue;
+
+ vlan_rest = nm_keyfile_plugin_kf_get_string (info->keyfile, setting_name, keys[i], NULL);
+ vlan_str = g_strdup_printf ("%s %s", index, vlan_rest);
+
+ vlan = nm_bridge_vlan_from_str (vlan_str, &err);
+ if (!vlan) {
+ handle_warn (info, keys[i], NM_KEYFILE_WARN_SEVERITY_WARN,
+ _("invalid bridge vlan: %s"),
+ err->message);
+ continue;
+ }
+ g_ptr_array_add (vlans, vlan);
+ }
+
+ if (vlans->len >= 1)
+ g_object_set (setting, key, vlans, NULL);
+}
+
+static void
qdisc_parser (KeyfileReaderInfo *info, NMSetting *setting, const char *key)
{
const char *setting_name = nm_setting_get_name (setting);
@@ -1843,6 +1894,34 @@ route_writer (KeyfileWriterInfo *info,
}
static void
+bridge_vlan_writer (KeyfileWriterInfo *info,
+ NMSetting *setting,
+ const char *key,
+ const GValue *value)
+{
+ gsize i;
+ GPtrArray *array;
+ nm_auto_free_gstring GString *value_str = NULL;
+
+ array = (GPtrArray *) g_value_get_boxed (value);
+ if (!array || !array->len)
+ return;
+
+ for (i = 0; i < array->len; i++) {
+ NMBridgeVlan *vlan = array->pdata[i];
+ char key_name[32];
+
+ nm_sprintf_buf (key_name, "vlan.%u", nm_bridge_vlan_get_vid (vlan));
+ nm_gstring_prepare (&value_str);
+ _nm_bridge_vlan_str_append_rest (vlan, value_str, FALSE);
+ nm_keyfile_plugin_kf_set_string (info->keyfile,
+ nm_setting_get_name (setting),
+ key_name,
+ value_str->str);
+ }
+}
+
+static void
qdisc_writer (KeyfileWriterInfo *info,
NMSetting *setting,
const char *key,
@@ -2272,6 +2351,20 @@ static const ParseInfoSetting *const parse_infos[_NM_META_SETTING_TYPE_NUM] = {
PARSE_INFO_PROPERTY (NM_SETTING_BRIDGE_MAC_ADDRESS,
.parser = mac_address_parser_ETHER,
),
+ PARSE_INFO_PROPERTY (NM_SETTING_BRIDGE_VLANS,
+ .parser_no_check_key = TRUE,
+ .parser = bridge_vlan_parser,
+ .writer = bridge_vlan_writer,
+ ),
+ ),
+ ),
+ PARSE_INFO_SETTING (NM_META_SETTING_TYPE_BRIDGE_PORT,
+ PARSE_INFO_PROPERTIES (
+ PARSE_INFO_PROPERTY (NM_SETTING_BRIDGE_PORT_VLANS,
+ .parser_no_check_key = TRUE,
+ .parser = bridge_vlan_parser,
+ .writer = bridge_vlan_writer,
+ ),
),
),
PARSE_INFO_SETTING (NM_META_SETTING_TYPE_CONNECTION,
diff --git a/libnm-core/tests/test-keyfile.c b/libnm-core/tests/test-keyfile.c
index 9b9d1c0415..fbdd7ab27b 100644
--- a/libnm-core/tests/test-keyfile.c
+++ b/libnm-core/tests/test-keyfile.c
@@ -744,6 +744,93 @@ test_vpn_1 (void)
/*****************************************************************************/
+static void
+test_bridge_vlans (void)
+{
+ gs_unref_keyfile GKeyFile *keyfile = NULL;
+ gs_unref_object NMConnection *con = NULL;
+ NMSettingBridge *s_bridge;
+ NMBridgeVlan *vlan;
+
+ con = nmtst_create_connection_from_keyfile (
+ "[connection]\n"
+ "id=t\n"
+ "type=bridge\n"
+ "interface-name=br4\n"
+ "\n"
+ "[bridge]\n"
+ "vlan.9=untagged\n"
+ "vlan.1=pvid untagged\n"
+ "",
+ "/test_bridge_port/vlans");
+ s_bridge = NM_SETTING_BRIDGE (nm_connection_get_setting (con, NM_TYPE_SETTING_BRIDGE));
+ g_assert (s_bridge);
+ g_assert_cmpuint (nm_setting_bridge_get_num_vlans (s_bridge), ==, 2);
+
+ vlan = nm_setting_bridge_get_vlan (s_bridge, 0);
+ g_assert (vlan);
+ g_assert_cmpuint (nm_bridge_vlan_get_vid (vlan), ==, 1);
+ g_assert_cmpint (nm_bridge_vlan_is_pvid (vlan), ==, TRUE);
+ g_assert_cmpint (nm_bridge_vlan_is_untagged (vlan), ==, TRUE);
+
+ vlan = nm_setting_bridge_get_vlan (s_bridge, 1);
+ g_assert (vlan);
+ g_assert_cmpuint (nm_bridge_vlan_get_vid (vlan), ==, 9);
+ g_assert_cmpint (nm_bridge_vlan_is_pvid (vlan), ==, FALSE);
+ g_assert_cmpint (nm_bridge_vlan_is_untagged (vlan), ==, TRUE);
+
+ CLEAR (&con, &keyfile);
+}
+
+static void
+test_bridge_port_vlans (void)
+{
+ gs_unref_keyfile GKeyFile *keyfile = NULL;
+ gs_unref_object NMConnection *con = NULL;
+ NMSettingBridgePort *s_port;
+ NMBridgeVlan *vlan;
+
+ con = nmtst_create_connection_from_keyfile (
+ "[connection]\n"
+ "id=t\n"
+ "type=dummy\n"
+ "interface-name=dummy1\n"
+ "master=br0\n"
+ "slave-type=bridge\n"
+ "\n"
+ "[bridge-port]\n"
+ "vlan.4000=\n"
+ "vlan.10=untagged\n"
+ "vlan.20=pvid untagged"
+ "",
+ "/test_bridge_port/vlans");
+ s_port = NM_SETTING_BRIDGE_PORT (nm_connection_get_setting (con, NM_TYPE_SETTING_BRIDGE_PORT));
+ g_assert (s_port);
+ g_assert_cmpuint (nm_setting_bridge_port_get_num_vlans (s_port), ==, 3);
+
+ vlan = nm_setting_bridge_port_get_vlan (s_port, 0);
+ g_assert (vlan);
+ g_assert_cmpuint (nm_bridge_vlan_get_vid (vlan), ==, 10);
+ g_assert_cmpint (nm_bridge_vlan_is_pvid (vlan), ==, FALSE);
+ g_assert_cmpint (nm_bridge_vlan_is_untagged (vlan), ==, TRUE);
+
+ vlan = nm_setting_bridge_port_get_vlan (s_port, 1);
+ g_assert (vlan);
+ g_assert_cmpuint (nm_bridge_vlan_get_vid (vlan), ==, 20);
+ g_assert_cmpint (nm_bridge_vlan_is_pvid (vlan), ==, TRUE);
+ g_assert_cmpint (nm_bridge_vlan_is_untagged (vlan), ==, TRUE);
+
+ vlan = nm_setting_bridge_port_get_vlan (s_port, 2);
+ g_assert (vlan);
+ g_assert_cmpuint (nm_bridge_vlan_get_vid (vlan), ==, 4000);
+ g_assert_cmpint (nm_bridge_vlan_is_pvid (vlan), ==, FALSE);
+ g_assert_cmpint (nm_bridge_vlan_is_untagged (vlan), ==, FALSE);
+
+ CLEAR (&con, &keyfile);
+}
+
+/*****************************************************************************/
+
NMTST_DEFINE ();
int main (int argc, char **argv)
@@ -757,6 +844,8 @@ int main (int argc, char **argv)
g_test_add_func ("/core/keyfile/test_team_conf_read/invalid", test_team_conf_read_invalid);
g_test_add_func ("/core/keyfile/test_user/1", test_user_1);
g_test_add_func ("/core/keyfile/test_vpn/1", test_vpn_1);
+ g_test_add_func ("/core/keyfile/bridge/vlans", test_bridge_vlans);
+ g_test_add_func ("/core/keyfile/bridge-port/vlans", test_bridge_port_vlans);
return g_test_run ();
}