1 /*
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 */
vlan_add(const char * if_name,int vid,const char * vlan_if_name)32 int vlan_add(const char *if_name, int vid, const char *vlan_if_name)
33 {
34 	int err, ret = -1;
35 	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 
61 	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));
65 		goto vlan_add_error;
66 	}
67 
68 	err = rtnl_link_alloc_cache(handle, AF_UNSPEC, &cache);
69 	if (err < 0) {
70 		cache = NULL;
71 		wpa_printf(MSG_ERROR, "VLAN: failed to alloc cache: %s",
72 			   nl_geterror(err));
73 		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 
99 	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));
103 		goto vlan_add_error;
104 	}
105 
106 	rtnl_link_set_link(rlink, if_idx);
107 	rtnl_link_set_name(rlink, vlan_if_name);
108 
109 	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));
113 		goto vlan_add_error;
114 	}
115 
116 	err = rtnl_link_add(handle, rlink, NLM_F_CREATE);
117 	if (err < 0) {
118 		wpa_printf(MSG_ERROR, "VLAN: failed to create link %s for "
119 			   "vlan %d on %s (%d): %s",
120 			   vlan_if_name, vid, if_name, if_idx,
121 			   nl_geterror(err));
122 		goto vlan_add_error;
123 	}
124 
125 	ret = 0;
126 
127 vlan_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 
vlan_rem(const char * if_name)138 int vlan_rem(const char *if_name)
139 {
140 	int err, ret = -1;
141 	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 
153 	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));
157 		goto vlan_rem_error;
158 	}
159 
160 	err = rtnl_link_alloc_cache(handle, AF_UNSPEC, &cache);
161 	if (err < 0) {
162 		cache = NULL;
163 		wpa_printf(MSG_ERROR, "VLAN: failed to alloc cache: %s",
164 			   nl_geterror(err));
165 		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 
175 	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));
179 		goto vlan_rem_error;
180 	}
181 
182 	ret = 0;
183 
184 vlan_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 }
193