blob: d4e0efb9b02420232d861ea753afb959c1b51529 [file] [log] [blame]
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001/*
2 * hostapd / VLAN netlink api
3 * Copyright (c) 2012, Michael Braun <michael-dev@fami-braun.de>
4 *
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
7 */
8
9#include "utils/includes.h"
10#include <sys/ioctl.h>
11#include <linux/sockios.h>
12#include <linux/if_vlan.h>
13#include <netlink/genl/genl.h>
14#include <netlink/genl/family.h>
15#include <netlink/genl/ctrl.h>
16#include <netlink/route/link.h>
17#include <netlink/route/link/vlan.h>
18
19#include "utils/common.h"
20#include "utils/eloop.h"
21#include "hostapd.h"
22#include "vlan_util.h"
23
24/*
25 * Add a vlan interface with name 'vlan_if_name', VLAN ID 'vid' and
26 * tagged interface 'if_name'.
27 *
28 * returns -1 on error
29 * returns 1 if the interface already exists
30 * returns 0 otherwise
31*/
32int vlan_add(const char *if_name, int vid, const char *vlan_if_name)
33{
Dmitry Shmidt05df46a2015-05-19 11:02:01 -070034 int err, ret = -1;
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070035 struct nl_sock *handle = NULL;
36 struct nl_cache *cache = NULL;
37 struct rtnl_link *rlink = NULL;
38 int if_idx = 0;
39
40 wpa_printf(MSG_DEBUG, "VLAN: vlan_add(if_name=%s, vid=%d, "
41 "vlan_if_name=%s)", if_name, vid, vlan_if_name);
42
43 if ((os_strlen(if_name) + 1) > IFNAMSIZ) {
44 wpa_printf(MSG_ERROR, "VLAN: Interface name too long: '%s'",
45 if_name);
46 return -1;
47 }
48
49 if ((os_strlen(vlan_if_name) + 1) > IFNAMSIZ) {
50 wpa_printf(MSG_ERROR, "VLAN: Interface name too long: '%s'",
51 vlan_if_name);
52 return -1;
53 }
54
55 handle = nl_socket_alloc();
56 if (!handle) {
57 wpa_printf(MSG_ERROR, "VLAN: failed to open netlink socket");
58 goto vlan_add_error;
59 }
60
Dmitry Shmidt05df46a2015-05-19 11:02:01 -070061 err = nl_connect(handle, NETLINK_ROUTE);
62 if (err < 0) {
63 wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
64 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070065 goto vlan_add_error;
66 }
67
Dmitry Shmidt05df46a2015-05-19 11:02:01 -070068 err = rtnl_link_alloc_cache(handle, AF_UNSPEC, &cache);
69 if (err < 0) {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070070 cache = NULL;
Dmitry Shmidt05df46a2015-05-19 11:02:01 -070071 wpa_printf(MSG_ERROR, "VLAN: failed to alloc cache: %s",
72 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070073 goto vlan_add_error;
74 }
75
76 if (!(if_idx = rtnl_link_name2i(cache, if_name))) {
77 /* link does not exist */
78 wpa_printf(MSG_ERROR, "VLAN: interface %s does not exist",
79 if_name);
80 goto vlan_add_error;
81 }
82
83 if ((rlink = rtnl_link_get_by_name(cache, vlan_if_name))) {
84 /* link does exist */
85 rtnl_link_put(rlink);
86 rlink = NULL;
87 wpa_printf(MSG_ERROR, "VLAN: interface %s already exists",
88 vlan_if_name);
89 ret = 1;
90 goto vlan_add_error;
91 }
92
93 rlink = rtnl_link_alloc();
94 if (!rlink) {
95 wpa_printf(MSG_ERROR, "VLAN: failed to allocate new link");
96 goto vlan_add_error;
97 }
98
Dmitry Shmidt05df46a2015-05-19 11:02:01 -070099 err = rtnl_link_set_type(rlink, "vlan");
100 if (err < 0) {
101 wpa_printf(MSG_ERROR, "VLAN: failed to set link type: %s",
102 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700103 goto vlan_add_error;
104 }
105
106 rtnl_link_set_link(rlink, if_idx);
107 rtnl_link_set_name(rlink, vlan_if_name);
108
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700109 err = rtnl_link_vlan_set_id(rlink, vid);
110 if (err < 0) {
111 wpa_printf(MSG_ERROR, "VLAN: failed to set link vlan id: %s",
112 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700113 goto vlan_add_error;
114 }
115
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700116 err = rtnl_link_add(handle, rlink, NLM_F_CREATE);
117 if (err < 0) {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700118 wpa_printf(MSG_ERROR, "VLAN: failed to create link %s for "
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700119 "vlan %d on %s (%d): %s",
120 vlan_if_name, vid, if_name, if_idx,
121 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700122 goto vlan_add_error;
123 }
124
125 ret = 0;
126
127vlan_add_error:
128 if (rlink)
129 rtnl_link_put(rlink);
130 if (cache)
131 nl_cache_free(cache);
132 if (handle)
133 nl_socket_free(handle);
134 return ret;
135}
136
137
138int vlan_rem(const char *if_name)
139{
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700140 int err, ret = -1;
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700141 struct nl_sock *handle = NULL;
142 struct nl_cache *cache = NULL;
143 struct rtnl_link *rlink = NULL;
144
145 wpa_printf(MSG_DEBUG, "VLAN: vlan_rem(if_name=%s)", if_name);
146
147 handle = nl_socket_alloc();
148 if (!handle) {
149 wpa_printf(MSG_ERROR, "VLAN: failed to open netlink socket");
150 goto vlan_rem_error;
151 }
152
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700153 err = nl_connect(handle, NETLINK_ROUTE);
154 if (err < 0) {
155 wpa_printf(MSG_ERROR, "VLAN: failed to connect to netlink: %s",
156 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700157 goto vlan_rem_error;
158 }
159
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700160 err = rtnl_link_alloc_cache(handle, AF_UNSPEC, &cache);
161 if (err < 0) {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700162 cache = NULL;
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700163 wpa_printf(MSG_ERROR, "VLAN: failed to alloc cache: %s",
164 nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700165 goto vlan_rem_error;
166 }
167
168 if (!(rlink = rtnl_link_get_by_name(cache, if_name))) {
169 /* link does not exist */
170 wpa_printf(MSG_ERROR, "VLAN: interface %s does not exists",
171 if_name);
172 goto vlan_rem_error;
173 }
174
Dmitry Shmidt05df46a2015-05-19 11:02:01 -0700175 err = rtnl_link_delete(handle, rlink);
176 if (err < 0) {
177 wpa_printf(MSG_ERROR, "VLAN: failed to remove link %s: %s",
178 if_name, nl_geterror(err));
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700179 goto vlan_rem_error;
180 }
181
182 ret = 0;
183
184vlan_rem_error:
185 if (rlink)
186 rtnl_link_put(rlink);
187 if (cache)
188 nl_cache_free(cache);
189 if (handle)
190 nl_socket_free(handle);
191 return ret;
192}