|
Packit Service |
dff8e4 |
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
Packit Service |
dff8e4 |
/*
|
|
Packit Service |
dff8e4 |
* Copyright (C) 2005 - 2018 Red Hat, Inc.
|
|
Packit Service |
dff8e4 |
* Copyright (C) 2006 - 2008 Novell, Inc.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#include "libnm-glib-aux/nm-default-glib-i18n-lib.h"
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#include "nm-wifi-utils-wext.h"
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#include <sys/ioctl.h>
|
|
Packit Service |
dff8e4 |
#include <net/ethernet.h>
|
|
Packit Service |
dff8e4 |
#include <unistd.h>
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Hacks necessary to #include wireless.h; yay for WEXT */
|
|
Packit Service |
dff8e4 |
#ifndef __user
|
|
Packit Service |
dff8e4 |
#define __user
|
|
Packit Service |
dff8e4 |
#endif
|
|
Packit Service |
dff8e4 |
#include <sys/types.h>
|
|
Packit Service |
dff8e4 |
#include <linux/types.h>
|
|
Packit Service |
dff8e4 |
#include <sys/socket.h>
|
|
Packit Service |
dff8e4 |
#include <linux/wireless.h>
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#include "libnm-log-core/nm-logging.h"
|
|
Packit Service |
dff8e4 |
#include "nm-wifi-utils-private.h"
|
|
Packit Service |
dff8e4 |
#include "libnm-platform/nm-platform-utils.h"
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
typedef struct {
|
|
Packit Service |
dff8e4 |
NMWifiUtils parent;
|
|
Packit Service |
dff8e4 |
int fd;
|
|
Packit Service |
dff8e4 |
struct iw_quality max_qual;
|
|
Packit Service |
dff8e4 |
gint8 num_freqs;
|
|
Packit Service |
dff8e4 |
guint32 freqs[IW_MAX_FREQUENCIES];
|
|
Packit Service |
dff8e4 |
} NMWifiUtilsWext;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
typedef struct {
|
|
Packit Service |
dff8e4 |
NMWifiUtilsClass parent;
|
|
Packit Service |
dff8e4 |
} NMWifiUtilsWextClass;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
G_DEFINE_TYPE(NMWifiUtilsWext, nm_wifi_utils_wext, NM_TYPE_WIFI_UTILS)
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Until a new wireless-tools comes out that has the defs and the structure,
|
|
Packit Service |
dff8e4 |
* need to copy them here.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
/* Scan capability flags - in (struct iw_range *)->scan_capa */
|
|
Packit Service |
dff8e4 |
#define NM_IW_SCAN_CAPA_NONE 0x00
|
|
Packit Service |
dff8e4 |
#define NM_IW_SCAN_CAPA_ESSID 0x01
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
struct iw_range_with_scan_capa {
|
|
Packit Service |
dff8e4 |
guint32 throughput;
|
|
Packit Service |
dff8e4 |
guint32 min_nwid;
|
|
Packit Service |
dff8e4 |
guint32 max_nwid;
|
|
Packit Service |
dff8e4 |
guint16 old_num_channels;
|
|
Packit Service |
dff8e4 |
guint8 old_num_frequency;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
guint8 scan_capa;
|
|
Packit Service |
dff8e4 |
/* don't need the rest... */
|
|
Packit Service |
dff8e4 |
};
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#define _NMLOG_PREFIX_NAME "wifi-wext"
|
|
Packit Service |
dff8e4 |
#define _NMLOG(level, domain, ...) \
|
|
Packit Service |
dff8e4 |
G_STMT_START \
|
|
Packit Service |
dff8e4 |
{ \
|
|
Packit Service |
dff8e4 |
nm_log((level), \
|
|
Packit Service |
dff8e4 |
(domain), \
|
|
Packit Service |
dff8e4 |
NULL, \
|
|
Packit Service |
dff8e4 |
NULL, \
|
|
Packit Service |
dff8e4 |
"%s: " _NM_UTILS_MACRO_FIRST(__VA_ARGS__), \
|
|
Packit Service |
dff8e4 |
_NMLOG_PREFIX_NAME _NM_UTILS_MACRO_REST(__VA_ARGS__)); \
|
|
Packit Service |
dff8e4 |
} \
|
|
Packit Service |
dff8e4 |
G_STMT_END
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
iw_freq_to_uint32(const struct iw_freq *freq)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
if (freq->e == 0) {
|
|
Packit Service |
dff8e4 |
/* Some drivers report channel not frequency. Convert to a
|
|
Packit Service |
dff8e4 |
* frequency; but this assumes that the device is in b/g mode.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
if ((freq->m >= 1) && (freq->m <= 13))
|
|
Packit Service |
dff8e4 |
return 2407 + (5 * freq->m);
|
|
Packit Service |
dff8e4 |
else if (freq->m == 14)
|
|
Packit Service |
dff8e4 |
return 2484;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return (guint32)((((double) freq->m) * nm_utils_exp10(freq->e)) / 1000000.0);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static void
|
|
Packit Service |
dff8e4 |
dispose(GObject *object)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = NM_WIFI_UTILS_WEXT(object);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext->fd = nm_close(wext->fd);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
get_ifname(int ifindex, char *buffer, const char *op)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
int errsv;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!nmp_utils_if_indextoname(ifindex, buffer)) {
|
|
Packit Service |
dff8e4 |
errsv = errno;
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"error getting interface name for ifindex %d, operation '%s': %s (%d)",
|
|
Packit Service |
dff8e4 |
ifindex,
|
|
Packit Service |
dff8e4 |
op,
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errsv),
|
|
Packit Service |
dff8e4 |
errsv);
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static _NM80211Mode
|
|
Packit Service |
dff8e4 |
wifi_wext_get_mode_ifname(NMWifiUtils *data, const char *ifname)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
int errsv;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCGIWMODE, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
errsv = errno;
|
|
Packit Service |
dff8e4 |
if (errsv != ENODEV) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI, "(%s): error %d getting card mode", ifname, errsv);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return _NM_802_11_MODE_UNKNOWN;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
switch (wrq.u.mode) {
|
|
Packit Service |
dff8e4 |
case IW_MODE_ADHOC:
|
|
Packit Service |
dff8e4 |
return _NM_802_11_MODE_ADHOC;
|
|
Packit Service |
dff8e4 |
case IW_MODE_MASTER:
|
|
Packit Service |
dff8e4 |
return _NM_802_11_MODE_AP;
|
|
Packit Service |
dff8e4 |
case IW_MODE_INFRA:
|
|
Packit Service |
dff8e4 |
case IW_MODE_AUTO: /* hack for WEXT devices reporting IW_MODE_AUTO */
|
|
Packit Service |
dff8e4 |
return _NM_802_11_MODE_INFRA;
|
|
Packit Service |
dff8e4 |
default:
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return _NM_802_11_MODE_UNKNOWN;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static _NM80211Mode
|
|
Packit Service |
dff8e4 |
wifi_wext_get_mode(NMWifiUtils *data)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "get-mode"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return wifi_wext_get_mode_ifname(data, ifname);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_set_mode(NMWifiUtils *data, const _NM80211Mode mode)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "set-mode"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (wifi_wext_get_mode_ifname(data, ifname) == mode)
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
switch (mode) {
|
|
Packit Service |
dff8e4 |
case _NM_802_11_MODE_ADHOC:
|
|
Packit Service |
dff8e4 |
wrq.u.mode = IW_MODE_ADHOC;
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
case _NM_802_11_MODE_AP:
|
|
Packit Service |
dff8e4 |
wrq.u.mode = IW_MODE_MASTER;
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
case _NM_802_11_MODE_INFRA:
|
|
Packit Service |
dff8e4 |
wrq.u.mode = IW_MODE_INFRA;
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
default:
|
|
Packit Service |
dff8e4 |
g_warn_if_reached();
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCSIWMODE, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
if (errno != ENODEV) {
|
|
Packit Service |
dff8e4 |
_LOGE(LOGD_PLATFORM | LOGD_WIFI, "(%s): error setting mode %d", ifname, mode);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_set_powersave(NMWifiUtils *data, guint32 powersave)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "set-powersave"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
if (powersave == 1) {
|
|
Packit Service |
dff8e4 |
wrq.u.power.flags = IW_POWER_ALL_R;
|
|
Packit Service |
dff8e4 |
} else
|
|
Packit Service |
dff8e4 |
wrq.u.power.disabled = 1;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCSIWPOWER, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
if (errno != ENODEV) {
|
|
Packit Service |
dff8e4 |
_LOGE(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): error setting powersave %" G_GUINT32_FORMAT,
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
powersave);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
wifi_wext_get_freq(NMWifiUtils *data)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "get-freq"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCGIWFREQ, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): error getting frequency: %s",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errno));
|
|
Packit Service |
dff8e4 |
return 0;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return iw_freq_to_uint32(&wrq.u.freq);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
wifi_wext_find_freq(NMWifiUtils *data, const guint32 *freqs)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
int i;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
for (i = 0; i < wext->num_freqs; i++) {
|
|
Packit Service |
dff8e4 |
while (*freqs) {
|
|
Packit Service |
dff8e4 |
if (wext->freqs[i] == *freqs)
|
|
Packit Service |
dff8e4 |
return *freqs;
|
|
Packit Service |
dff8e4 |
freqs++;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return 0;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_get_bssid(NMWifiUtils *data, NMEtherAddr *out_bssid)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "get-bssid"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(wrq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCGIWAP, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): error getting associated BSSID: %s",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errno));
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
memcpy(out_bssid, &(wrq.u.ap_addr.sa_data), ETH_ALEN);
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
wifi_wext_get_rate(NMWifiUtils *data)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
int err;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "get-rate"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(wrq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
err = ioctl(wext->fd, SIOCGIWRATE, &wrq;;
|
|
Packit Service |
dff8e4 |
return ((err == 0) ? wrq.u.bitrate.value / 1000 : 0);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static int
|
|
Packit Service |
dff8e4 |
wext_qual_to_percent(const struct iw_quality *qual, const struct iw_quality *max_qual)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
int percent = -1;
|
|
Packit Service |
dff8e4 |
int level_percent = -1;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
g_return_val_if_fail(qual != NULL, -1);
|
|
Packit Service |
dff8e4 |
g_return_val_if_fail(max_qual != NULL, -1);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Magically convert the many different WEXT quality representations to a percentage */
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
_LOGD(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"QL: qual %d/%u/0x%X, level %d/%u/0x%X, noise %d/%u/0x%X, updated: 0x%X ** MAX: qual "
|
|
Packit Service |
dff8e4 |
"%d/%u/0x%X, level %d/%u/0x%X, noise %d/%u/0x%X, updated: 0x%X",
|
|
Packit Service |
dff8e4 |
(__s8) qual->qual,
|
|
Packit Service |
dff8e4 |
qual->qual,
|
|
Packit Service |
dff8e4 |
qual->qual,
|
|
Packit Service |
dff8e4 |
(__s8) qual->level,
|
|
Packit Service |
dff8e4 |
qual->level,
|
|
Packit Service |
dff8e4 |
qual->level,
|
|
Packit Service |
dff8e4 |
(__s8) qual->noise,
|
|
Packit Service |
dff8e4 |
qual->noise,
|
|
Packit Service |
dff8e4 |
qual->noise,
|
|
Packit Service |
dff8e4 |
qual->updated,
|
|
Packit Service |
dff8e4 |
(__s8) max_qual->qual,
|
|
Packit Service |
dff8e4 |
max_qual->qual,
|
|
Packit Service |
dff8e4 |
max_qual->qual,
|
|
Packit Service |
dff8e4 |
(__s8) max_qual->level,
|
|
Packit Service |
dff8e4 |
max_qual->level,
|
|
Packit Service |
dff8e4 |
max_qual->level,
|
|
Packit Service |
dff8e4 |
(__s8) max_qual->noise,
|
|
Packit Service |
dff8e4 |
max_qual->noise,
|
|
Packit Service |
dff8e4 |
max_qual->noise,
|
|
Packit Service |
dff8e4 |
max_qual->updated);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Try using the card's idea of the signal quality first as long as it tells us what the max quality is.
|
|
Packit Service |
dff8e4 |
* Drivers that fill in quality values MUST treat them as percentages, ie the "Link Quality" MUST be
|
|
Packit Service |
dff8e4 |
* bounded by 0 and max_qual->qual, and MUST change in a linear fashion. Within those bounds, drivers
|
|
Packit Service |
dff8e4 |
* are free to use whatever they want to calculate "Link Quality".
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
if ((max_qual->qual != 0) && !(max_qual->updated & IW_QUAL_QUAL_INVALID)
|
|
Packit Service |
dff8e4 |
&& !(qual->updated & IW_QUAL_QUAL_INVALID))
|
|
Packit Service |
dff8e4 |
percent = (int) (100 * ((double) qual->qual / (double) max_qual->qual));
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* If the driver doesn't specify a complete and valid quality, we have two options:
|
|
Packit Service |
dff8e4 |
*
|
|
Packit Service |
dff8e4 |
* 1) dBm: driver must specify max_qual->level = 0, and have valid values for
|
|
Packit Service |
dff8e4 |
* qual->level and (qual->noise OR max_qual->noise)
|
|
Packit Service |
dff8e4 |
* 2) raw RSSI: driver must specify max_qual->level > 0, and have valid values for
|
|
Packit Service |
dff8e4 |
* qual->level and max_qual->level
|
|
Packit Service |
dff8e4 |
*
|
|
Packit Service |
dff8e4 |
* This is the WEXT spec. If this interpretation is wrong, I'll fix it. Otherwise,
|
|
Packit Service |
dff8e4 |
* If drivers don't conform to it, they are wrong and need to be fixed.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if ((max_qual->level == 0)
|
|
Packit Service |
dff8e4 |
&& !(max_qual->updated & IW_QUAL_LEVEL_INVALID) /* Valid max_qual->level == 0 */
|
|
Packit Service |
dff8e4 |
&& !(qual->updated & IW_QUAL_LEVEL_INVALID) /* Must have valid qual->level */
|
|
Packit Service |
dff8e4 |
&& (((max_qual->noise > 0)
|
|
Packit Service |
dff8e4 |
&& !(max_qual->updated & IW_QUAL_NOISE_INVALID)) /* Must have valid max_qual->noise */
|
|
Packit Service |
dff8e4 |
|| ((qual->noise > 0)
|
|
Packit Service |
dff8e4 |
&& !(qual->updated & IW_QUAL_NOISE_INVALID))) /* OR valid qual->noise */
|
|
Packit Service |
dff8e4 |
) {
|
|
Packit Service |
dff8e4 |
/* Absolute power values (dBm) */
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Reasonable fallbacks for dumb drivers that don't specify either level. */
|
|
Packit Service |
dff8e4 |
#define FALLBACK_NOISE_FLOOR_DBM -90
|
|
Packit Service |
dff8e4 |
#define FALLBACK_SIGNAL_MAX_DBM -20
|
|
Packit Service |
dff8e4 |
int max_level = FALLBACK_SIGNAL_MAX_DBM;
|
|
Packit Service |
dff8e4 |
int noise = FALLBACK_NOISE_FLOOR_DBM;
|
|
Packit Service |
dff8e4 |
int level = qual->level - 0x100;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
level = CLAMP(level, FALLBACK_NOISE_FLOOR_DBM, FALLBACK_SIGNAL_MAX_DBM);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if ((qual->noise > 0) && !(qual->updated & IW_QUAL_NOISE_INVALID))
|
|
Packit Service |
dff8e4 |
noise = qual->noise - 0x100;
|
|
Packit Service |
dff8e4 |
else if ((max_qual->noise > 0) && !(max_qual->updated & IW_QUAL_NOISE_INVALID))
|
|
Packit Service |
dff8e4 |
noise = max_qual->noise - 0x100;
|
|
Packit Service |
dff8e4 |
noise = CLAMP(noise, FALLBACK_NOISE_FLOOR_DBM, FALLBACK_SIGNAL_MAX_DBM - 1);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* A sort of signal-to-noise ratio calculation */
|
|
Packit Service |
dff8e4 |
level_percent = (int) (100
|
|
Packit Service |
dff8e4 |
- 70
|
|
Packit Service |
dff8e4 |
* (((double) max_level - (double) level)
|
|
Packit Service |
dff8e4 |
/ ((double) max_level - (double) noise)));
|
|
Packit Service |
dff8e4 |
_LOGD(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"QL1: level_percent is %d. max_level %d, level %d, noise_floor %d.",
|
|
Packit Service |
dff8e4 |
level_percent,
|
|
Packit Service |
dff8e4 |
max_level,
|
|
Packit Service |
dff8e4 |
level,
|
|
Packit Service |
dff8e4 |
noise);
|
|
Packit Service |
dff8e4 |
} else if ((max_qual->level != 0)
|
|
Packit Service |
dff8e4 |
&& !(max_qual->updated
|
|
Packit Service |
dff8e4 |
& IW_QUAL_LEVEL_INVALID) /* Valid max_qual->level as upper bound */
|
|
Packit Service |
dff8e4 |
&& !(qual->updated & IW_QUAL_LEVEL_INVALID)) {
|
|
Packit Service |
dff8e4 |
/* Relative power values (RSSI) */
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
int level = qual->level;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Signal level is relavtive (0 -> max_qual->level) */
|
|
Packit Service |
dff8e4 |
level = CLAMP(level, 0, max_qual->level);
|
|
Packit Service |
dff8e4 |
level_percent = (int) (100 * ((double) level / (double) max_qual->level));
|
|
Packit Service |
dff8e4 |
_LOGD(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"QL2: level_percent is %d. max_level %d, level %d.",
|
|
Packit Service |
dff8e4 |
level_percent,
|
|
Packit Service |
dff8e4 |
max_qual->level,
|
|
Packit Service |
dff8e4 |
level);
|
|
Packit Service |
dff8e4 |
} else if (percent == -1) {
|
|
Packit Service |
dff8e4 |
_LOGD(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"QL: Could not get quality %% value from driver. Driver is probably buggy.");
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* If the quality percent was 0 or doesn't exist, then try to use signal levels instead */
|
|
Packit Service |
dff8e4 |
if ((percent < 1) && (level_percent >= 0))
|
|
Packit Service |
dff8e4 |
percent = level_percent;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
_LOGD(LOGD_WIFI, "QL: Final quality percent is %d (%d).", percent, CLAMP(percent, 0, 100));
|
|
Packit Service |
dff8e4 |
return (CLAMP(percent, 0, 100));
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static int
|
|
Packit Service |
dff8e4 |
wifi_wext_get_qual(NMWifiUtils *data)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext * wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
struct iw_statistics stats;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "get-qual"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&stats, 0, sizeof(stats));
|
|
Packit Service |
dff8e4 |
wrq.u.data.pointer = &stat;;
|
|
Packit Service |
dff8e4 |
wrq.u.data.length = sizeof(stats);
|
|
Packit Service |
dff8e4 |
wrq.u.data.flags = 1; /* Clear updated flag */
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCGIWSTATS, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): error getting signal strength: %s",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errno));
|
|
Packit Service |
dff8e4 |
return -1;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return wext_qual_to_percent(&stats.qual, &wext->max_qual);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_get_station(NMWifiUtils *data,
|
|
Packit Service |
dff8e4 |
NMEtherAddr *out_bssid,
|
|
Packit Service |
dff8e4 |
int * out_quality,
|
|
Packit Service |
dff8e4 |
guint32 * out_rate)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMEtherAddr local_addr;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!out_bssid && !out_quality && !out_rate) {
|
|
Packit Service |
dff8e4 |
/* hm, the caller requested no parameter at all?
|
|
Packit Service |
dff8e4 |
* Don't simply return TRUE, but at least check that
|
|
Packit Service |
dff8e4 |
* we can successfully fetch the bssid. */
|
|
Packit Service |
dff8e4 |
out_bssid = &local_addr;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (out_bssid) {
|
|
Packit Service |
dff8e4 |
if (!wifi_wext_get_bssid(data, out_bssid))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
if (out_quality) {
|
|
Packit Service |
dff8e4 |
*out_quality = wifi_wext_get_qual(data);
|
|
Packit Service |
dff8e4 |
if (*out_quality < 0)
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
if (out_rate) {
|
|
Packit Service |
dff8e4 |
*out_rate = wifi_wext_get_rate(data);
|
|
Packit Service |
dff8e4 |
if (*out_rate == 0)
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/*****************************************************************************/
|
|
Packit Service |
dff8e4 |
/* OLPC Mesh-only functions */
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
wifi_wext_get_mesh_channel(NMWifiUtils *data)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
guint32 freq;
|
|
Packit Service |
dff8e4 |
int i;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
freq = nm_wifi_utils_get_freq(data);
|
|
Packit Service |
dff8e4 |
for (i = 0; i < wext->num_freqs; i++) {
|
|
Packit Service |
dff8e4 |
if (freq == wext->freqs[i])
|
|
Packit Service |
dff8e4 |
return i + 1;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return 0;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_set_mesh_channel(NMWifiUtils *data, guint32 channel)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "set-mesh-channel"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (channel > 0) {
|
|
Packit Service |
dff8e4 |
wrq.u.freq.flags = IW_FREQ_FIXED;
|
|
Packit Service |
dff8e4 |
wrq.u.freq.e = 0;
|
|
Packit Service |
dff8e4 |
wrq.u.freq.m = channel;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCSIWFREQ, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
_LOGE(LOGD_PLATFORM | LOGD_WIFI | LOGD_OLPC,
|
|
Packit Service |
dff8e4 |
"(%s): error setting channel to %d: %s",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
channel,
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errno));
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wifi_wext_set_mesh_ssid(NMWifiUtils *data, const guint8 *ssid, gsize len)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext *wext = (NMWifiUtilsWext *) data;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
char buf[IW_ESSID_MAX_SIZE + 1];
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
int errsv;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!get_ifname(data->ifindex, ifname, "set-mesh-ssid"))
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(buf, 0, sizeof(buf));
|
|
Packit Service |
dff8e4 |
memcpy(buf, ssid, MIN(sizeof(buf) - 1, len));
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wrq.u.essid.pointer = (caddr_t) buf;
|
|
Packit Service |
dff8e4 |
wrq.u.essid.length = len;
|
|
Packit Service |
dff8e4 |
wrq.u.essid.flags = (len > 0) ? 1 : 0; /* 1=enable SSID, 0=disable/any */
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCSIWESSID, &wrq) == 0)
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
errsv = errno;
|
|
Packit Service |
dff8e4 |
if (errsv != ENODEV) {
|
|
Packit Service |
dff8e4 |
gs_free char *ssid_str = NULL;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
_LOGE(LOGD_PLATFORM | LOGD_WIFI | LOGD_OLPC,
|
|
Packit Service |
dff8e4 |
"(%s): error setting SSID to %s: %s",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
(ssid_str = _nm_utils_ssid_to_string_arr(ssid, len)),
|
|
Packit Service |
dff8e4 |
nm_strerror_native(errsv));
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/*****************************************************************************/
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wext_can_scan_ifname(NMWifiUtilsWext *wext, const char *ifname)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCSIWSCAN, &wrq) < 0) {
|
|
Packit Service |
dff8e4 |
if (errno == EOPNOTSUPP)
|
|
Packit Service |
dff8e4 |
return FALSE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static gboolean
|
|
Packit Service |
dff8e4 |
wext_get_range_ifname(NMWifiUtilsWext *wext,
|
|
Packit Service |
dff8e4 |
const char * ifname,
|
|
Packit Service |
dff8e4 |
struct iw_range *range,
|
|
Packit Service |
dff8e4 |
guint32 * response_len)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
int i = 26;
|
|
Packit Service |
dff8e4 |
gboolean success = FALSE;
|
|
Packit Service |
dff8e4 |
struct iwreq wrq;
|
|
Packit Service |
dff8e4 |
int errsv;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&wrq, 0, sizeof(struct iwreq));
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(wrq.ifr_name, ifname);
|
|
Packit Service |
dff8e4 |
wrq.u.data.pointer = (caddr_t) range;
|
|
Packit Service |
dff8e4 |
wrq.u.data.length = sizeof(struct iw_range);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Need to give some drivers time to recover after suspend/resume
|
|
Packit Service |
dff8e4 |
* (ex ipw3945 takes a few seconds to talk to its regulatory daemon;
|
|
Packit Service |
dff8e4 |
* see rh bz#362421)
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
while (i-- > 0) {
|
|
Packit Service |
dff8e4 |
if (ioctl(wext->fd, SIOCGIWRANGE, &wrq) == 0) {
|
|
Packit Service |
dff8e4 |
if (response_len)
|
|
Packit Service |
dff8e4 |
*response_len = wrq.u.data.length;
|
|
Packit Service |
dff8e4 |
success = TRUE;
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
} else {
|
|
Packit Service |
dff8e4 |
errsv = errno;
|
|
Packit Service |
dff8e4 |
if (errsv != EAGAIN) {
|
|
Packit Service |
dff8e4 |
_LOGE(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): couldn't get driver range information (%d).",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
errsv);
|
|
Packit Service |
dff8e4 |
break;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
g_usleep(G_USEC_PER_SEC / 4);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (i <= 0) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): driver took too long to respond to IWRANGE query.",
|
|
Packit Service |
dff8e4 |
ifname);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return success;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
#define WPA_CAPS \
|
|
Packit Service |
dff8e4 |
(_NM_WIFI_DEVICE_CAP_CIPHER_TKIP | _NM_WIFI_DEVICE_CAP_CIPHER_CCMP | _NM_WIFI_DEVICE_CAP_WPA \
|
|
Packit Service |
dff8e4 |
| _NM_WIFI_DEVICE_CAP_RSN)
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static guint32
|
|
Packit Service |
dff8e4 |
wext_get_caps(NMWifiUtilsWext *wext, const char *ifname, struct iw_range *range)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
guint32 caps = _NM_WIFI_DEVICE_CAP_NONE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
g_return_val_if_fail(wext != NULL, _NM_WIFI_DEVICE_CAP_NONE);
|
|
Packit Service |
dff8e4 |
g_return_val_if_fail(range != NULL, _NM_WIFI_DEVICE_CAP_NONE);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* All drivers should support WEP by default */
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_CIPHER_WEP40 | _NM_WIFI_DEVICE_CAP_CIPHER_WEP104;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (range->enc_capa & IW_ENC_CAPA_CIPHER_TKIP)
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_CIPHER_TKIP;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (range->enc_capa & IW_ENC_CAPA_CIPHER_CCMP)
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_CIPHER_CCMP;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (range->enc_capa & IW_ENC_CAPA_WPA)
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_WPA;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (range->enc_capa & IW_ENC_CAPA_WPA2)
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_RSN;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Check for cipher support but not WPA support */
|
|
Packit Service |
dff8e4 |
if ((caps & (_NM_WIFI_DEVICE_CAP_CIPHER_TKIP | _NM_WIFI_DEVICE_CAP_CIPHER_CCMP))
|
|
Packit Service |
dff8e4 |
&& !(caps & (_NM_WIFI_DEVICE_CAP_WPA | _NM_WIFI_DEVICE_CAP_RSN))) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"%s: device supports WPA ciphers but not WPA protocol; WPA unavailable.",
|
|
Packit Service |
dff8e4 |
ifname);
|
|
Packit Service |
dff8e4 |
caps &= ~WPA_CAPS;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Check for WPA support but not cipher support */
|
|
Packit Service |
dff8e4 |
if ((caps & (_NM_WIFI_DEVICE_CAP_WPA | _NM_WIFI_DEVICE_CAP_RSN))
|
|
Packit Service |
dff8e4 |
&& !(caps & (_NM_WIFI_DEVICE_CAP_CIPHER_TKIP | _NM_WIFI_DEVICE_CAP_CIPHER_CCMP))) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"%s: device supports WPA protocol but not WPA ciphers; WPA unavailable.",
|
|
Packit Service |
dff8e4 |
ifname);
|
|
Packit Service |
dff8e4 |
caps &= ~WPA_CAPS;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* There's no way to detect Ad-Hoc/AP mode support with WEXT
|
|
Packit Service |
dff8e4 |
* (other than actually trying to do it), so just assume that
|
|
Packit Service |
dff8e4 |
* Ad-Hoc is supported and AP isn't.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
caps |= _NM_WIFI_DEVICE_CAP_ADHOC;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return caps;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/*****************************************************************************/
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static void
|
|
Packit Service |
dff8e4 |
nm_wifi_utils_wext_init(NMWifiUtilsWext *self)
|
|
Packit Service |
dff8e4 |
{}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
static void
|
|
Packit Service |
dff8e4 |
nm_wifi_utils_wext_class_init(NMWifiUtilsWextClass *klass)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
GObjectClass * object_class = G_OBJECT_CLASS(klass);
|
|
Packit Service |
dff8e4 |
NMWifiUtilsClass *wifi_utils_class = NM_WIFI_UTILS_CLASS(klass);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
object_class->dispose = dispose;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wifi_utils_class->get_mode = wifi_wext_get_mode;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->set_mode = wifi_wext_set_mode;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->set_powersave = wifi_wext_set_powersave;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->get_freq = wifi_wext_get_freq;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->find_freq = wifi_wext_find_freq;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->get_station = wifi_wext_get_station;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->get_mesh_channel = wifi_wext_get_mesh_channel;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->set_mesh_channel = wifi_wext_set_mesh_channel;
|
|
Packit Service |
dff8e4 |
wifi_utils_class->set_mesh_ssid = wifi_wext_set_mesh_ssid;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
NMWifiUtils *
|
|
Packit Service |
dff8e4 |
nm_wifi_utils_wext_new(int ifindex, gboolean check_scan)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
NMWifiUtilsWext * wext;
|
|
Packit Service |
dff8e4 |
struct iw_range range;
|
|
Packit Service |
dff8e4 |
guint32 response_len = 0;
|
|
Packit Service |
dff8e4 |
struct iw_range_with_scan_capa *scan_capa_range;
|
|
Packit Service |
dff8e4 |
int i;
|
|
Packit Service |
dff8e4 |
gboolean freq_valid = FALSE, has_5ghz = FALSE, has_2ghz = FALSE;
|
|
Packit Service |
dff8e4 |
char ifname[IFNAMSIZ];
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if (!nmp_utils_if_indextoname(ifindex, ifname)) {
|
|
Packit Service |
dff8e4 |
_LOGW(LOGD_PLATFORM | LOGD_WIFI, "can't determine interface name for ifindex %d", ifindex);
|
|
Packit Service |
dff8e4 |
return NULL;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext = g_object_new(NM_TYPE_WIFI_UTILS_WEXT, NULL);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext->parent.ifindex = ifindex;
|
|
Packit Service |
dff8e4 |
wext->fd = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
|
|
Packit Service |
dff8e4 |
if (wext->fd < 0)
|
|
Packit Service |
dff8e4 |
goto error;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
memset(&range, 0, sizeof(struct iw_range));
|
|
Packit Service |
dff8e4 |
if (wext_get_range_ifname(wext, ifname, &range, &response_len) == FALSE) {
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI, "(%s): driver WEXT range request failed", ifname);
|
|
Packit Service |
dff8e4 |
goto error;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
if ((response_len < 300) || (range.we_version_compiled < 21)) {
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): driver WEXT version too old (got %d, expected >= 21)",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
range.we_version_compiled);
|
|
Packit Service |
dff8e4 |
goto error;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext->max_qual.qual = range.max_qual.qual;
|
|
Packit Service |
dff8e4 |
wext->max_qual.level = range.max_qual.level;
|
|
Packit Service |
dff8e4 |
wext->max_qual.noise = range.max_qual.noise;
|
|
Packit Service |
dff8e4 |
wext->max_qual.updated = range.max_qual.updated;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext->num_freqs = MIN(range.num_frequency, IW_MAX_FREQUENCIES);
|
|
Packit Service |
dff8e4 |
for (i = 0; i < wext->num_freqs; i++) {
|
|
Packit Service |
dff8e4 |
wext->freqs[i] = iw_freq_to_uint32(&range.freq[i]);
|
|
Packit Service |
dff8e4 |
freq_valid = TRUE;
|
|
Packit Service |
dff8e4 |
if (wext->freqs[i] > 2400 && wext->freqs[i] < 2500)
|
|
Packit Service |
dff8e4 |
has_2ghz = TRUE;
|
|
Packit Service |
dff8e4 |
else if (wext->freqs[i] > 4900 && wext->freqs[i] < 6000)
|
|
Packit Service |
dff8e4 |
has_5ghz = TRUE;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Check for scanning capability; cards that can't scan are not supported */
|
|
Packit Service |
dff8e4 |
if (check_scan && (wext_can_scan_ifname(wext, ifname) == FALSE)) {
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI, "(%s): drivers that cannot scan are unsupported", ifname);
|
|
Packit Service |
dff8e4 |
goto error;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* Check for the ability to scan specific SSIDs. Until the scan_capa
|
|
Packit Service |
dff8e4 |
* field gets added to wireless-tools, need to work around that by casting
|
|
Packit Service |
dff8e4 |
* to the custom structure.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
scan_capa_range = (struct iw_range_with_scan_capa *) ⦥
|
|
Packit Service |
dff8e4 |
if (scan_capa_range->scan_capa & NM_IW_SCAN_CAPA_ESSID) {
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): driver supports SSID scans (scan_capa 0x%02X).",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
scan_capa_range->scan_capa);
|
|
Packit Service |
dff8e4 |
} else {
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI,
|
|
Packit Service |
dff8e4 |
"(%s): driver does not support SSID scans (scan_capa 0x%02X).",
|
|
Packit Service |
dff8e4 |
ifname,
|
|
Packit Service |
dff8e4 |
scan_capa_range->scan_capa);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
wext->parent.caps = wext_get_caps(wext, ifname, &range);
|
|
Packit Service |
dff8e4 |
if (freq_valid)
|
|
Packit Service |
dff8e4 |
wext->parent.caps |= _NM_WIFI_DEVICE_CAP_FREQ_VALID;
|
|
Packit Service |
dff8e4 |
if (has_2ghz)
|
|
Packit Service |
dff8e4 |
wext->parent.caps |= _NM_WIFI_DEVICE_CAP_FREQ_2GHZ;
|
|
Packit Service |
dff8e4 |
if (has_5ghz)
|
|
Packit Service |
dff8e4 |
wext->parent.caps |= _NM_WIFI_DEVICE_CAP_FREQ_5GHZ;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
_LOGI(LOGD_PLATFORM | LOGD_WIFI, "(%s): using WEXT for Wi-Fi device control", ifname);
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
return (NMWifiUtils *) wext;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
error:
|
|
Packit Service |
dff8e4 |
g_object_unref(wext);
|
|
Packit Service |
dff8e4 |
return NULL;
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
gboolean
|
|
Packit Service |
dff8e4 |
nm_wifi_utils_wext_is_wifi(const char *iface)
|
|
Packit Service |
dff8e4 |
{
|
|
Packit Service |
dff8e4 |
int fd;
|
|
Packit Service |
dff8e4 |
struct iwreq iwr;
|
|
Packit Service |
dff8e4 |
gboolean is_wifi = FALSE;
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
/* performing an ioctl on a non-existing name may cause the automatic
|
|
Packit Service |
dff8e4 |
* loading of kernel modules, which should be avoided.
|
|
Packit Service |
dff8e4 |
*
|
|
Packit Service |
dff8e4 |
* Usually, we should thus make sure that an interface with this name
|
|
Packit Service |
dff8e4 |
* exists.
|
|
Packit Service |
dff8e4 |
*
|
|
Packit Service |
dff8e4 |
* Note that wifi_wext_is_wifi() has only one caller which just verified
|
|
Packit Service |
dff8e4 |
* that an interface with this name exists.
|
|
Packit Service |
dff8e4 |
*/
|
|
Packit Service |
dff8e4 |
|
|
Packit Service |
dff8e4 |
fd = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
|
|
Packit Service |
dff8e4 |
if (fd >= 0) {
|
|
Packit Service |
dff8e4 |
nm_utils_ifname_cpy(iwr.ifr_ifrn.ifrn_name, iface);
|
|
Packit Service |
dff8e4 |
if (ioctl(fd, SIOCGIWNAME, &iwr) == 0)
|
|
Packit Service |
dff8e4 |
is_wifi = TRUE;
|
|
Packit Service |
dff8e4 |
nm_close(fd);
|
|
Packit Service |
dff8e4 |
}
|
|
Packit Service |
dff8e4 |
return is_wifi;
|
|
Packit Service |
dff8e4 |
}
|