diff options
author | Beniamino Galvani <bgalvani@redhat.com> | 2019-03-19 17:33:40 +0100 |
---|---|---|
committer | Beniamino Galvani <bgalvani@redhat.com> | 2019-03-26 17:19:39 +0100 |
commit | c66d17d4fc3fd06e4e9853ff0929c3844fbed1b6 (patch) | |
tree | 8eda2eb034bacc40293e87f373bf9b42d0ba99df | |
parent | cf71860183dcabd47846ff7f07e0ae28ddab9123 (diff) | |
download | NetworkManager-c66d17d4fc3fd06e4e9853ff0929c3844fbed1b6.tar.gz |
keyfile: support bridge vlans
-rw-r--r-- | libnm-core/nm-keyfile.c | 93 | ||||
-rw-r--r-- | libnm-core/tests/test-keyfile.c | 89 |
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 (); } |