/* * * Wireless daemon for Linux * * Copyright (C) 2020 Intel Corporation. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <stdio.h> #include <stdlib.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/ioctl.h> #include <net/if.h> #include <netinet/in.h> #include <unistd.h> #include <ell/ell.h> #include "linux/nl80211.h" #include "src/mpdu.h" #include "src/nl80211util.h" static struct l_genl *genl; static struct l_genl_family *nl80211; static int exit_status; static uint64_t wdev_id; static uint8_t wdev_addr[6]; static uint32_t freq; static const uint8_t probe_req_body[] = { /* SSID */ 0x00, 0x07, 'D', 'I', 'R', 'E', 'C', 'T', '-', /* Supported Rates */ 0x01, 0x08, 0x0c, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c, /* DS Parameter Set */ 0x03, 0x01, 0x00, /* HT Capabilities */ 0x2d, 0x1a, 0xef, 0x11, 0x17, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x2c, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* WPS */ 0xdd, 0x6c, 0x00, 0x50, 0xf2, 0x04, /* > Version */ 0x10, 0x4a, 0x00, 0x01, 0x10, /* > Request Type */ 0x10, 0x3a, 0x00, 0x01, 0x00, /* > Config Methods */ 0x10, 0x08, 0x00, 0x02, 0x13, 0x80, /* > UUID E */ 0x10, 0x47, 0x00, 0x10, 0x46, 0x92, 0x49, 0x6f, 0xce, 0x1e, 0x5f, 0xd1, 0xa5, 0x45, 0x9b, 0x1c, 0xa5, 0xde, 0xb9, 0x41, /* > Primary Device Type */ 0x10, 0x54, 0x00, 0x08, 0x00, 0x01, 0x00, 0x50, 0xf2, 0x04, 0x00, 0x01, /* > RF Bands */ 0x10, 0x3c, 0x00, 0x01, 0x01, /* > Association State */ 0x10, 0x02, 0x00, 0x02, 0x00, 0x00, /* > Configuration Error */ 0x10, 0x09, 0x00, 0x02, 0x00, 0x00, /* > Device Password ID */ 0x10, 0x12, 0x00, 0x02, 0x00, 0x00, /* > Manufacturer */ 0x10, 0x21, 0x00, 0x01, 0x20, /* > Model Name */ 0x10, 0x23, 0x00, 0x01, 0x20, /* > Model Numbers */ 0x10, 0x24, 0x00, 0x01, 0x20, /* > Device Name */ 0x10, 0x11, 0x00, 0x04, 't', 'e', 's', 't', /* > Vendor Extension > Version2 */ 0x10, 0x49, 0x00, 0x06, 0x00, 0x37, 0x2a, 0x00, 0x01, 0x20, /* P2P */ 0xdd, 0x11, 0x50, 0x6f, 0x9a, 0x09, /* > P2P Capability */ 0x02, 0x02, 0x00, 0x04, 0x00, /* > Listen Channel */ 0x06, 0x05, 0x00, 'X', 'X', 0x04, 0x51, 0x01, }; static void frame_cb(struct l_genl_msg *msg, void *user_data) { int err = l_genl_msg_get_error(msg); if (err < 0) { l_error("CMD_FRAME failed: %s (%i)", strerror(-err), -err); exit_status = EXIT_FAILURE; } else { l_info("Frame queued"); exit_status = EXIT_SUCCESS; } l_main_quit(); } static void get_interface_callback(struct l_genl_msg *msg, void *user_data) { uint32_t ifindex; uint32_t iftype; const char *ifname; const uint8_t *ifaddr; uint64_t cur_wdev_id; struct ifreq ifr; int sock; int r; /* * For now hoose the first interface with iftype station, require it * to be UP and have an ifindex. */ if (wdev_id) return; if (nl80211_parse_attrs(msg, NL80211_ATTR_IFINDEX, &ifindex, NL80211_ATTR_WDEV, &cur_wdev_id, NL80211_ATTR_IFTYPE, &iftype, NL80211_ATTR_IFNAME, &ifname, NL80211_ATTR_MAC, &ifaddr, NL80211_ATTR_UNSPEC) < 0) return; if (iftype != NL80211_IFTYPE_STATION) return; sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP); if (sock == -1) return; memset(&ifr, 0, sizeof(ifr)); l_strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name)); r = ioctl(sock, SIOCGIFFLAGS, &ifr); close(sock); /* IFF_RUNNING not required */ if (r == -1 || !(ifr.ifr_flags & IFF_UP)) return; l_info("Selected interface %s", ifname); wdev_id = cur_wdev_id; memcpy(wdev_addr, ifaddr, 6); } static void get_interface_done(void *user_data) { struct l_genl_msg *msg; uint8_t frame_buf[256] __attribute__ ((aligned)); struct mmpdu_header *hdr = (void *) frame_buf; static const uint8_t bcast_addr[6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; size_t frame_len; if (!wdev_id) { l_error("No suitable interface found"); exit_status = EXIT_FAILURE; l_main_quit(); return; } memset(frame_buf, 0, sizeof(*hdr)); hdr->fc.protocol_version = 0; hdr->fc.type = MPDU_TYPE_MANAGEMENT; hdr->fc.subtype = MPDU_MANAGEMENT_SUBTYPE_PROBE_REQUEST; memcpy(hdr->address_1, bcast_addr, 6); /* DA */ memcpy(hdr->address_2, wdev_addr, 6); /* SA */ memcpy(hdr->address_3, bcast_addr, 6); /* BSSID */ frame_len = (uint8_t *) mmpdu_body(hdr) - (uint8_t *) hdr; memcpy((void *) mmpdu_body(hdr), probe_req_body, sizeof(probe_req_body)); frame_len += sizeof(probe_req_body); msg = l_genl_msg_new_sized(NL80211_CMD_FRAME, 128 + frame_len); l_genl_msg_append_attr(msg, NL80211_ATTR_WDEV, 8, &wdev_id); if (freq) { l_genl_msg_append_attr(msg, NL80211_ATTR_WIPHY_FREQ, 4, &freq); l_genl_msg_append_attr(msg, NL80211_ATTR_OFFCHANNEL_TX_OK, 0, NULL); } l_genl_msg_append_attr(msg, NL80211_ATTR_FRAME, frame_len, frame_buf); l_genl_msg_append_attr(msg, NL80211_ATTR_DONT_WAIT_FOR_ACK, 0, NULL); l_genl_msg_append_attr(msg, NL80211_ATTR_TX_NO_CCK_RATE, 0, NULL); if (!l_genl_family_send(nl80211, msg, frame_cb, user_data, NULL)) { l_error("l_genl_family_send failed"); exit_status = EXIT_FAILURE; l_main_quit(); return; } } static void dump_interfaces(void) { struct l_genl_msg *msg; msg = l_genl_msg_new(NL80211_CMD_GET_INTERFACE); if (!l_genl_family_dump(nl80211, msg, get_interface_callback, NULL, get_interface_done)) { l_genl_msg_unref(msg); l_error("Getting nl80211 interface information failed"); exit_status = EXIT_FAILURE; l_main_quit(); return; } } static void family_discovered(const struct l_genl_family_info *info, void *user_data) { if (!strcmp(l_genl_family_info_get_name(info), NL80211_GENL_NAME)) nl80211 = l_genl_family_new(genl, NL80211_GENL_NAME); } static void discovery_done(void *user_data) { if (!nl80211) { l_error("nl80211 doesn't exist.\n" "Load it manually using modprobe cfg80211"); goto quit; } dump_interfaces(); return; quit: exit_status = EXIT_FAILURE; l_main_quit(); } int main(int argc, char *argv[]) { if (argc >= 2) { char *endp; if (!strcmp(argv[1], "-h")) { fprintf(stderr, "Usage: %s [<frequency>]\n\n" "Send out a broadcast Probe Request frame. " "A wireless interface must be UP. If a " "frequency is not given, the frame is " "transmitted on the current channel.\n", argv[0]); return EXIT_SUCCESS; } freq = strtol(argv[1], &endp, 0); if (*endp != '\0') { fprintf(stderr, "Can't parse '%s'\n", endp); return EXIT_FAILURE; } } if (!l_main_init()) return EXIT_FAILURE; l_log_set_stderr(); exit_status = EXIT_FAILURE; genl = l_genl_new(); if (!genl) { l_error("Failed to initialize generic netlink"); goto done; } if (!l_genl_discover_families(genl, family_discovered, NULL, discovery_done)) { l_error("Unable to start family discovery"); l_genl_unref(genl); goto done; } l_main_run(); l_genl_family_free(nl80211); l_genl_unref(genl); done: l_main_exit(); return exit_status; }