dockapps/wmwlmon/wl.c
2012-02-12 22:50:31 +00:00

573 lines
12 KiB
C

/* $Id: wl.c,v 1.18 2008/05/13 04:42:17 hacki Exp $ */
/*
* Copyright (c) 2005, 2006 Marcus Glocker <marcus@nazgul.ch>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/* Ported to FreeBSD by Nathan Lay <nslay@hotmail.com> 4/30/06 */
#include <sys/param.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#ifdef __OpenBSD__
#include <dev/ic/if_wi_ieee.h>
#endif
#ifdef __FreeBSD__
#include <dev/wi/if_wavelan_ieee.h>
#endif
#include <err.h>
#include <ifaddrs.h>
#include <net/if.h>
#include <net/if_media.h>
#include <net80211/ieee80211.h>
#include <net80211/ieee80211_ioctl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include "wl.h"
/*
* global variables for this file
*/
#ifdef __OpenBSD__
static char *speed[25] = {
"DS1", "1",
"DS2", "2",
"DS5", "5.5",
"DS11", "11",
"OFDM6", "6",
"OFDM9", "9",
"OFDM12", "12s",
"OFDM18", "18",
"OFDM24", "24",
"OFDM36", "36",
"OFDM48", "48",
"OFDM54", "54",
NULL
};
#endif
#ifdef __FreeBSD__
static char *speed[] = {
"FH/1Mbps", "1",
"FH/2Mbps", "2",
"DS/1Mbps", "1",
"DS/2Mbps", "2",
"DS/5.5Mbps", "5.5",
"DS/11Mbps", "11",
"DS/22Mbps", "22",
"OFDM/6Mbps", "6",
"OFDM/9Mbps", "9",
"OFDM/12Mbps", "12",
"OFDM/18Mbps", "18",
"OFDM/24Mbps", "24",
"OFDM/36Mbps", "36",
"OFDM/48Mbps", "48",
"OFDM/54Mbps", "54",
"OFDM/72Mbps", "72",
"DS/354Kbps", "0.354",
"DS/512Kbps", "0.512",
NULL
};
#endif
/*
* get_wep()
* get wep status
* Return:
* 0 = wep disabled, 1 = wep enabled, -1 = error
*/
int
get_wep(const char *interface)
{
int r = 0, s, inwkey;
#ifdef __OpenBSD__
struct ieee80211_nwkey nwkey;
#endif
#ifdef __FreeBSD__
struct ieee80211req nwkey;
#endif
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (-1);
memset(&nwkey, 0, sizeof(nwkey));
#ifdef __FreeBSD__
nwkey.i_type = IEEE80211_IOC_WEP;
#endif
strlcpy(nwkey.i_name, interface, sizeof(nwkey.i_name));
#ifdef __OpenBSD__
if ((inwkey = ioctl(s, SIOCG80211NWKEY, (caddr_t)&nwkey)) == -1) {
close(s);
return (-1);
}
#endif
#ifdef __FreeBSD__
if ((inwkey = ioctl(s, SIOCG80211, (caddr_t)&nwkey)) == -1) {
close(s);
return (-1);
}
#endif
close(s);
#ifdef __OpenBSD__
if (inwkey == 0 && nwkey.i_wepon > 0)
r = 1;
#endif
#ifdef __FreeBSD__
if (inwkey == 0 && nwkey.i_val > 0)
r = 1;
#endif
return (r);
}
/*
* get_channel()
* get channel number
* Return:
* <channel number> = success, 0 = no data, -1 = error
*/
int
get_channel(const char *interface)
{
int s, ichan;
#ifdef __OpenBSD__
struct ieee80211chanreq channel;
#endif
#ifdef __FreeBSD__
struct ieee80211req channel;
#endif
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (-1);
memset(&channel, 0, sizeof(channel));
#ifdef __FreeBSD__
channel.i_type = IEEE80211_IOC_CHANNEL;
#endif
strlcpy(channel.i_name, interface, sizeof(channel.i_name));
#ifdef __OpenBSD__
if ((ichan = ioctl(s, SIOCG80211CHANNEL, (caddr_t)&channel)) == -1) {
close(s);
return (-1);
}
#endif
#ifdef __FreeBSD__
if ((ichan = ioctl(s, SIOCG80211, (caddr_t)&channel)) == -1) {
close(s);
return (-1);
}
#endif
close(s);
if (ichan == 0) {
#ifdef __OpenBSD__
if (channel.i_channel < 1000)
return (channel.i_channel);
#endif
#ifdef __FreeBSD__
if (channel.i_val < 1000)
return (channel.i_val);
#endif
}
return (0);
}
/*
* get_signal()
* get signal strength
* Return:
* <signal strength> = success, 0 = no data, -1 = error
*/
int
get_signal(const char *interface, const char *network)
{
#ifdef __OpenBSD__
int i = 0, s, len;
struct ieee80211_nodereq_all na;
struct ieee80211_nodereq nr[8];
#endif
#ifdef __FreeBSD__
int s, len;
uint8_t buf[24 * 1024], *cp;
struct ieee80211req na;
#endif
char network_id[IEEE80211_NWID_LEN + 1];
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (-1);
memset(&na, 0, sizeof(na));
#ifdef __OpenBSD__
memset(&nr, 0, sizeof(nr));
na.na_node = nr;
na.na_size = sizeof(nr);
strlcpy(na.na_ifname, interface, sizeof(na.na_ifname));
#endif
#ifdef __FreeBSD__
strlcpy(na.i_name, interface, sizeof(na.i_name));
na.i_type = IEEE80211_IOC_SCAN_RESULTS;
na.i_data = buf;
na.i_len = sizeof(buf);
#endif
#ifdef __OpenBSD__
if (ioctl(s, SIOCG80211ALLNODES, &na) == -1) {
close(s);
return (-1);
}
#endif
#ifdef __FreeBSD__
if (ioctl(s, SIOCG80211, (caddr_t)&na) == -1) {
close(s);
return (-1);
}
#endif
close(s);
#ifdef __OpenBSD__
for (i = 0; i < na.na_nodes; i++) {
if (nr[i].nr_nwid_len < sizeof(network_id))
len = nr[i].nr_nwid_len + 1;
else
len = sizeof(network_id);
strlcpy(network_id, (const char *)nr[i].nr_nwid, len);
if (!strcmp(network_id, network))
return (nr[i].nr_rssi);
}
#endif
#ifdef __FreeBSD__
/* This is how ifconfig does it */
len = na.i_len;
cp = buf;
do {
struct ieee80211req_scan_result *sr;
uint8_t *vp;
sr = (struct ieee80211req_scan_result *)cp;
vp = (u_int8_t *)(sr + 1);
strlcpy(network_id, (const char *)vp, sr->isr_ssid_len + 1);
if (!strcmp(network_id, network))
return (sr->isr_rssi);
cp += sr->isr_len;
len -= sr->isr_len;
if (sr->isr_len <= 0) /* Some weird lockup */
break;
} while (len >= sizeof(struct ieee80211req_scan_result));
#endif
return (0);
}
/*
* get_wi_signal()
* get signal strength for wi interfaces
* Return:
* <signal strength> = success, -1 = error
*/
int
get_wi_signal(const char *interface)
{
int s;
struct ifreq ifr;
struct wi_req wreq;
#ifdef __OpenBSD__
float link;
#endif
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (-1);
bzero((char *)&wreq, sizeof(wreq));
bzero((char *)&ifr, sizeof(ifr));
wreq.wi_len = WI_MAX_DATALEN;
wreq.wi_type = WI_RID_COMMS_QUALITY;
ifr.ifr_data = (caddr_t)&wreq;
strlcpy(ifr.ifr_name, interface, sizeof(ifr.ifr_name));
if (ioctl(s, SIOCGWAVELAN, &ifr) == -1) {
close(s);
return (-1);
}
close(s);
#ifdef __OpenBSD__
link = wreq.wi_val[0];
/*
* for future reference ...
* level = wreq.wi_val[1];
* noise = wreq.wi_val[2];
*/
return (letoh16(link));
#endif
#ifdef __FreeBSD__
return (wreq.wi_val[1]);
#endif
}
/*
* get_speed()
* get media speed
* Return:
* <pointer to speed> = success, NULL = no data / error
*/
char *
get_speed(const char *interface)
{
int s, mword;
struct ifmediareq ifmr;
const struct ifmedia_description *desc;
#ifdef __OpenBSD__
const struct ifmedia_description ifm_subtype_descriptions[] =
IFM_SUBTYPE_DESCRIPTIONS;
#endif
#ifdef __FreeBSD__
const struct ifmedia_description ifm_subtype_descriptions[] =
IFM_SUBTYPE_IEEE80211_DESCRIPTIONS;
#endif
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (NULL);
memset(&ifmr, 0, sizeof(ifmr));
strlcpy(ifmr.ifm_name, interface, sizeof(ifmr.ifm_name));
if (ioctl(s, SIOCGIFMEDIA, (caddr_t)&ifmr) == -1) {
close(s);
return (NULL);
}
close(s);
mword = ifmr.ifm_active;
for (desc = ifm_subtype_descriptions; desc->ifmt_string != NULL;
desc++) {
if (IFM_TYPE_MATCH(desc->ifmt_word, mword) &&
IFM_SUBTYPE(desc->ifmt_word) == IFM_SUBTYPE(mword))
return (translate_speed(desc->ifmt_string));
}
return (NULL);
}
/*
* get_status()
* get network status
* Return:
* <pointer to status> = success, NULL = no data / error
*/
char *
get_status(const char *interface)
{
#ifdef __OpenBSD__
int s, bitno;
static char status[64];
const struct ifmedia_status_description *ifms;
const struct ifmedia_status_description ifm_status_descriptions[] =
IFM_STATUS_DESCRIPTIONS;
const int ifm_status_valid_list[] =
IFM_STATUS_VALID_LIST;
#endif
#ifdef __FreeBSD__
int s;
#endif
struct ifmediareq ifmr;
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (NULL);
memset(&ifmr, 0, sizeof(ifmr));
strlcpy(ifmr.ifm_name, interface, sizeof(ifmr.ifm_name));
if (ioctl(s, SIOCGIFMEDIA, (caddr_t)&ifmr) == -1) {
close(s);
return (NULL);
}
close(s);
#ifdef __OpenBSD__
for (bitno = 0; ifm_status_valid_list[bitno] != 0; bitno++) {
for (ifms = ifm_status_descriptions; ifms->ifms_valid != 0;
ifms++) {
if (ifms->ifms_type != IFM_TYPE(ifmr.ifm_current) ||
ifms->ifms_valid != ifm_status_valid_list[bitno])
continue;
strlcpy(status, IFM_STATUS_DESC(ifms, ifmr.ifm_status),
sizeof(status));
return (status);
}
}
return (NULL);
#endif
#ifdef __FreeBSD__
if (ifmr.ifm_status & IFM_ACTIVE)
return ("active");
return ("no carrier");
#endif
}
/*
* get_nwid()
* get wireless network id
* Return:
* pointer to network id = success, NULL = no data / error
*/
char *
get_nwid(const char *interface)
{
int s, len, inwid;
static char network_id[IEEE80211_NWID_LEN + 1];
#ifdef __OpenBSD__
struct ifreq ifr;
struct ieee80211_nwid nwid;
#endif
#ifdef __FreeBSD__
struct ieee80211req nwid;
#endif
if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
return (NULL);
#ifdef __OpenBSD__
memset(&ifr, 0, sizeof(ifr));
ifr.ifr_data = (caddr_t)&nwid;
strlcpy(ifr.ifr_name, interface, sizeof(ifr.ifr_name));
#endif
#ifdef __FreeBSD__
memset(&nwid, 0, sizeof(nwid));
strlcpy(nwid.i_name, interface, sizeof(nwid.i_name));
nwid.i_type = IEEE80211_IOC_SSID;
nwid.i_data = network_id;
nwid.i_len = sizeof(network_id);
#endif
#ifdef __OpenBSD__
if ((inwid = ioctl(s, SIOCG80211NWID, (caddr_t)&ifr)) == -1) {
close(s);
return (NULL);
}
#endif
#ifdef __FreeBSD__
if ((inwid = ioctl(s, SIOCG80211, (caddr_t)&nwid)) == -1) {
close(s);
return (NULL);
}
#endif
close(s);
if (inwid == 0) {
if (nwid.i_len < sizeof(network_id))
len = nwid.i_len + 1;
else
len = sizeof(network_id);
#ifdef __OpenBSD__
strlcpy(network_id, (const char *)nwid.i_nwid, len);
#endif
#ifdef __FreeBSD__
network_id[len - 1] = '\0';
#endif
return (network_id);
}
return (NULL);
}
/*
* get_first_wnic()
* scans interfaces and returns the first found wireless interface
* Return:
* <pointer to wnic> = success, NULL = no nic found
*/
char *
get_first_wnic(void)
{
char *r = NULL;
char nic[IF_NAMESIZE];
struct ifaddrs *ifap = NULL, *ifa = NULL;
memset(nic, 0, sizeof(nic));
if (getifaddrs(&ifap) != 0)
errx(1, "getifaddrs");
for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
if (strcmp(nic, ifa->ifa_name)) {
if (get_wep(ifa->ifa_name) != -1) {
r = strdup(ifa->ifa_name);
break;
}
}
strlcpy(nic, ifa->ifa_name, sizeof(nic));
}
freeifaddrs(ifap);
return (r);
}
/*
* check_nic()
* check if the monitored interface still exists
* Return:
* 0 = interface gone, 1 = interface exists
*/
int
check_nic(const char *interface)
{
int r = 0;
char nic[IF_NAMESIZE];
struct ifaddrs *ifap = NULL, *ifa = NULL;
memset(nic, 0, sizeof(nic));
if (getifaddrs(&ifap) != 0)
errx(1, "getifaddrs");
for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
if (strcmp(nic, ifa->ifa_name)) {
if (!strcmp(ifa->ifa_name, interface)) {
r = 1;
break;
}
}
strlcpy(nic, ifa->ifa_name, sizeof(nic));
}
freeifaddrs(ifap);
return (r);
}
/*
* translate_speed()
* translate the result of media speed to human readable string
* Return:
* <pointer to speed> = success, NULL = no data / error
*/
char *
translate_speed(const char *mode)
{
int i;
for (i = 0; speed[i] != NULL; i++) {
if (!strcmp(mode, speed[i]))
return (speed[i + 1]);
}
return (NULL);
}