blob: 57c3f0a7a4e2790022a7811c7bb42c70bdcccbae [file] [log] [blame]
Johannes Berg2ef1be62008-04-02 17:40:11 +02001#include <linux/nl80211.h>
2#include <net/if.h>
Johannes Berg45c72122007-09-28 23:47:38 +02003#include <errno.h>
Johannes Bergd5ac8ad2008-04-03 15:24:13 +02004#include <string.h>
Johannes Berg2ef1be62008-04-02 17:40:11 +02005
Johannes Berg45c72122007-09-28 23:47:38 +02006#include <netlink/genl/genl.h>
7#include <netlink/genl/family.h>
8#include <netlink/genl/ctrl.h>
9#include <netlink/msg.h>
10#include <netlink/attr.h>
Johannes Berg45c72122007-09-28 23:47:38 +020011
12#include "iw.h"
13
14/* return 0 if not found, 1 if ok, -1 on error */
15static int get_if_type(int *argc, char ***argv, enum nl80211_iftype *type)
16{
17 char *tpstr;
18
19 if (*argc < 2)
20 return 0;
21
22 if (strcmp((*argv)[0], "type"))
23 return 0;
24
25 tpstr = (*argv)[1];
26 *argc -= 2;
27 *argv += 2;
28
29 if (strcmp(tpstr, "adhoc") == 0 ||
30 strcmp(tpstr, "ibss") == 0) {
31 *type = NL80211_IFTYPE_ADHOC;
32 return 1;
33 } else if (strcmp(tpstr, "monitor") == 0) {
34 *type = NL80211_IFTYPE_MONITOR;
35 return 1;
Johannes Berg4d3a72d2008-04-14 15:41:31 +020036 } else if (strcmp(tpstr, "__ap") == 0) {
Johannes Berg45c72122007-09-28 23:47:38 +020037 *type = NL80211_IFTYPE_AP;
38 return 1;
Johannes Berg4d3a72d2008-04-14 15:41:31 +020039 } else if (strcmp(tpstr, "__ap_vlan") == 0) {
Johannes Berg45c72122007-09-28 23:47:38 +020040 *type = NL80211_IFTYPE_AP_VLAN;
41 return 1;
42 } else if (strcmp(tpstr, "wds") == 0) {
43 *type = NL80211_IFTYPE_WDS;
44 return 1;
45 } else if (strcmp(tpstr, "station") == 0) {
46 *type = NL80211_IFTYPE_STATION;
47 return 1;
Luis Carlos Cobo3d1e8702008-04-01 12:03:41 -070048 } else if (strcmp(tpstr, "mp") == 0 ||
49 strcmp(tpstr, "mesh") == 0) {
50 *type = NL80211_IFTYPE_MESH_POINT;
51 return 1;
Johannes Berg45c72122007-09-28 23:47:38 +020052 }
53
54
55 fprintf(stderr, "invalid interface type %s\n", tpstr);
56 return -1;
57}
58
59static int handle_interface_add(struct nl80211_state *state,
Johannes Bergbd396f22008-09-16 16:56:09 +020060 struct nl_msg *msg,
61 int argc, char **argv)
Johannes Berg45c72122007-09-28 23:47:38 +020062{
Andrew Lutomirski2dfd6bf2007-12-20 18:50:07 +010063 char *name;
Luis Carlos Cobo3d1e8702008-04-01 12:03:41 -070064 char *mesh_id = NULL;
Johannes Berg45c72122007-09-28 23:47:38 +020065 enum nl80211_iftype type;
Johannes Berg23dfe292008-04-03 15:39:27 +020066 int tpset, err = -ENOBUFS;
Johannes Berg45c72122007-09-28 23:47:38 +020067
Johannes Bergbd396f22008-09-16 16:56:09 +020068 if (argc < 1)
Johannes Berg45c72122007-09-28 23:47:38 +020069 return -1;
Johannes Berg45c72122007-09-28 23:47:38 +020070
Andrew Lutomirski2dfd6bf2007-12-20 18:50:07 +010071 name = argv[0];
Johannes Berg45c72122007-09-28 23:47:38 +020072 argc--;
73 argv++;
74
Andrew Lutomirski2dfd6bf2007-12-20 18:50:07 +010075 tpset = get_if_type(&argc, &argv, &type);
76 if (tpset == 0)
77 fprintf(stderr, "you must specify an interface type\n");
78 if (tpset <= 0)
79 return -1;
Johannes Berg45c72122007-09-28 23:47:38 +020080
81 if (argc) {
Luis Carlos Cobo3d1e8702008-04-01 12:03:41 -070082 if (strcmp(argv[0], "mesh_id") != 0) {
83 fprintf(stderr, "option %s not supported\n", argv[0]);
84 return -1;
85 }
86 argc--;
87 argv++;
88
89 if (!argc) {
90 fprintf(stderr, "not enough arguments\n");
91 return -1;
92 }
93 mesh_id = argv[0];
94 argc--;
95 argv++;
96 }
97
98 if (argc) {
Johannes Berg45c72122007-09-28 23:47:38 +020099 fprintf(stderr, "too many arguments\n");
100 return -1;
101 }
102
Johannes Berg45c72122007-09-28 23:47:38 +0200103 NLA_PUT_STRING(msg, NL80211_ATTR_IFNAME, name);
104 if (tpset)
105 NLA_PUT_U32(msg, NL80211_ATTR_IFTYPE, type);
Luis Carlos Cobo3d1e8702008-04-01 12:03:41 -0700106 if (mesh_id)
107 NLA_PUT(msg, NL80211_ATTR_MESH_ID, strlen(mesh_id), mesh_id);
Johannes Berg45c72122007-09-28 23:47:38 +0200108
109 if ((err = nl_send_auto_complete(state->nl_handle, msg)) < 0 ||
110 (err = nl_wait_for_ack(state->nl_handle)) < 0) {
111 nla_put_failure:
112 fprintf(stderr, "failed to create interface: %d\n", err);
Johannes Bergbd396f22008-09-16 16:56:09 +0200113 return 1;
Johannes Berg45c72122007-09-28 23:47:38 +0200114 }
115
Johannes Berg45c72122007-09-28 23:47:38 +0200116 return 0;
117}
Johannes Bergbd396f22008-09-16 16:56:09 +0200118COMMAND(interface, add, "<name> type <type> [mesh_id <meshid>]",
119 NL80211_CMD_NEW_INTERFACE, 0, CIB_PHY, handle_interface_add);
Johannes Berg1cd3b6c2008-09-16 17:52:56 +0200120COMMAND(interface, add, "<name> type <type> [mesh_id <meshid>]",
121 NL80211_CMD_NEW_INTERFACE, 0, CIB_NETDEV, handle_interface_add);
Johannes Berg45c72122007-09-28 23:47:38 +0200122
Johannes Berg3fcfe402007-09-28 23:50:25 +0200123static int handle_interface_del(struct nl80211_state *state,
Johannes Bergbd396f22008-09-16 16:56:09 +0200124 struct nl_msg *msg,
125 int argc, char **argv)
Johannes Berg3fcfe402007-09-28 23:50:25 +0200126{
Johannes Berg23dfe292008-04-03 15:39:27 +0200127 int err = -ENOBUFS;
Johannes Berg3fcfe402007-09-28 23:50:25 +0200128
129 if ((err = nl_send_auto_complete(state->nl_handle, msg)) < 0 ||
130 (err = nl_wait_for_ack(state->nl_handle)) < 0) {
Johannes Berg3fcfe402007-09-28 23:50:25 +0200131 fprintf(stderr, "failed to remove interface: %d\n", err);
132 nlmsg_free(msg);
Johannes Bergbd396f22008-09-16 16:56:09 +0200133 return 1;
Johannes Berg3fcfe402007-09-28 23:50:25 +0200134 }
135
Johannes Berg3fcfe402007-09-28 23:50:25 +0200136 return 0;
137}
Johannes Bergbd396f22008-09-16 16:56:09 +0200138TOPLEVEL(del, NULL, NL80211_CMD_DEL_INTERFACE, 0, CIB_NETDEV, handle_interface_del);
Johannes Berg3fcfe402007-09-28 23:50:25 +0200139
Johannes Berg541ef422008-09-16 14:50:11 +0200140static int print_iface_handler(struct nl_msg *msg, void *arg)
141{
142 struct genlmsghdr *gnlh = nlmsg_data(nlmsg_hdr(msg));
143 struct nlattr *tb_msg[NL80211_ATTR_MAX + 1];
144
145 nla_parse(tb_msg, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
146 genlmsg_attrlen(gnlh, 0), NULL);
147
148 if (tb_msg[NL80211_ATTR_IFNAME])
149 printf("Interface %s\n", nla_get_string(tb_msg[NL80211_ATTR_IFNAME]));
150 if (tb_msg[NL80211_ATTR_IFINDEX])
151 printf("\tifindex %d\n", nla_get_u32(tb_msg[NL80211_ATTR_IFINDEX]));
152 if (tb_msg[NL80211_ATTR_IFTYPE])
153 printf("\ttype %s\n", iftype_name(nla_get_u32(tb_msg[NL80211_ATTR_IFTYPE])));
154
155 return NL_SKIP;
156}
157
158static int ack_wait_handler(struct nl_msg *msg, void *arg)
159{
160 int *finished = arg;
161
162 *finished = 1;
163 return NL_STOP;
164}
165
166static int handle_interface_info(struct nl80211_state *state,
Johannes Bergbd396f22008-09-16 16:56:09 +0200167 struct nl_msg *msg,
168 int argc, char **argv)
Johannes Berg541ef422008-09-16 14:50:11 +0200169{
170 int err = -ENOBUFS;
Johannes Berg541ef422008-09-16 14:50:11 +0200171 struct nl_cb *cb = NULL;
172 int finished = 0;
173
Johannes Berg541ef422008-09-16 14:50:11 +0200174 cb = nl_cb_alloc(NL_CB_CUSTOM);
175 if (!cb)
176 goto out;
177
178 if (nl_send_auto_complete(state->nl_handle, msg) < 0)
179 goto out;
180
181 nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, print_iface_handler, NULL);
182 nl_cb_set(cb, NL_CB_ACK, NL_CB_CUSTOM, ack_wait_handler, &finished);
183
184 err = nl_recvmsgs(state->nl_handle, cb);
185
186 if (!finished)
187 err = nl_wait_for_ack(state->nl_handle);
188
189 if (err)
190 fprintf(stderr, "failed to get information: %d\n", err);
191
192 out:
Johannes Berg541ef422008-09-16 14:50:11 +0200193 nl_cb_put(cb);
194 return 0;
195}
Johannes Bergbd396f22008-09-16 16:56:09 +0200196TOPLEVEL(info, NULL, NL80211_CMD_GET_INTERFACE, 0, CIB_NETDEV, handle_interface_info);