mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2026-02-05 08:50:27 +01:00
Now that NMDevice reads the hwaddr directly from netlink, it's silly to have every device subtype maintain its own hw-address property (using data that it gets from the NMDevice base class). Remove all the device-specific hw-address properties, and add one to NMDevice instead. (Because of the way nm-properties-changed-signal works, this has no effect on the D-Bus API.) Subclasses now call nm_device_get_hw_address() in places where they used to just refer to priv->hw_addr (and to simplify this, we now allow passing NULL for the out length parameter, since the subclasses almost always know what the length will be already). Also reorganize/simplify a few other methods to take advantage of the fact that NMDevice is now keeping track of the hw-address directly. https://bugzilla.gnome.org/show_bug.cgi?id=699391
1512 lines
42 KiB
C
1512 lines
42 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
/* NetworkManager -- Network link manager
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright (C) 2010 - 2011 Red Hat, Inc.
|
|
* Copyright (C) 2009 Novell, Inc.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <sys/ioctl.h>
|
|
#include <net/ethernet.h>
|
|
#include <sys/socket.h>
|
|
#include <linux/if.h>
|
|
#include <netinet/ether.h>
|
|
|
|
#include <WiMaxAPI.h>
|
|
#include <WiMaxAPIEx.h>
|
|
|
|
#include "nm-device-wimax.h"
|
|
#include "nm-wimax-util.h"
|
|
#include "nm-logging.h"
|
|
#include "nm-device-private.h"
|
|
#include "nm-system.h"
|
|
#include "NetworkManagerUtils.h"
|
|
#include "nm-properties-changed-signal.h"
|
|
#include "nm-connection.h"
|
|
#include "nm-setting-connection.h"
|
|
#include "nm-setting-wimax.h"
|
|
#include "nm-utils.h"
|
|
#include "nm-rfkill.h"
|
|
#include "iwmxsdk.h"
|
|
#include "nm-enum-types.h"
|
|
|
|
static gboolean impl_device_get_nsp_list (NMDeviceWimax *device, GPtrArray **list, GError **error);
|
|
|
|
#include "nm-device-wimax-glue.h"
|
|
|
|
G_DEFINE_TYPE (NMDeviceWimax, nm_device_wimax, NM_TYPE_DEVICE)
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_ACTIVE_NSP,
|
|
PROP_CENTER_FREQ,
|
|
PROP_RSSI,
|
|
PROP_CINR,
|
|
PROP_TX_POWER,
|
|
PROP_BSID,
|
|
|
|
LAST_PROP
|
|
};
|
|
|
|
enum {
|
|
NSP_ADDED,
|
|
NSP_REMOVED,
|
|
PROPERTIES_CHANGED,
|
|
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
#define NM_DEVICE_WIMAX_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), \
|
|
NM_TYPE_DEVICE_WIMAX, \
|
|
NMDeviceWimaxPrivate))
|
|
|
|
typedef struct {
|
|
gboolean disposed;
|
|
|
|
struct wmxsdk *sdk;
|
|
WIMAX_API_DEVICE_STATUS status;
|
|
gboolean connect_failed;
|
|
|
|
gboolean enabled;
|
|
gboolean wimaxd_enabled;
|
|
guint activation_timeout_id;
|
|
|
|
/* Track whether stage1 (Prepare) is completed yet or not */
|
|
gboolean prepare_done;
|
|
|
|
guint sdk_action_defer_id;
|
|
|
|
guint link_timeout_id;
|
|
guint poll_id;
|
|
|
|
GSList *nsp_list;
|
|
NMWimaxNsp *current_nsp;
|
|
|
|
/* interesting stuff when connected */
|
|
guint center_freq;
|
|
gint rssi;
|
|
gint cinr;
|
|
gint tx_power;
|
|
char *bsid;
|
|
} NMDeviceWimaxPrivate;
|
|
|
|
/***********************************************************/
|
|
|
|
#define NM_WIMAX_ERROR (nm_wimax_error_quark ())
|
|
|
|
static GQuark
|
|
nm_wimax_error_quark (void)
|
|
{
|
|
static GQuark quark = 0;
|
|
if (!quark)
|
|
quark = g_quark_from_static_string ("nm-wimax-error");
|
|
return quark;
|
|
}
|
|
|
|
/***********************************************************/
|
|
|
|
guint32
|
|
nm_device_wimax_get_center_frequency (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), 0);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->center_freq;
|
|
}
|
|
|
|
gint
|
|
nm_device_wimax_get_rssi (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), 0);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->rssi;
|
|
}
|
|
|
|
gint
|
|
nm_device_wimax_get_cinr (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), 0);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->cinr;
|
|
}
|
|
|
|
gint
|
|
nm_device_wimax_get_tx_power (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), 0);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->tx_power;
|
|
}
|
|
|
|
const char *
|
|
nm_device_wimax_get_bsid (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), NULL);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->bsid;
|
|
}
|
|
|
|
static gboolean
|
|
impl_device_get_nsp_list (NMDeviceWimax *self, GPtrArray **nsps, GError **error)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
GSList *iter;
|
|
|
|
*nsps = g_ptr_array_sized_new (g_slist_length (priv->nsp_list));
|
|
for (iter = priv->nsp_list; iter; iter = iter->next) {
|
|
const char *path;
|
|
|
|
path = nm_wimax_nsp_get_dbus_path (NM_WIMAX_NSP (iter->data));
|
|
if (path)
|
|
g_ptr_array_add (*nsps, g_strdup (path));
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
set_current_nsp (NMDeviceWimax *self, NMWimaxNsp *new_nsp)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
NMWimaxNsp *old_nsp;
|
|
gboolean path_changed = FALSE;
|
|
|
|
old_nsp = priv->current_nsp;
|
|
priv->current_nsp = NULL;
|
|
|
|
if (new_nsp)
|
|
priv->current_nsp = g_object_ref (new_nsp);
|
|
|
|
if (old_nsp && new_nsp) {
|
|
path_changed = (g_strcmp0 (nm_wimax_nsp_get_dbus_path (old_nsp),
|
|
nm_wimax_nsp_get_dbus_path (new_nsp)) != 0);
|
|
}
|
|
|
|
/* Only notify if it's really changed */
|
|
if (old_nsp != new_nsp || path_changed)
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_ACTIVE_NSP);
|
|
|
|
if (old_nsp)
|
|
g_object_unref (old_nsp);
|
|
}
|
|
|
|
NMWimaxNsp *
|
|
nm_device_wimax_get_active_nsp (NMDeviceWimax *self)
|
|
{
|
|
g_return_val_if_fail (NM_IS_DEVICE_WIMAX (self), NULL);
|
|
|
|
return NM_DEVICE_WIMAX_GET_PRIVATE (self)->current_nsp;
|
|
}
|
|
|
|
static gboolean
|
|
activation_timed_out (gpointer data)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (data);
|
|
|
|
priv->activation_timeout_id = 0;
|
|
nm_device_state_changed (NM_DEVICE (data), NM_DEVICE_STATE_FAILED, NM_DEVICE_STATE_REASON_CONFIG_FAILED);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
remove_all_nsps (NMDeviceWimax *self)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
set_current_nsp (self, NULL);
|
|
|
|
while (g_slist_length (priv->nsp_list)) {
|
|
NMWimaxNsp *nsp = NM_WIMAX_NSP (priv->nsp_list->data);
|
|
|
|
priv->nsp_list = g_slist_remove (priv->nsp_list, nsp);
|
|
g_signal_emit (self, signals[NSP_REMOVED], 0, nsp);
|
|
g_object_unref (nsp);
|
|
}
|
|
|
|
nm_device_recheck_available_connections (NM_DEVICE (self));
|
|
|
|
g_slist_free (priv->nsp_list);
|
|
priv->nsp_list = NULL;
|
|
}
|
|
|
|
static NMWimaxNsp *
|
|
get_nsp_by_name (NMDeviceWimax *self, const char *name)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
GSList *iter;
|
|
|
|
for (iter = priv->nsp_list; iter; iter = iter->next) {
|
|
NMWimaxNsp *nsp = NM_WIMAX_NSP (iter->data);
|
|
|
|
if (!g_strcmp0 (nm_wimax_nsp_get_name (nsp), name))
|
|
return nsp;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static gboolean
|
|
update_availability (NMDeviceWimax *self, gboolean old_available)
|
|
{
|
|
NMDevice *device = NM_DEVICE (self);
|
|
NMDeviceState state;
|
|
gboolean new_available, changed = FALSE;
|
|
|
|
new_available = nm_device_is_available (device);
|
|
if (new_available == old_available)
|
|
return FALSE;
|
|
|
|
state = nm_device_get_state (device);
|
|
if (state == NM_DEVICE_STATE_UNAVAILABLE) {
|
|
if (new_available == TRUE) {
|
|
nm_device_state_changed (device,
|
|
NM_DEVICE_STATE_DISCONNECTED,
|
|
NM_DEVICE_STATE_REASON_NONE);
|
|
changed = TRUE;
|
|
}
|
|
} else if (state >= NM_DEVICE_STATE_DISCONNECTED) {
|
|
if (new_available == FALSE) {
|
|
nm_device_state_changed (device,
|
|
NM_DEVICE_STATE_UNAVAILABLE,
|
|
NM_DEVICE_STATE_REASON_NONE);
|
|
changed = TRUE;
|
|
}
|
|
}
|
|
|
|
return changed;
|
|
}
|
|
|
|
/* NMDeviceInterface interface */
|
|
|
|
static void
|
|
set_enabled (NMDevice *device, gboolean enabled)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (device);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
gboolean old_available;
|
|
int ret;
|
|
const char *iface;
|
|
|
|
iface = nm_device_get_iface (NM_DEVICE (self));
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): setting radio enabled %d -> %d",
|
|
iface, priv->enabled, enabled);
|
|
if (priv->enabled == enabled)
|
|
return;
|
|
|
|
old_available = nm_device_is_available (NM_DEVICE (device));
|
|
priv->enabled = enabled;
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): radio now %s",
|
|
iface, priv->enabled ? "enabled" : "disabled");
|
|
|
|
/* Set the WiMAX device RF state to the current user-specified enabled state */
|
|
if (priv->sdk) {
|
|
ret = iwmx_sdk_rf_state_set (priv->sdk,
|
|
enabled ? WIMAX_API_RF_ON : WIMAX_API_RF_OFF);
|
|
if (ret < 0 && ret != -EINPROGRESS) {
|
|
nm_log_warn (LOGD_WIMAX, "(%s): failed to %s radio",
|
|
iface, priv->enabled ? "enable" : "disable");
|
|
}
|
|
}
|
|
|
|
update_availability (self, old_available);
|
|
}
|
|
|
|
/* NMDevice methods */
|
|
|
|
static void
|
|
take_down (NMDevice *device)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (device);
|
|
|
|
set_current_nsp (self, NULL);
|
|
remove_all_nsps (self);
|
|
}
|
|
|
|
static gboolean
|
|
hw_bring_up (NMDevice *dev, gboolean *no_firmware)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (dev);
|
|
|
|
if (!priv->enabled || !priv->wimaxd_enabled)
|
|
return FALSE;
|
|
|
|
return NM_DEVICE_GET_CLASS (dev)->hw_bring_up (dev, no_firmware);
|
|
}
|
|
|
|
static const GByteArray *
|
|
get_connection_hw_address (NMDevice *device,
|
|
NMConnection *connection)
|
|
{
|
|
NMSettingWimax *s_wimax;
|
|
|
|
s_wimax = nm_connection_get_setting_wimax (connection);
|
|
return s_wimax ? nm_setting_wimax_get_mac_address (s_wimax) : NULL;
|
|
}
|
|
|
|
static gboolean
|
|
check_connection_compatible (NMDevice *device,
|
|
NMConnection *connection,
|
|
GError **error)
|
|
{
|
|
NMSettingConnection *s_con;
|
|
NMSettingWimax *s_wimax;
|
|
const char *connection_type;
|
|
const GByteArray *mac;
|
|
|
|
if (!NM_DEVICE_CLASS (nm_device_wimax_parent_class)->check_connection_compatible (device, connection, error))
|
|
return FALSE;
|
|
|
|
s_con = nm_connection_get_setting_connection (connection);
|
|
g_assert (s_con);
|
|
|
|
connection_type = nm_setting_connection_get_connection_type (s_con);
|
|
if (strcmp (connection_type, NM_SETTING_WIMAX_SETTING_NAME)) {
|
|
g_set_error (error,
|
|
NM_WIMAX_ERROR, NM_WIMAX_ERROR_CONNECTION_NOT_WIMAX,
|
|
"The connection was not a WiMAX connection.");
|
|
return FALSE;
|
|
}
|
|
|
|
s_wimax = nm_connection_get_setting_wimax (connection);
|
|
if (!s_wimax) {
|
|
g_set_error (error,
|
|
NM_WIMAX_ERROR, NM_WIMAX_ERROR_CONNECTION_INVALID,
|
|
"The connection was not a valid WiMAX connection.");
|
|
return FALSE;
|
|
}
|
|
|
|
mac = nm_setting_wimax_get_mac_address (s_wimax);
|
|
if (mac && memcmp (mac->data, nm_device_get_hw_address (device, NULL), ETH_ALEN)) {
|
|
g_set_error (error,
|
|
NM_WIMAX_ERROR, NM_WIMAX_ERROR_CONNECTION_INCOMPATIBLE,
|
|
"The connection's MAC address did not match this device.");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
check_connection_available (NMDevice *device, NMConnection *connection)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (device);
|
|
const GSList *ns_iter = NULL;
|
|
|
|
/* Ensure the connection applies to an NSP in the scan list */
|
|
for (ns_iter = priv->nsp_list; ns_iter; ns_iter = ns_iter->next) {
|
|
if (nm_wimax_nsp_check_compatible (NM_WIMAX_NSP (ns_iter->data), connection))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
complete_connection (NMDevice *device,
|
|
NMConnection *connection,
|
|
const char *specific_object,
|
|
const GSList *existing_connections,
|
|
GError **error)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (device);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
NMSettingWimax *s_wimax;
|
|
const GByteArray *setting_mac;
|
|
const guint8 *hw_address;
|
|
char *format;
|
|
const char *nsp_name = NULL;
|
|
NMWimaxNsp *nsp = NULL;
|
|
GSList *iter;
|
|
|
|
s_wimax = nm_connection_get_setting_wimax (connection);
|
|
|
|
if (!specific_object) {
|
|
/* If not given a specific object, we need at minimum an NSP name */
|
|
if (!s_wimax) {
|
|
g_set_error_literal (error,
|
|
NM_WIMAX_ERROR,
|
|
NM_WIMAX_ERROR_CONNECTION_INVALID,
|
|
"A 'wimax' setting is required if no NSP path was given.");
|
|
return FALSE;
|
|
}
|
|
|
|
nsp_name = nm_setting_wimax_get_network_name (s_wimax);
|
|
if (!nsp_name || !strlen (nsp_name)) {
|
|
g_set_error_literal (error,
|
|
NM_WIMAX_ERROR,
|
|
NM_WIMAX_ERROR_CONNECTION_INVALID,
|
|
"A 'wimax' setting with a valid network name is required if no NSP path was given.");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Find a compatible NSP in the list */
|
|
nsp = get_nsp_by_name (self, nsp_name);
|
|
|
|
/* If we still don't have an NSP, then the WiMAX settings needs to be
|
|
* fully specified by the client. Might not be able to find the NSP
|
|
* if the scan didn't find the NSP yet.
|
|
*/
|
|
if (!nsp) {
|
|
if (!nm_setting_verify (NM_SETTING (s_wimax), NULL, error))
|
|
return FALSE;
|
|
}
|
|
} else {
|
|
/* Find a compatible NSP in the list */
|
|
for (iter = priv->nsp_list; iter; iter = g_slist_next (iter)) {
|
|
if (!strcmp (specific_object, nm_wimax_nsp_get_dbus_path (NM_WIMAX_NSP (iter->data)))) {
|
|
nsp = NM_WIMAX_NSP (iter->data);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!nsp) {
|
|
g_set_error (error,
|
|
NM_WIMAX_ERROR,
|
|
NM_WIMAX_ERROR_NSP_NOT_FOUND,
|
|
"The NSP %s was not in the scan list.",
|
|
specific_object);
|
|
return FALSE;
|
|
}
|
|
|
|
nsp_name = nm_wimax_nsp_get_name (nsp);
|
|
}
|
|
|
|
/* Add a WiMAX setting if one doesn't exist */
|
|
if (!s_wimax) {
|
|
s_wimax = (NMSettingWimax *) nm_setting_wimax_new ();
|
|
nm_connection_add_setting (connection, NM_SETTING (s_wimax));
|
|
}
|
|
|
|
g_assert (nsp_name);
|
|
format = g_strdup_printf ("%s %%d", nsp_name);
|
|
nm_utils_complete_generic (connection,
|
|
NM_SETTING_WIMAX_SETTING_NAME,
|
|
existing_connections,
|
|
format,
|
|
nsp_name,
|
|
TRUE);
|
|
g_free (format);
|
|
g_object_set (G_OBJECT (s_wimax), NM_SETTING_WIMAX_NETWORK_NAME, nsp_name, NULL);
|
|
|
|
setting_mac = nm_setting_wimax_get_mac_address (s_wimax);
|
|
hw_address = nm_device_get_hw_address (device, NULL);
|
|
if (setting_mac) {
|
|
/* Make sure the setting MAC (if any) matches the device's permanent MAC */
|
|
if (memcmp (setting_mac->data, hw_address, ETH_ALEN)) {
|
|
g_set_error (error,
|
|
NM_SETTING_WIMAX_ERROR,
|
|
NM_SETTING_WIMAX_ERROR_INVALID_PROPERTY,
|
|
NM_SETTING_WIMAX_MAC_ADDRESS);
|
|
return FALSE;
|
|
}
|
|
} else {
|
|
GByteArray *mac;
|
|
const guint8 null_mac[ETH_ALEN] = { 0, 0, 0, 0, 0, 0 };
|
|
|
|
/* Lock the connection to this device by default */
|
|
if (memcmp (hw_address, null_mac, ETH_ALEN)) {
|
|
mac = g_byte_array_sized_new (ETH_ALEN);
|
|
g_byte_array_append (mac, hw_address, ETH_ALEN);
|
|
g_object_set (G_OBJECT (s_wimax), NM_SETTING_WIMAX_MAC_ADDRESS, mac, NULL);
|
|
g_byte_array_free (mac, TRUE);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
can_auto_connect (NMDevice *device,
|
|
NMConnection *connection,
|
|
char **specific_object)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (device);
|
|
GSList *iter;
|
|
|
|
if (!NM_DEVICE_CLASS (nm_device_wimax_parent_class)->can_auto_connect (device, connection, specific_object))
|
|
return FALSE;
|
|
|
|
for (iter = priv->nsp_list; iter; iter = iter->next) {
|
|
NMWimaxNsp *nsp = NM_WIMAX_NSP (iter->data);
|
|
|
|
if (nm_wimax_nsp_check_compatible (nsp, connection)) {
|
|
*specific_object = (char *) nm_wimax_nsp_get_dbus_path (nsp);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static guint32
|
|
get_generic_capabilities (NMDevice *dev)
|
|
{
|
|
return NM_DEVICE_CAP_NM_SUPPORTED;
|
|
}
|
|
|
|
static gboolean
|
|
is_available (NMDevice *device)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (device);
|
|
const char *iface = nm_device_get_iface (device);
|
|
|
|
if (!priv->enabled) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): not available because not enabled", iface);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!priv->wimaxd_enabled) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): not available because not enabled in wimaxd", iface);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!nm_wimax_util_sdk_is_initialized ()) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): not available because WiMAX SDK not initialized", iface);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!priv->sdk) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): not available because not known to WiMAX SDK", iface);
|
|
return FALSE;
|
|
}
|
|
|
|
return iwmxsdk_status_get (priv->sdk) >= WIMAX_API_DEVICE_STATUS_Ready;
|
|
}
|
|
|
|
static void
|
|
clear_activation_timeout (NMDeviceWimax *self)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (priv->activation_timeout_id) {
|
|
g_source_remove (priv->activation_timeout_id);
|
|
priv->activation_timeout_id = 0;
|
|
}
|
|
|
|
priv->connect_failed = FALSE;
|
|
}
|
|
|
|
static void
|
|
clear_link_timeout (NMDeviceWimax *self)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (priv->link_timeout_id) {
|
|
g_source_remove (priv->link_timeout_id);
|
|
priv->link_timeout_id = 0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
clear_connected_poll (NMDeviceWimax *self)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (priv->poll_id) {
|
|
g_source_remove (priv->poll_id);
|
|
priv->poll_id = 0;
|
|
}
|
|
}
|
|
|
|
static NMActStageReturn
|
|
act_stage1_prepare (NMDevice *device, NMDeviceStateReason *reason)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (device);
|
|
NMActRequest *req;
|
|
GSList *iter;
|
|
const char *path;
|
|
NMWimaxNsp *nsp = NULL;
|
|
|
|
clear_link_timeout (NM_DEVICE_WIMAX (device));
|
|
|
|
*reason = NM_DEVICE_STATE_REASON_NONE;
|
|
|
|
req = nm_device_get_act_request (device);
|
|
if (!req)
|
|
return NM_ACT_STAGE_RETURN_FAILURE;
|
|
|
|
path = nm_active_connection_get_specific_object (NM_ACTIVE_CONNECTION (req));
|
|
if (!path)
|
|
return NM_ACT_STAGE_RETURN_FAILURE;
|
|
|
|
/* Find the NSP in the scan list */
|
|
for (iter = priv->nsp_list; iter; iter = iter->next) {
|
|
NMWimaxNsp *candidate = NM_WIMAX_NSP (iter->data);
|
|
|
|
if (!strcmp (path, nm_wimax_nsp_get_dbus_path (candidate))) {
|
|
nsp = candidate;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Couldn't find the NSP for some reason */
|
|
if (nsp == NULL)
|
|
return NM_ACT_STAGE_RETURN_FAILURE;
|
|
|
|
set_current_nsp (NM_DEVICE_WIMAX (device), nsp);
|
|
|
|
priv->prepare_done = TRUE;
|
|
|
|
/* If the device is scanning, it won't connect, so we have to wait until
|
|
* it's not scanning to proceed to stage 2.
|
|
*/
|
|
if (priv->status == WIMAX_API_DEVICE_STATUS_Scanning)
|
|
return NM_ACT_STAGE_RETURN_POSTPONE;
|
|
|
|
return NM_ACT_STAGE_RETURN_SUCCESS;
|
|
}
|
|
|
|
static NMActStageReturn
|
|
act_stage2_config (NMDevice *device, NMDeviceStateReason *reason)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (device);
|
|
NMConnection *connection;
|
|
NMSettingWimax *s_wimax;
|
|
const char *nsp_name, *iface;
|
|
int ret;
|
|
|
|
iface = nm_device_get_iface (device);
|
|
g_assert (iface);
|
|
|
|
connection = nm_device_get_connection (device);
|
|
g_assert (connection);
|
|
|
|
s_wimax = nm_connection_get_setting_wimax (connection);
|
|
g_assert (s_wimax);
|
|
|
|
nsp_name = nm_setting_wimax_get_network_name (s_wimax);
|
|
g_assert (nsp_name);
|
|
|
|
nm_log_info (LOGD_WIMAX, "(%s): connecting to NSP '%s'",
|
|
iface, nsp_name);
|
|
|
|
priv->connect_failed = FALSE;
|
|
ret = iwmx_sdk_connect (priv->sdk, nsp_name);
|
|
if (ret < 0 && ret != -EINPROGRESS) {
|
|
nm_log_err (LOGD_WIMAX, "(%s): failed to connect to NSP '%s'",
|
|
iface, nsp_name);
|
|
*reason = NM_DEVICE_STATE_REASON_CONFIG_FAILED;
|
|
return NM_ACT_STAGE_RETURN_FAILURE;
|
|
}
|
|
|
|
/* FIXME: Is 40 seconds good estimation? I have no idea */
|
|
priv->activation_timeout_id = g_timeout_add_seconds (40, activation_timed_out, device);
|
|
|
|
return NM_ACT_STAGE_RETURN_POSTPONE;
|
|
}
|
|
|
|
static void
|
|
force_disconnect (NMDeviceWimax *self, struct wmxsdk *sdk)
|
|
{
|
|
WIMAX_API_DEVICE_STATUS status;
|
|
int ret;
|
|
const char *iface;
|
|
|
|
g_return_if_fail (sdk != NULL);
|
|
|
|
iface = nm_device_get_iface (NM_DEVICE (self));
|
|
|
|
status = iwmxsdk_status_get (sdk);
|
|
if ((int) status < 0) {
|
|
nm_log_err (LOGD_WIMAX, "(%s): failed to read WiMAX device status: %d",
|
|
iface, status);
|
|
return;
|
|
}
|
|
|
|
if ( status == WIMAX_API_DEVICE_STATUS_Connecting
|
|
|| status == WIMAX_API_DEVICE_STATUS_Data_Connected) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): requesting disconnect", iface);
|
|
ret = iwmx_sdk_disconnect (sdk);
|
|
if (ret < 0 && ret != -EINPROGRESS) {
|
|
nm_log_err (LOGD_WIMAX, "(%s): failed to disconnect WiMAX device: %d",
|
|
iface, ret);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
deactivate (NMDevice *device)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (device);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
clear_activation_timeout (self);
|
|
clear_link_timeout (self);
|
|
clear_connected_poll (self);
|
|
|
|
set_current_nsp (self, NULL);
|
|
|
|
if (priv->sdk) {
|
|
/* Read explicit status here just to make sure we have the most
|
|
* up-to-date status and to ensure we disconnect if needed.
|
|
*/
|
|
force_disconnect (self, priv->sdk);
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
static void
|
|
wmx_state_change_cb (struct wmxsdk *wmxsdk,
|
|
WIMAX_API_DEVICE_STATUS new_status,
|
|
WIMAX_API_DEVICE_STATUS old_status,
|
|
WIMAX_API_STATUS_REASON reason,
|
|
WIMAX_API_CONNECTION_PROGRESS_INFO progress,
|
|
void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
NMDeviceState state;
|
|
const char *iface;
|
|
gboolean old_available = FALSE;
|
|
const char *nsp_name = NULL;
|
|
|
|
iface = nm_device_get_iface (NM_DEVICE (self));
|
|
nm_log_info (LOGD_WIMAX, "(%s): wimax state change %s -> %s (%s (%d))",
|
|
iface,
|
|
iwmx_sdk_dev_status_to_str (old_status),
|
|
iwmx_sdk_dev_status_to_str (new_status),
|
|
iwmx_sdk_con_progress_to_str (progress),
|
|
progress);
|
|
|
|
if (new_status == old_status)
|
|
return;
|
|
|
|
state = nm_device_get_state (NM_DEVICE (self));
|
|
old_available = nm_device_is_available (NM_DEVICE (self));
|
|
|
|
priv->status = new_status;
|
|
if (priv->current_nsp)
|
|
nsp_name = nm_wimax_nsp_get_name (priv->current_nsp);
|
|
|
|
switch (new_status) {
|
|
case WIMAX_API_DEVICE_STATUS_UnInitialized:
|
|
case WIMAX_API_DEVICE_STATUS_RF_OFF_HW_SW:
|
|
case WIMAX_API_DEVICE_STATUS_RF_OFF_HW:
|
|
case WIMAX_API_DEVICE_STATUS_RF_OFF_SW:
|
|
if (priv->wimaxd_enabled) {
|
|
priv->wimaxd_enabled = FALSE;
|
|
if (update_availability (self, old_available))
|
|
return;
|
|
}
|
|
break;
|
|
case WIMAX_API_DEVICE_STATUS_Connecting:
|
|
case WIMAX_API_DEVICE_STATUS_Data_Connected:
|
|
/* If for some reason we're initially connected, force a disconnect here */
|
|
if (state < NM_DEVICE_STATE_DISCONNECTED)
|
|
force_disconnect (self, wmxsdk);
|
|
/* Fall through */
|
|
case WIMAX_API_DEVICE_STATUS_Ready:
|
|
case WIMAX_API_DEVICE_STATUS_Scanning:
|
|
if (priv->wimaxd_enabled == FALSE) {
|
|
priv->wimaxd_enabled = TRUE;
|
|
if (update_availability (self, old_available))
|
|
return;
|
|
}
|
|
break;
|
|
default:
|
|
nm_log_warn (LOGD_WIMAX, "(%s): unhandled WiMAX device state %d",
|
|
iface, new_status);
|
|
break;
|
|
}
|
|
|
|
/* Handle activation success and failure */
|
|
if (nm_device_is_activating (NM_DEVICE (self))) {
|
|
if (new_status == WIMAX_API_DEVICE_STATUS_Data_Connected) {
|
|
/* Success */
|
|
clear_activation_timeout (self);
|
|
|
|
nm_log_info (LOGD_WIMAX, "(%s): connected to '%s'",
|
|
iface, nsp_name);
|
|
nm_device_activate_schedule_stage3_ip_config_start (NM_DEVICE (self));
|
|
return;
|
|
}
|
|
|
|
if (priv->connect_failed) {
|
|
/* Connection attempt failed */
|
|
nm_log_info (LOGD_WIMAX, "(%s): connection to '%s' failed: (%d) %s",
|
|
iface, nsp_name, reason, iwmx_sdk_reason_to_str (reason));
|
|
nm_device_state_changed (NM_DEVICE (self),
|
|
NM_DEVICE_STATE_FAILED,
|
|
NM_DEVICE_STATE_REASON_CONFIG_FAILED);
|
|
return;
|
|
}
|
|
|
|
/* If stage2 was postponed because the device was scanning or something,
|
|
* then check if we need to move to stage2 now that the device might be
|
|
* ready.
|
|
*/
|
|
if (state == NM_DEVICE_STATE_PREPARE && priv->prepare_done) {
|
|
if ( new_status == WIMAX_API_DEVICE_STATUS_Ready
|
|
|| new_status == WIMAX_API_DEVICE_STATUS_Connecting) {
|
|
nm_device_activate_schedule_stage2_device_config (NM_DEVICE (self));
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Handle disconnection */
|
|
if (state == NM_DEVICE_STATE_ACTIVATED) {
|
|
if ( old_status == WIMAX_API_DEVICE_STATUS_Data_Connected
|
|
&& new_status < WIMAX_API_DEVICE_STATUS_Connecting) {
|
|
|
|
nm_log_info (LOGD_WIMAX, "(%s): disconnected from '%s': (%d) %s",
|
|
iface, nsp_name, reason, iwmx_sdk_reason_to_str (reason));
|
|
|
|
nm_device_state_changed (NM_DEVICE (self),
|
|
NM_DEVICE_STATE_FAILED,
|
|
NM_DEVICE_STATE_REASON_CONFIG_FAILED);
|
|
}
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
link_timeout_cb (gpointer user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
priv->link_timeout_id = 0;
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): link timed out", nm_device_get_iface (NM_DEVICE (self)));
|
|
nm_device_state_changed (NM_DEVICE (self),
|
|
NM_DEVICE_STATE_FAILED,
|
|
NM_DEVICE_STATE_REASON_CARRIER);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
wmx_media_status_cb (struct wmxsdk *wmxsdk,
|
|
WIMAX_API_MEDIA_STATUS new_status,
|
|
void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
NMDeviceState state;
|
|
const char *iface;
|
|
|
|
iface = nm_device_get_iface (NM_DEVICE (self));
|
|
state = nm_device_get_state (NM_DEVICE (self));
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): media status change to %s",
|
|
iface, iwmx_sdk_media_status_to_str (new_status));
|
|
|
|
/* We only care about media events while activated */
|
|
if (state != NM_DEVICE_STATE_ACTIVATED)
|
|
return;
|
|
|
|
clear_link_timeout (self);
|
|
|
|
switch (new_status) {
|
|
case WIMAX_API_MEDIA_STATUS_LINK_UP:
|
|
break;
|
|
case WIMAX_API_MEDIA_STATUS_LINK_DOWN:
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): starting link timeout", iface);
|
|
priv->link_timeout_id = g_timeout_add_seconds (15, link_timeout_cb, self);
|
|
break;
|
|
case WIMAX_API_MEDIA_STATUS_LINK_RENEW:
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): renewing DHCP lease", iface);
|
|
if (!nm_device_dhcp4_renew (NM_DEVICE (self), TRUE)) {
|
|
nm_device_state_changed (NM_DEVICE (self),
|
|
NM_DEVICE_STATE_FAILED,
|
|
NM_DEVICE_STATE_REASON_DHCP_FAILED);
|
|
}
|
|
break;
|
|
default:
|
|
nm_log_err (LOGD_WIMAX, "(%s): unhandled media status %d", iface, new_status);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
wmx_connect_result_cb (struct wmxsdk *wmxsdk,
|
|
WIMAX_API_NETWORK_CONNECTION_RESP result,
|
|
void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (nm_device_is_activating (NM_DEVICE (self))) {
|
|
priv->connect_failed = (result == WIMAX_API_CONNECTION_SUCCESS);
|
|
/* Wait for the state change so we can get the reason code; we
|
|
* cache the connect failure so we don't have to wait for the
|
|
* activation timeout.
|
|
*/
|
|
}
|
|
}
|
|
|
|
static void
|
|
remove_outdated_nsps (NMDeviceWimax *self,
|
|
WIMAX_API_NSP_INFO_EX *nsp_list,
|
|
guint32 list_size)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
GSList *iter;
|
|
GSList *to_remove = NULL;
|
|
|
|
for (iter = priv->nsp_list; iter; iter = iter->next) {
|
|
NMWimaxNsp *nsp = NM_WIMAX_NSP (iter->data);
|
|
gboolean found = FALSE;
|
|
int i;
|
|
|
|
for (i = 0; i < list_size; i++) {
|
|
WIMAX_API_NSP_INFO_EX *info = &nsp_list[i];
|
|
|
|
if (!g_strcmp0 (nm_wimax_nsp_get_name (nsp), (char *) info->NSPName)) {
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
to_remove = g_slist_prepend (to_remove, nsp);
|
|
}
|
|
|
|
for (iter = to_remove; iter; iter = iter->next) {
|
|
NMWimaxNsp *nsp = NM_WIMAX_NSP (iter->data);
|
|
|
|
g_signal_emit (self, signals[NSP_REMOVED], 0, nsp);
|
|
priv->nsp_list = g_slist_remove (priv->nsp_list, nsp);
|
|
g_object_unref (nsp);
|
|
}
|
|
|
|
if (g_slist_length(to_remove) > 0)
|
|
nm_device_recheck_available_connections (NM_DEVICE (self));
|
|
|
|
g_slist_free (to_remove);
|
|
}
|
|
|
|
static void
|
|
wmx_scan_result_cb (struct wmxsdk *wmxsdk,
|
|
WIMAX_API_NSP_INFO_EX *nsps,
|
|
guint num_nsps,
|
|
void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
const char *iface = nm_device_get_iface (NM_DEVICE (self));
|
|
int i;
|
|
|
|
remove_outdated_nsps (self, nsps, num_nsps);
|
|
|
|
/* Add new NSPs and update existing ones */
|
|
for (i = 0; i < num_nsps; i++) {
|
|
WIMAX_API_NSP_INFO_EX *sdk_nsp = &nsps[i];
|
|
const char *nsp_name = (const char *) sdk_nsp->NSPName;
|
|
NMWimaxNspNetworkType net_type;
|
|
guint signalq;
|
|
NMWimaxNsp *nsp;
|
|
gboolean new_nsp;
|
|
|
|
nsp = get_nsp_by_name (self, nsp_name);
|
|
new_nsp = (nsp == NULL);
|
|
if (new_nsp) {
|
|
nsp = nm_wimax_nsp_new (nsp_name);
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): new WiMAX NSP '%s'", iface, nsp_name);
|
|
}
|
|
|
|
net_type = nm_wimax_util_convert_network_type (sdk_nsp->networkType);
|
|
if (net_type != nm_wimax_nsp_get_network_type (nsp))
|
|
g_object_set (nsp, NM_WIMAX_NSP_NETWORK_TYPE, net_type, NULL);
|
|
|
|
signalq = CLAMP (sdk_nsp->linkQuality, 0, 100);
|
|
if (signalq != nm_wimax_nsp_get_signal_quality (nsp))
|
|
g_object_set (nsp, NM_WIMAX_NSP_SIGNAL_QUALITY, signalq, NULL);
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): WiMAX NSP '%s' quality %d%% type %d",
|
|
iface, nsp_name, sdk_nsp->linkQuality, net_type);
|
|
|
|
if (new_nsp) {
|
|
priv->nsp_list = g_slist_append (priv->nsp_list, nsp);
|
|
nm_wimax_nsp_export_to_dbus (nsp);
|
|
g_signal_emit (self, signals[NSP_ADDED], 0, nsp);
|
|
nm_device_recheck_available_connections (NM_DEVICE (self));
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
wmx_removed_cb (struct wmxsdk *wmxsdk, void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (!priv->sdk) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): removed unhandled WiMAX interface", wmxsdk->ifname);
|
|
return;
|
|
}
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): removed WiMAX interface", wmxsdk->ifname);
|
|
|
|
/* Clear callbacks just in case we don't hold the last reference */
|
|
iwmx_sdk_set_callbacks (priv->sdk, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
wmxsdk_unref (priv->sdk);
|
|
priv->sdk = NULL;
|
|
|
|
priv->status = WIMAX_API_DEVICE_STATUS_UnInitialized;
|
|
nm_device_state_changed (NM_DEVICE (self),
|
|
NM_DEVICE_STATE_UNAVAILABLE,
|
|
NM_DEVICE_STATE_REASON_NONE);
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
static inline gint
|
|
sdk_rssi_to_dbm (guint raw_rssi)
|
|
{
|
|
/* Values range from 0x00 to 0x53, where -123dBm is encoded as 0x00 and
|
|
* -40dBm encoded as 0x53 in 1dB increments.
|
|
*/
|
|
return raw_rssi - 123;
|
|
}
|
|
|
|
static inline gint
|
|
sdk_cinr_to_db (guint raw_cinr)
|
|
{
|
|
/* Values range from 0x00 to 0x3F, where -10dB is encoded as 0x00 and
|
|
* 53dB encoded as 0x3F in 1dB increments.
|
|
*/
|
|
return raw_cinr - 10;
|
|
}
|
|
|
|
static inline gint
|
|
sdk_tx_pow_to_dbm (guint raw_tx_pow)
|
|
{
|
|
/* Values range from 0x00 to 0xFF, where -84dBm is encoded as 0x00 and
|
|
* 43.5dBm is encoded as 0xFF in 0.5dB increments. Normalize so that
|
|
* 0 dBm == 0.
|
|
*/
|
|
return (int) (((double) raw_tx_pow / 2.0) - 84) * 2;
|
|
}
|
|
|
|
static void
|
|
set_link_status (NMDeviceWimax *self, WIMAX_API_LINK_STATUS_INFO_EX *link_status)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
guint center_freq = 0;
|
|
gint conv_rssi = 0, conv_cinr = 0, conv_tx_pow = 0;
|
|
char *new_bsid = NULL;
|
|
|
|
if (link_status) {
|
|
center_freq = link_status->centerFrequency;
|
|
conv_rssi = sdk_rssi_to_dbm (link_status->RSSI);
|
|
conv_cinr = sdk_cinr_to_db (link_status->CINR);
|
|
conv_tx_pow = sdk_tx_pow_to_dbm (link_status->txPWR);
|
|
new_bsid = g_strdup_printf ("%02X:%02X:%02X:%02X:%02X:%02X",
|
|
link_status->bsId[0], link_status->bsId[1],
|
|
link_status->bsId[2], link_status->bsId[3],
|
|
link_status->bsId[4], link_status->bsId[5]);
|
|
}
|
|
|
|
if (priv->center_freq != center_freq) {
|
|
priv->center_freq = center_freq;
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_CENTER_FREQUENCY);
|
|
}
|
|
|
|
if (priv->rssi != conv_rssi) {
|
|
priv->rssi = conv_rssi;
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_RSSI);
|
|
}
|
|
|
|
if (priv->cinr != conv_cinr) {
|
|
priv->cinr = conv_cinr;
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_CINR);
|
|
}
|
|
|
|
if (priv->tx_power != conv_tx_pow) {
|
|
priv->tx_power = conv_tx_pow;
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_TX_POWER);
|
|
}
|
|
|
|
if (g_strcmp0 (priv->bsid, new_bsid) != 0) {
|
|
g_free (priv->bsid);
|
|
priv->bsid = new_bsid;
|
|
g_object_notify (G_OBJECT (self), NM_DEVICE_WIMAX_BSID);
|
|
} else
|
|
g_free (new_bsid);
|
|
}
|
|
|
|
static gboolean
|
|
connected_poll_cb (gpointer user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
WIMAX_API_CONNECTED_NSP_INFO_EX *sdk_nsp;
|
|
WIMAX_API_LINK_STATUS_INFO_EX *link_status;
|
|
|
|
g_return_val_if_fail (priv->sdk != NULL, FALSE);
|
|
|
|
/* Get details of the connected NSP */
|
|
sdk_nsp = iwmx_sdk_get_connected_network (priv->sdk);
|
|
if (sdk_nsp) {
|
|
const char *nsp_name = (const char *) sdk_nsp->NSPName;
|
|
NMWimaxNsp *nsp;
|
|
|
|
nsp = get_nsp_by_name (self, nsp_name);
|
|
if (nsp) {
|
|
NMWimaxNspNetworkType net_type;
|
|
guint signalq;
|
|
|
|
net_type = nm_wimax_util_convert_network_type (sdk_nsp->networkType);
|
|
if (net_type != nm_wimax_nsp_get_network_type (nsp))
|
|
g_object_set (nsp, NM_WIMAX_NSP_NETWORK_TYPE, net_type, NULL);
|
|
|
|
signalq = sdk_nsp->linkQuality;
|
|
if (signalq != nm_wimax_nsp_get_signal_quality (nsp))
|
|
g_object_set (nsp, NM_WIMAX_NSP_SIGNAL_QUALITY, signalq, NULL);
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): WiMAX NSP '%s' quality %d%% type %d",
|
|
nm_device_get_iface (NM_DEVICE (self)),
|
|
nsp_name, sdk_nsp->linkQuality, net_type);
|
|
}
|
|
free (sdk_nsp);
|
|
}
|
|
|
|
/* Get details of the current radio link */
|
|
link_status = iwmx_sdk_get_link_status_info (priv->sdk);
|
|
if (link_status) {
|
|
set_link_status (self, link_status);
|
|
free (link_status);
|
|
}
|
|
|
|
return TRUE; /* reschedule */
|
|
}
|
|
|
|
static void
|
|
device_state_changed (NMDevice *device,
|
|
NMDeviceState new_state,
|
|
NMDeviceState old_state,
|
|
NMDeviceStateReason reason)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (device);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
/* Reset our stage1 (Prepare) done marker since it's only valid while in stage1 */
|
|
priv->prepare_done = FALSE;
|
|
|
|
if (new_state < NM_DEVICE_STATE_DISCONNECTED)
|
|
remove_all_nsps (self);
|
|
|
|
/* Request initial NSP list when device is first started */
|
|
if ( new_state == NM_DEVICE_STATE_DISCONNECTED
|
|
&& old_state < NM_DEVICE_STATE_DISCONNECTED) {
|
|
if (priv->sdk)
|
|
iwmx_sdk_get_networks (priv->sdk);
|
|
}
|
|
|
|
if (new_state == NM_DEVICE_STATE_FAILED || new_state <= NM_DEVICE_STATE_DISCONNECTED) {
|
|
set_current_nsp (self, NULL);
|
|
clear_activation_timeout (self);
|
|
}
|
|
|
|
if (new_state == NM_DEVICE_STATE_ACTIVATED) {
|
|
/* poll link quality and BSID */
|
|
clear_connected_poll (self);
|
|
priv->poll_id = g_timeout_add_seconds (10, connected_poll_cb, self);
|
|
connected_poll_cb (self);
|
|
} else {
|
|
clear_link_timeout (self);
|
|
clear_connected_poll (self);
|
|
set_link_status (self, NULL);
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
static gboolean
|
|
sdk_action_defer_cb (gpointer user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
gboolean old_available = nm_device_is_available (NM_DEVICE (self));
|
|
|
|
NM_DEVICE_WIMAX_GET_PRIVATE (self)->sdk_action_defer_id = 0;
|
|
update_availability (self, old_available);
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
wmx_new_sdk_cb (struct wmxsdk *sdk, void *user_data)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (user_data);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
/* We only track one wmxsdk at a time because the WiMAX SDK is pretty stupid */
|
|
if (priv->sdk) {
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): WiMAX interface already known", sdk->ifname);
|
|
return;
|
|
}
|
|
|
|
nm_log_dbg (LOGD_WIMAX, "(%s): new WiMAX interface (%s)", sdk->ifname, sdk->name);
|
|
|
|
/* Now that we have an SDK, schedule an idle handler to start the device up */
|
|
priv->sdk = wmxsdk_ref (sdk);
|
|
iwmx_sdk_set_callbacks(priv->sdk,
|
|
wmx_state_change_cb,
|
|
wmx_media_status_cb,
|
|
wmx_connect_result_cb,
|
|
wmx_scan_result_cb,
|
|
wmx_removed_cb,
|
|
self);
|
|
iwmx_sdk_set_fast_reconnect_enabled (priv->sdk, 0);
|
|
|
|
if (!priv->sdk_action_defer_id)
|
|
priv->sdk_action_defer_id = g_idle_add (sdk_action_defer_cb, self);
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
NMDevice *
|
|
nm_device_wimax_new (const char *udi,
|
|
const char *iface,
|
|
const char *driver)
|
|
{
|
|
NMDevice *device;
|
|
|
|
g_return_val_if_fail (udi != NULL, NULL);
|
|
g_return_val_if_fail (iface != NULL, NULL);
|
|
g_return_val_if_fail (driver != NULL, NULL);
|
|
|
|
device = (NMDevice *) g_object_new (NM_TYPE_DEVICE_WIMAX,
|
|
NM_DEVICE_UDI, udi,
|
|
NM_DEVICE_IFACE, iface,
|
|
NM_DEVICE_DRIVER, driver,
|
|
NM_DEVICE_TYPE_DESC, "WiMAX",
|
|
NM_DEVICE_DEVICE_TYPE, NM_DEVICE_TYPE_WIMAX,
|
|
NM_DEVICE_RFKILL_TYPE, RFKILL_TYPE_WIMAX,
|
|
NULL);
|
|
if (device) {
|
|
struct wmxsdk *sdk;
|
|
|
|
nm_wimax_util_sdk_ref ();
|
|
|
|
/* See if the SDK already knows about this interface */
|
|
sdk = iwmx_sdk_get_wmxsdk_for_iface (iface);
|
|
if (sdk)
|
|
wmx_new_sdk_cb (sdk, device);
|
|
|
|
/* If it doesn't, we want to be notified when it does */
|
|
iwmx_sdk_new_callback_register (wmx_new_sdk_cb, device);
|
|
}
|
|
|
|
return device;
|
|
}
|
|
|
|
static void
|
|
nm_device_wimax_init (NMDeviceWimax *self)
|
|
{
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
priv->status = WIMAX_API_DEVICE_STATUS_UnInitialized;
|
|
}
|
|
|
|
static void
|
|
set_property (GObject *object, guint prop_id,
|
|
const GValue *value, GParamSpec *pspec)
|
|
{
|
|
switch (prop_id) {
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
get_property (GObject *object, guint prop_id,
|
|
GValue *value, GParamSpec *pspec)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (object);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
switch (prop_id) {
|
|
case PROP_ACTIVE_NSP:
|
|
if (priv->current_nsp)
|
|
g_value_set_boxed (value, nm_wimax_nsp_get_dbus_path (priv->current_nsp));
|
|
else
|
|
g_value_set_boxed (value, "/");
|
|
break;
|
|
case PROP_CENTER_FREQ:
|
|
g_value_set_uint (value, priv->center_freq);
|
|
break;
|
|
case PROP_RSSI:
|
|
g_value_set_int (value, priv->rssi);
|
|
break;
|
|
case PROP_CINR:
|
|
g_value_set_int (value, priv->cinr);
|
|
break;
|
|
case PROP_TX_POWER:
|
|
g_value_set_int (value, priv->tx_power);
|
|
break;
|
|
case PROP_BSID:
|
|
g_value_set_string (value, priv->bsid);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
dispose (GObject *object)
|
|
{
|
|
NMDeviceWimax *self = NM_DEVICE_WIMAX (object);
|
|
NMDeviceWimaxPrivate *priv = NM_DEVICE_WIMAX_GET_PRIVATE (self);
|
|
|
|
if (priv->disposed)
|
|
goto done;
|
|
|
|
priv->disposed = TRUE;
|
|
|
|
clear_activation_timeout (self);
|
|
clear_link_timeout (self);
|
|
clear_connected_poll (self);
|
|
|
|
if (priv->sdk_action_defer_id)
|
|
g_source_remove (priv->sdk_action_defer_id);
|
|
|
|
if (priv->sdk) {
|
|
iwmx_sdk_set_callbacks (priv->sdk, NULL, NULL, NULL, NULL, NULL, NULL);
|
|
wmxsdk_unref (priv->sdk);
|
|
}
|
|
|
|
g_free (priv->bsid);
|
|
|
|
set_current_nsp (self, NULL);
|
|
|
|
g_slist_foreach (priv->nsp_list, (GFunc) g_object_unref, NULL);
|
|
g_slist_free (priv->nsp_list);
|
|
|
|
iwmx_sdk_new_callback_unregister (wmx_new_sdk_cb, self);
|
|
nm_wimax_util_sdk_unref ();
|
|
|
|
done:
|
|
G_OBJECT_CLASS (nm_device_wimax_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
nm_device_wimax_class_init (NMDeviceWimaxClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
NMDeviceClass *device_class = NM_DEVICE_CLASS (klass);
|
|
|
|
g_type_class_add_private (object_class, sizeof (NMDeviceWimaxPrivate));
|
|
|
|
/* Virtual methods */
|
|
object_class->set_property = set_property;
|
|
object_class->get_property = get_property;
|
|
object_class->dispose = dispose;
|
|
|
|
device_class->take_down = take_down;
|
|
device_class->hw_bring_up = hw_bring_up;
|
|
device_class->check_connection_compatible = check_connection_compatible;
|
|
device_class->check_connection_available = check_connection_available;
|
|
device_class->complete_connection = complete_connection;
|
|
device_class->can_auto_connect = can_auto_connect;
|
|
device_class->get_generic_capabilities = get_generic_capabilities;
|
|
device_class->is_available = is_available;
|
|
device_class->act_stage1_prepare = act_stage1_prepare;
|
|
device_class->act_stage2_config = act_stage2_config;
|
|
device_class->deactivate = deactivate;
|
|
device_class->set_enabled = set_enabled;
|
|
device_class->get_connection_hw_address = get_connection_hw_address;
|
|
|
|
device_class->state_changed = device_state_changed;
|
|
|
|
/* Properties */
|
|
g_object_class_install_property (object_class, PROP_ACTIVE_NSP,
|
|
g_param_spec_boxed (NM_DEVICE_WIMAX_ACTIVE_NSP,
|
|
"Active NSP",
|
|
"Currently active NSP",
|
|
DBUS_TYPE_G_OBJECT_PATH,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property
|
|
(object_class, PROP_CENTER_FREQ,
|
|
g_param_spec_uint (NM_DEVICE_WIMAX_CENTER_FREQUENCY,
|
|
"Center frequency",
|
|
"Center frequency",
|
|
0, G_MAXUINT, 0,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property
|
|
(object_class, PROP_RSSI,
|
|
g_param_spec_int (NM_DEVICE_WIMAX_RSSI,
|
|
"RSSI",
|
|
"RSSI",
|
|
G_MININT, G_MAXINT, 0,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property
|
|
(object_class, PROP_CINR,
|
|
g_param_spec_int (NM_DEVICE_WIMAX_CINR,
|
|
"CINR",
|
|
"CINR",
|
|
G_MININT, G_MAXINT, 0,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property
|
|
(object_class, PROP_TX_POWER,
|
|
g_param_spec_int (NM_DEVICE_WIMAX_TX_POWER,
|
|
"TX Power",
|
|
"TX Power",
|
|
G_MININT, G_MAXINT, 0,
|
|
G_PARAM_READABLE));
|
|
|
|
g_object_class_install_property
|
|
(object_class, PROP_BSID,
|
|
g_param_spec_string (NM_DEVICE_WIMAX_BSID,
|
|
"BSID",
|
|
"BSID",
|
|
NULL,
|
|
G_PARAM_READABLE));
|
|
|
|
/* Signals */
|
|
signals[NSP_ADDED] =
|
|
g_signal_new ("nsp-added",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (NMDeviceWimaxClass, nsp_added),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__OBJECT,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_OBJECT);
|
|
|
|
signals[NSP_REMOVED] =
|
|
g_signal_new ("nsp-removed",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (NMDeviceWimaxClass, nsp_removed),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__OBJECT,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_OBJECT);
|
|
|
|
signals[PROPERTIES_CHANGED] =
|
|
nm_properties_changed_signal_new (object_class, G_STRUCT_OFFSET (NMDeviceWimaxClass, properties_changed));
|
|
|
|
|
|
dbus_g_object_type_install_info (G_TYPE_FROM_CLASS (klass),
|
|
&dbus_glib_nm_device_wimax_object_info);
|
|
|
|
dbus_g_error_domain_register (NM_WIMAX_ERROR, NULL, NM_TYPE_WIMAX_ERROR);
|
|
}
|