keyfile: support bridge vlans
This commit is contained in:
@@ -1518,6 +1518,57 @@ team_config_parser (KeyfileReaderInfo *info, NMSetting *setting, const char *key
|
||||
g_object_set (G_OBJECT (setting), key, conf, NULL);
|
||||
}
|
||||
|
||||
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)
|
||||
{
|
||||
@@ -1842,6 +1893,34 @@ route_writer (KeyfileWriterInfo *info,
|
||||
write_ip_values (info->keyfile, setting_name, array, NULL, TRUE);
|
||||
}
|
||||
|
||||
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,
|
||||
@@ -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,
|
||||
|
@@ -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 ();
|
||||
}
|
||||
|
Reference in New Issue
Block a user