]> git.ipfire.org Git - thirdparty/iw.git/blame - link.c
add a few new commands/events
[thirdparty/iw.git] / link.c
CommitLineData
febeb0c0
JB
1#include <net/if.h>
2#include <errno.h>
3#include <string.h>
4#include <ctype.h>
5#include <stdbool.h>
6
7#include <netlink/genl/genl.h>
8#include <netlink/genl/family.h>
9#include <netlink/genl/ctrl.h>
10#include <netlink/msg.h>
11#include <netlink/attr.h>
12
13#include "nl80211.h"
14#include "iw.h"
15
16struct link_result {
17 uint8_t bssid[8];
18 bool link_found;
19 bool anything_found;
20};
21
22static struct link_result lr = { .link_found = false };
23
24static int link_bss_handler(struct nl_msg *msg, void *arg)
25{
26 struct nlattr *tb[NL80211_ATTR_MAX + 1];
27 struct genlmsghdr *gnlh = nlmsg_data(nlmsg_hdr(msg));
28 struct nlattr *bss[NL80211_BSS_MAX + 1];
29 static struct nla_policy bss_policy[NL80211_BSS_MAX + 1] = {
30 [NL80211_BSS_TSF] = { .type = NLA_U64 },
31 [NL80211_BSS_FREQUENCY] = { .type = NLA_U32 },
32 [NL80211_BSS_BSSID] = { },
33 [NL80211_BSS_BEACON_INTERVAL] = { .type = NLA_U16 },
34 [NL80211_BSS_CAPABILITY] = { .type = NLA_U16 },
35 [NL80211_BSS_INFORMATION_ELEMENTS] = { },
36 [NL80211_BSS_SIGNAL_MBM] = { .type = NLA_U32 },
37 [NL80211_BSS_SIGNAL_UNSPEC] = { .type = NLA_U8 },
38 [NL80211_BSS_STATUS] = { .type = NLA_U32 },
39 };
40 struct link_result *result = arg;
41 char mac_addr[20], dev[20];
42
43 nla_parse(tb, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
44 genlmsg_attrlen(gnlh, 0), NULL);
45
46 if (!tb[NL80211_ATTR_BSS]) {
5fe70c0e 47 fprintf(stderr, "bss info missing!\n");
febeb0c0
JB
48 return NL_SKIP;
49 }
50 if (nla_parse_nested(bss, NL80211_BSS_MAX,
51 tb[NL80211_ATTR_BSS],
52 bss_policy)) {
5fe70c0e 53 fprintf(stderr, "failed to parse nested attributes!\n");
febeb0c0
JB
54 return NL_SKIP;
55 }
56
57 if (!bss[NL80211_BSS_BSSID])
58 return NL_SKIP;
59
60 if (!bss[NL80211_BSS_STATUS])
61 return NL_SKIP;
62
63 mac_addr_n2a(mac_addr, nla_data(bss[NL80211_BSS_BSSID]));
64 if_indextoname(nla_get_u32(tb[NL80211_ATTR_IFINDEX]), dev);
65
66 switch (nla_get_u32(bss[NL80211_BSS_STATUS])) {
67 case NL80211_BSS_STATUS_ASSOCIATED:
68 printf("Connected to %s (on %s)\n", mac_addr, dev);
69 break;
70 case NL80211_BSS_STATUS_AUTHENTICATED:
71 printf("Authenticated with %s (on %s)\n", mac_addr, dev);
72 return NL_SKIP;
28b1efb7
JB
73 case NL80211_BSS_STATUS_IBSS_JOINED:
74 printf("Joined IBSS %s (on %s)\n", mac_addr, dev);
75 break;
febeb0c0
JB
76 default:
77 return NL_SKIP;
78 }
79
80 result->anything_found = true;
81
82 if (bss[NL80211_BSS_INFORMATION_ELEMENTS])
83 print_ies(nla_data(bss[NL80211_BSS_INFORMATION_ELEMENTS]),
84 nla_len(bss[NL80211_BSS_INFORMATION_ELEMENTS]),
85 false, PRINT_LINK);
86
87 if (bss[NL80211_BSS_FREQUENCY])
88 printf("\tfreq: %d\n",
89 nla_get_u32(bss[NL80211_BSS_FREQUENCY]));
90
91 if (nla_get_u32(bss[NL80211_BSS_STATUS]) != NL80211_BSS_STATUS_ASSOCIATED)
92 return NL_SKIP;
93
94 /* only in the assoc case do we want more info from station get */
95 result->link_found = true;
96 memcpy(result->bssid, nla_data(bss[NL80211_BSS_BSSID]), 6);
97 return NL_SKIP;
98}
99
100static int handle_scan_for_link(struct nl80211_state *state,
febeb0c0 101 struct nl_msg *msg,
05514f95
JB
102 int argc, char **argv,
103 enum id_input id)
febeb0c0
JB
104{
105 if (argc > 0)
106 return 1;
107
34b23014 108 register_handler(link_bss_handler, &lr);
febeb0c0
JB
109 return 0;
110}
febeb0c0
JB
111
112static int print_link_sta(struct nl_msg *msg, void *arg)
113{
114 struct nlattr *tb[NL80211_ATTR_MAX + 1];
115 struct genlmsghdr *gnlh = nlmsg_data(nlmsg_hdr(msg));
116 struct nlattr *sinfo[NL80211_STA_INFO_MAX + 1];
dd8eef95 117 struct nlattr *binfo[NL80211_STA_BSS_PARAM_MAX + 1];
febeb0c0
JB
118 static struct nla_policy stats_policy[NL80211_STA_INFO_MAX + 1] = {
119 [NL80211_STA_INFO_INACTIVE_TIME] = { .type = NLA_U32 },
120 [NL80211_STA_INFO_RX_BYTES] = { .type = NLA_U32 },
121 [NL80211_STA_INFO_TX_BYTES] = { .type = NLA_U32 },
122 [NL80211_STA_INFO_RX_PACKETS] = { .type = NLA_U32 },
123 [NL80211_STA_INFO_TX_PACKETS] = { .type = NLA_U32 },
124 [NL80211_STA_INFO_SIGNAL] = { .type = NLA_U8 },
125 [NL80211_STA_INFO_TX_BITRATE] = { .type = NLA_NESTED },
126 [NL80211_STA_INFO_LLID] = { .type = NLA_U16 },
127 [NL80211_STA_INFO_PLID] = { .type = NLA_U16 },
128 [NL80211_STA_INFO_PLINK_STATE] = { .type = NLA_U8 },
129 };
dd8eef95
PS
130 static struct nla_policy bss_policy[NL80211_STA_BSS_PARAM_MAX + 1] = {
131 [NL80211_STA_BSS_PARAM_CTS_PROT] = { .type = NLA_FLAG },
132 [NL80211_STA_BSS_PARAM_SHORT_PREAMBLE] = { .type = NLA_FLAG },
133 [NL80211_STA_BSS_PARAM_SHORT_SLOT_TIME] = { .type = NLA_FLAG },
134 [NL80211_STA_BSS_PARAM_DTIM_PERIOD] = { .type = NLA_U8 },
135 [NL80211_STA_BSS_PARAM_BEACON_INTERVAL] = { .type = NLA_U16 },
136 };
febeb0c0
JB
137
138 nla_parse(tb, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
139 genlmsg_attrlen(gnlh, 0), NULL);
140
141 if (!tb[NL80211_ATTR_STA_INFO]) {
5fe70c0e 142 fprintf(stderr, "sta stats missing!\n");
febeb0c0
JB
143 return NL_SKIP;
144 }
145 if (nla_parse_nested(sinfo, NL80211_STA_INFO_MAX,
146 tb[NL80211_ATTR_STA_INFO],
147 stats_policy)) {
5fe70c0e 148 fprintf(stderr, "failed to parse nested attributes!\n");
febeb0c0
JB
149 return NL_SKIP;
150 }
151
152 if (sinfo[NL80211_STA_INFO_RX_BYTES] && sinfo[NL80211_STA_INFO_RX_PACKETS])
26181415 153 printf("\tRX: %u bytes (%u packets)\n",
febeb0c0
JB
154 nla_get_u32(sinfo[NL80211_STA_INFO_RX_BYTES]),
155 nla_get_u32(sinfo[NL80211_STA_INFO_RX_PACKETS]));
156 if (sinfo[NL80211_STA_INFO_TX_BYTES] && sinfo[NL80211_STA_INFO_TX_PACKETS])
26181415 157 printf("\tTX: %u bytes (%u packets)\n",
febeb0c0
JB
158 nla_get_u32(sinfo[NL80211_STA_INFO_TX_BYTES]),
159 nla_get_u32(sinfo[NL80211_STA_INFO_TX_PACKETS]));
160 if (sinfo[NL80211_STA_INFO_SIGNAL])
161 printf("\tsignal: %d dBm\n",
162 (int8_t)nla_get_u8(sinfo[NL80211_STA_INFO_SIGNAL]));
163
164 if (sinfo[NL80211_STA_INFO_TX_BITRATE]) {
64179590
JB
165 char buf[100];
166
e94f0201 167 parse_bitrate(sinfo[NL80211_STA_INFO_TX_BITRATE], buf, sizeof(buf));
64179590 168 printf("\ttx bitrate: %s\n", buf);
febeb0c0
JB
169 }
170
dd8eef95
PS
171 if (sinfo[NL80211_STA_INFO_BSS_PARAM]) {
172 if (nla_parse_nested(binfo, NL80211_STA_BSS_PARAM_MAX,
173 sinfo[NL80211_STA_INFO_BSS_PARAM],
174 bss_policy)) {
175 fprintf(stderr, "failed to parse nested bss parameters!\n");
176 } else {
177 char *delim = "";
178 printf("\n\tbss flags:\t");
179 if (binfo[NL80211_STA_BSS_PARAM_CTS_PROT]) {
180 printf("CTS-protection");
181 delim = " ";
182 }
183 if (binfo[NL80211_STA_BSS_PARAM_SHORT_PREAMBLE]) {
184 printf("%sshort-preamble", delim);
185 delim = " ";
186 }
187 if (binfo[NL80211_STA_BSS_PARAM_SHORT_SLOT_TIME])
188 printf("%sshort-slot-time", delim);
189 printf("\n\tdtim period:\t%d",
190 nla_get_u8(binfo[NL80211_STA_BSS_PARAM_DTIM_PERIOD]));
191 printf("\n\tbeacon int:\t%d",
192 nla_get_u16(binfo[NL80211_STA_BSS_PARAM_BEACON_INTERVAL]));
193 printf("\n");
194 }
195 }
196
febeb0c0
JB
197 return NL_SKIP;
198}
199
200static int handle_link_sta(struct nl80211_state *state,
febeb0c0 201 struct nl_msg *msg,
05514f95
JB
202 int argc, char **argv,
203 enum id_input id)
febeb0c0
JB
204{
205 unsigned char mac_addr[ETH_ALEN];
206
207 if (argc < 1)
208 return 1;
209
210 if (mac_addr_a2n(mac_addr, argv[0])) {
211 fprintf(stderr, "invalid mac address\n");
212 return 2;
213 }
214
215 argc--;
216 argv++;
217
218 if (argc)
219 return 1;
220
221 NLA_PUT(msg, NL80211_ATTR_MAC, ETH_ALEN, mac_addr);
222
34b23014 223 register_handler(print_link_sta, NULL);
febeb0c0
JB
224
225 return 0;
226 nla_put_failure:
227 return -ENOBUFS;
228}
febeb0c0 229
34b23014 230static int handle_link(struct nl80211_state *state,
05514f95
JB
231 struct nl_msg *msg, int argc, char **argv,
232 enum id_input id)
febeb0c0
JB
233{
234 char *link_argv[] = {
235 NULL,
236 "link",
237 "get_bss",
238 NULL,
239 };
240 char *station_argv[] = {
241 NULL,
242 "link",
243 "get_sta",
244 NULL,
245 NULL,
246 };
247 char bssid_buf[3*6];
248 int err;
249
250 link_argv[0] = argv[0];
75f4204c 251 err = handle_cmd(state, id, 3, link_argv);
febeb0c0
JB
252 if (err)
253 return err;
254
255 if (!lr.link_found) {
256 if (!lr.anything_found)
bb68dbd4 257 printf("Not connected.\n");
febeb0c0
JB
258 return 0;
259 }
260
261 mac_addr_n2a(bssid_buf, lr.bssid);
262 bssid_buf[17] = '\0';
263
264 station_argv[0] = argv[0];
265 station_argv[3] = bssid_buf;
75f4204c 266 return handle_cmd(state, id, 4, station_argv);
febeb0c0
JB
267}
268TOPLEVEL(link, NULL, 0, 0, CIB_NETDEV, handle_link,
269 "Print information about the current link, if any.");
4698bfc2
JB
270HIDDEN(link, get_sta, "", NL80211_CMD_GET_STATION, 0,
271 CIB_NETDEV, handle_link_sta);
272HIDDEN(link, get_bss, NULL, NL80211_CMD_GET_SCAN, NLM_F_DUMP,
273 CIB_NETDEV, handle_scan_for_link);