2012-08-10 11:55:33 +02:00
|
|
|
/*
|
|
|
|
* hostapd / VLAN netlink api
|
|
|
|
* Copyright (c) 2012, Michael Braun <michael-dev@fami-braun.de>
|
|
|
|
*
|
|
|
|
* This software may be distributed under the terms of the BSD license.
|
|
|
|
* See README for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "utils/includes.h"
|
|
|
|
#include <netlink/route/link.h>
|
|
|
|
#include <netlink/route/link/vlan.h>
|
|
|
|
|
|
|
|
#include "utils/common.h"
|
|
|
|
#include "vlan_util.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add a vlan interface with name 'vlan_if_name', VLAN ID 'vid' and
|
|
|
|
* tagged interface 'if_name'.
|
|
|
|
*
|
|
|
|
* returns -1 on error
|
|
|
|
* returns 1 if the interface already exists
|
|
|
|
* returns 0 otherwise
|
|
|
|
*/
|
|
|
|
int vlan_add(const char *if_name, int vid, const char *vlan_if_name)
|
|
|
|
{
|
2015-05-06 13:44:04 +02:00
|
|
|
int err, ret = -1;
|
2012-08-10 11:55:33 +02:00
|
|
|
struct nl_sock *handle = NULL;
|
|
|
|
struct rtnl_link *rlink = NULL;
|
|
|
|
int if_idx = 0;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "VLAN: vlan_add(if_name=%s, vid=%d, "
|
|
|
|
"vlan_if_name=%s)", if_name, vid, vlan_if_name);
|
|
|
|
|
|
|
|
if ((os_strlen(if_name) + 1) > IFNAMSIZ) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: Interface name too long: '%s'",
|
|
|
|
if_name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((os_strlen(vlan_if_name) + 1) > IFNAMSIZ) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: Interface name too long: '%s'",
|
|
|
|
vlan_if_name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
handle = nl_socket_alloc();
|
|
|
|
if (!handle) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to open netlink socket");
|
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = nl_connect(handle, NETLINK_ROUTE);
|
|
|
|
if (err < 0) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
|
|
|
|
nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
2016-02-24 12:53:50 +01:00
|
|
|
err = rtnl_link_get_kernel(handle, 0, if_name, &rlink);
|
2015-05-06 13:44:04 +02:00
|
|
|
if (err < 0) {
|
2012-08-10 11:55:33 +02:00
|
|
|
/* link does not exist */
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: interface %s does not exist",
|
|
|
|
if_name);
|
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
2016-02-24 12:53:50 +01:00
|
|
|
if_idx = rtnl_link_get_ifindex(rlink);
|
|
|
|
rtnl_link_put(rlink);
|
|
|
|
rlink = NULL;
|
2012-08-10 11:55:33 +02:00
|
|
|
|
2016-02-24 12:53:50 +01:00
|
|
|
err = rtnl_link_get_kernel(handle, 0, vlan_if_name, &rlink);
|
|
|
|
if (err >= 0) {
|
2012-08-10 11:55:33 +02:00
|
|
|
/* link does exist */
|
|
|
|
rtnl_link_put(rlink);
|
|
|
|
rlink = NULL;
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: interface %s already exists",
|
|
|
|
vlan_if_name);
|
|
|
|
ret = 1;
|
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
rlink = rtnl_link_alloc();
|
|
|
|
if (!rlink) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to allocate new link");
|
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = rtnl_link_set_type(rlink, "vlan");
|
|
|
|
if (err < 0) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to set link type: %s",
|
|
|
|
nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
rtnl_link_set_link(rlink, if_idx);
|
|
|
|
rtnl_link_set_name(rlink, vlan_if_name);
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = rtnl_link_vlan_set_id(rlink, vid);
|
|
|
|
if (err < 0) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to set link vlan id: %s",
|
|
|
|
nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = rtnl_link_add(handle, rlink, NLM_F_CREATE);
|
|
|
|
if (err < 0) {
|
2012-08-10 11:55:33 +02:00
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to create link %s for "
|
2015-05-06 13:44:04 +02:00
|
|
|
"vlan %d on %s (%d): %s",
|
|
|
|
vlan_if_name, vid, if_name, if_idx,
|
|
|
|
nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_add_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
vlan_add_error:
|
|
|
|
if (rlink)
|
|
|
|
rtnl_link_put(rlink);
|
|
|
|
if (handle)
|
|
|
|
nl_socket_free(handle);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int vlan_rem(const char *if_name)
|
|
|
|
{
|
2015-05-06 13:44:04 +02:00
|
|
|
int err, ret = -1;
|
2012-08-10 11:55:33 +02:00
|
|
|
struct nl_sock *handle = NULL;
|
|
|
|
struct rtnl_link *rlink = NULL;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "VLAN: vlan_rem(if_name=%s)", if_name);
|
|
|
|
|
|
|
|
handle = nl_socket_alloc();
|
|
|
|
if (!handle) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to open netlink socket");
|
|
|
|
goto vlan_rem_error;
|
|
|
|
}
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = nl_connect(handle, NETLINK_ROUTE);
|
|
|
|
if (err < 0) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
|
|
|
|
nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_rem_error;
|
|
|
|
}
|
|
|
|
|
2016-02-24 12:53:50 +01:00
|
|
|
err = rtnl_link_get_kernel(handle, 0, if_name, &rlink);
|
2015-05-06 13:44:04 +02:00
|
|
|
if (err < 0) {
|
2012-08-10 11:55:33 +02:00
|
|
|
/* link does not exist */
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: interface %s does not exists",
|
|
|
|
if_name);
|
|
|
|
goto vlan_rem_error;
|
|
|
|
}
|
|
|
|
|
2015-05-06 13:44:04 +02:00
|
|
|
err = rtnl_link_delete(handle, rlink);
|
|
|
|
if (err < 0) {
|
|
|
|
wpa_printf(MSG_ERROR, "VLAN: failed to remove link %s: %s",
|
|
|
|
if_name, nl_geterror(err));
|
2012-08-10 11:55:33 +02:00
|
|
|
goto vlan_rem_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
vlan_rem_error:
|
|
|
|
if (rlink)
|
|
|
|
rtnl_link_put(rlink);
|
|
|
|
if (handle)
|
|
|
|
nl_socket_free(handle);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-03-25 16:21:41 +01:00
|
|
|
|
|
|
|
|
|
|
|
int vlan_set_name_type(unsigned int name_type)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|