Merged the hostap_2.6 updates, and the Leap of Faith work, from the hostap_update...
[mech_eap.git] / libeap / src / ap / vlan_util.c
diff --git a/libeap/src/ap/vlan_util.c b/libeap/src/ap/vlan_util.c
new file mode 100644 (file)
index 0000000..56d1d3d
--- /dev/null
@@ -0,0 +1,174 @@
+/*
+ * 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)
+{
+       int err, ret = -1;
+       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;
+       }
+
+       err = nl_connect(handle, NETLINK_ROUTE);
+       if (err < 0) {
+               wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
+                          nl_geterror(err));
+               goto vlan_add_error;
+       }
+
+       err = rtnl_link_get_kernel(handle, 0, if_name, &rlink);
+       if (err < 0) {
+               /* link does not exist */
+               wpa_printf(MSG_ERROR, "VLAN: interface %s does not exist",
+                          if_name);
+               goto vlan_add_error;
+       }
+       if_idx = rtnl_link_get_ifindex(rlink);
+       rtnl_link_put(rlink);
+       rlink = NULL;
+
+       err = rtnl_link_get_kernel(handle, 0, vlan_if_name, &rlink);
+       if (err >= 0) {
+               /* 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;
+       }
+
+       err = rtnl_link_set_type(rlink, "vlan");
+       if (err < 0) {
+               wpa_printf(MSG_ERROR, "VLAN: failed to set link type: %s",
+                          nl_geterror(err));
+               goto vlan_add_error;
+       }
+
+       rtnl_link_set_link(rlink, if_idx);
+       rtnl_link_set_name(rlink, vlan_if_name);
+
+       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));
+               goto vlan_add_error;
+       }
+
+       err = rtnl_link_add(handle, rlink, NLM_F_CREATE);
+       if (err < 0) {
+               wpa_printf(MSG_ERROR, "VLAN: failed to create link %s for "
+                          "vlan %d on %s (%d): %s",
+                          vlan_if_name, vid, if_name, if_idx,
+                          nl_geterror(err));
+               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)
+{
+       int err, ret = -1;
+       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;
+       }
+
+       err = nl_connect(handle, NETLINK_ROUTE);
+       if (err < 0) {
+               wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
+                          nl_geterror(err));
+               goto vlan_rem_error;
+       }
+
+       err = rtnl_link_get_kernel(handle, 0, if_name, &rlink);
+       if (err < 0) {
+               /* link does not exist */
+               wpa_printf(MSG_ERROR, "VLAN: interface %s does not exists",
+                          if_name);
+               goto vlan_rem_error;
+       }
+
+       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));
+               goto vlan_rem_error;
+       }
+
+       ret = 0;
+
+vlan_rem_error:
+       if (rlink)
+               rtnl_link_put(rlink);
+       if (handle)
+               nl_socket_free(handle);
+       return ret;
+}
+
+
+int vlan_set_name_type(unsigned int name_type)
+{
+       return 0;
+}