asynchronous and deferred port detection

Allow plugins to perform asynchronous port detection, and to defer port detection
until later.  This moves the prober bits into MMPluginBase so that all plugins
can take adavantage of it only when needed; the probing is not done at udev time.
Furthermore, plugins like Novatel can flip the secondary ports over the AT mode
through	deferred detection, by deferring the secondary ports until the main port
has been detected and AT$NWDMAT	has been sent.

This commit also finishes the port of the rest of the plugins (except mbm) over
to the new port detection methods and plugin API.
This commit is contained in:
Dan Williams
2009-06-28 14:05:05 -04:00
parent 112f2da19d
commit 6077763d90
47 changed files with 2972 additions and 2597 deletions

View File

@@ -36,6 +36,8 @@ typedef struct {
GUdevClient *udev;
GSList *plugins;
GHashTable *modems;
GHashTable *supports;
} MMManagerPrivate;
static MMPlugin *
@@ -99,6 +101,7 @@ load_plugins (MMManager *manager)
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (manager);
GDir *dir;
const char *fname;
MMPlugin *generic_plugin = NULL;
if (!g_module_supported ()) {
g_warning ("GModules are not supported on your platform!");
@@ -122,10 +125,18 @@ load_plugins (MMManager *manager)
plugin = load_plugin (path);
g_free (path);
if (plugin)
priv->plugins = g_slist_append (priv->plugins, plugin);
if (plugin) {
if (!strcmp (mm_plugin_get_name (plugin), MM_PLUGIN_GENERIC_NAME))
generic_plugin = plugin;
else
priv->plugins = g_slist_append (priv->plugins, plugin);
}
}
/* Make sure the generic plugin is last */
if (generic_plugin)
priv->plugins = g_slist_append (priv->plugins, generic_plugin);
g_dir_close (dir);
}
@@ -179,7 +190,7 @@ modem_valid (MMModem *modem, GParamSpec *pspec, gpointer user_data)
device = mm_modem_get_device (modem);
g_assert (device);
g_debug ("Exported modem %s", device);
g_debug ("Exported modem %s as %s", device, path);
g_free (device);
g_signal_emit (manager, signals[DEVICE_ADDED], 0, modem);
@@ -236,107 +247,281 @@ impl_manager_enumerate_devices (MMManager *manager,
return TRUE;
}
typedef struct {
MMModem *modem;
const char *subsys;
const char *name;
} FindPortInfo;
static void
find_port (gpointer key, gpointer data, gpointer user_data)
{
FindPortInfo *info = user_data;
MMModem *modem = MM_MODEM (data);
if (!info->modem && mm_modem_owns_port (modem, info->subsys, info->name))
info->modem = modem;
}
static MMModem *
find_modem_for_port (MMManager *manager, const char *subsys, const char *name)
{
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (manager);
FindPortInfo info = { NULL, subsys, name };
GHashTableIter iter;
gpointer key, value;
g_hash_table_foreach (priv->modems, find_port, &info);
return info.modem;
g_hash_table_iter_init (&iter, priv->modems);
while (g_hash_table_iter_next (&iter, &key, &value)) {
MMModem *modem = MM_MODEM (value);
if (mm_modem_owns_port (modem, subsys, name))
return modem;
}
return NULL;
}
typedef struct {
MMManager *manager;
char *subsys;
char *name;
GSList *plugins;
GSList *cur_plugin;
guint defer_id;
guint32 best_level;
MMPlugin *best_plugin;
} SupportsInfo;
static SupportsInfo *
supports_info_new (MMManager *self, const char *subsys, const char *name)
{
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (self);
SupportsInfo *info;
info = g_malloc0 (sizeof (SupportsInfo));
info->manager = self;
info->subsys = g_strdup (subsys);
info->name = g_strdup (name);
info->plugins = g_slist_copy (priv->plugins);
info->cur_plugin = info->plugins;
return info;
}
static void
supports_info_free (SupportsInfo *info)
{
/* Cancel any in-process operation on the first plugin */
if (info->cur_plugin)
mm_plugin_cancel_supports_port (MM_PLUGIN (info->cur_plugin->data), info->subsys, info->name);
if (info->defer_id)
g_source_remove (info->defer_id);
g_free (info->subsys);
g_free (info->name);
g_slist_free (info->plugins);
memset (info, 0, sizeof (SupportsInfo));
g_free (info);
}
static char *
get_key (const char *subsys, const char *name)
{
return g_strdup_printf ("%s%s", subsys, name);
}
static void supports_callback (MMPlugin *plugin,
const char *subsys,
const char *name,
guint32 level,
gpointer user_data);
static void try_supports_port (MMManager *manager,
MMPlugin *plugin,
const char *subsys,
const char *name,
SupportsInfo *info);
static gboolean
supports_defer_timeout (gpointer user_data)
{
SupportsInfo *info = user_data;
g_debug ("(%s): re-checking support...", info->name);
try_supports_port (info->manager,
MM_PLUGIN (info->cur_plugin->data),
info->subsys,
info->name,
info);
return FALSE;
}
static void
try_supports_port (MMManager *manager,
MMPlugin *plugin,
const char *subsys,
const char *name,
SupportsInfo *info)
{
MMPluginSupportsResult result;
result = mm_plugin_supports_port (plugin, subsys, name, supports_callback, info);
switch (result) {
case MM_PLUGIN_SUPPORTS_PORT_UNSUPPORTED:
/* If the plugin knows it doesn't support the modem, just call the
* callback and indicate 0 support.
*/
supports_callback (plugin, subsys, name, 0, info);
break;
case MM_PLUGIN_SUPPORTS_PORT_DEFER:
g_debug ("(%s): (%s) deferring support check", mm_plugin_get_name (plugin), name);
if (info->defer_id)
g_source_remove (info->defer_id);
/* defer port detection for a bit as requested by the plugin */
info->defer_id = g_timeout_add (3000, supports_defer_timeout, info);
break;
case MM_PLUGIN_SUPPORTS_PORT_IN_PROGRESS:
default:
break;
}
}
static void
supports_callback (MMPlugin *plugin,
const char *subsys,
const char *name,
guint32 level,
gpointer user_data)
{
SupportsInfo *info = user_data;
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (info->manager);
MMModem *modem;
GError *error = NULL;
char *key;
GSList *iter;
MMPlugin *next_plugin = NULL;
info->cur_plugin = info->cur_plugin->next;
if (info->cur_plugin)
next_plugin = MM_PLUGIN (info->cur_plugin->data);
/* Is this plugin's result better than any one we've tried before? */
if (level > info->best_level) {
info->best_level = level;
info->best_plugin = plugin;
}
/* Prevent the generic plugin from probing devices that are already supported
* by other plugins. For Huawei for example, secondary ports shouldn't
* be probed, but the generic plugin would happily do so if allowed to.
*/
if ( next_plugin
&& !strcmp (mm_plugin_get_name (next_plugin), MM_PLUGIN_GENERIC_NAME)
&& info->best_plugin)
next_plugin = NULL;
/* Try the next plugin */
if (next_plugin) {
try_supports_port (info->manager, next_plugin, info->subsys, info->name, info);
return;
}
/* No more plugins to try */
if (info->best_plugin) {
/* Create the modem */
modem = mm_plugin_grab_port (info->best_plugin, info->subsys, info->name, &error);
if (modem) {
guint32 modem_type = MM_MODEM_TYPE_UNKNOWN;
const char *type_name = "UNKNOWN";
g_object_get (G_OBJECT (modem), MM_MODEM_TYPE, &modem_type, NULL);
if (modem_type == MM_MODEM_TYPE_GSM)
type_name = "GSM";
else if (modem_type == MM_MODEM_TYPE_CDMA)
type_name = "CDMA";
g_message ("(%s): %s modem %s claimed port %s",
mm_plugin_get_name (info->best_plugin),
type_name,
mm_modem_get_device (modem),
name);
add_modem (info->manager, modem);
} else {
g_warning ("%s: plugin '%s' claimed to support %s/%s but couldn't: (%d) %s",
__func__,
mm_plugin_get_name (info->best_plugin),
subsys,
name,
error ? error->code : -1,
(error && error->message) ? error->message : "(unknown)");
}
}
/* Tell each plugin to clean up any outstanding supports task */
for (iter = info->plugins; iter; iter = g_slist_next (iter))
mm_plugin_cancel_supports_port (MM_PLUGIN (iter->data), subsys, name);
g_slist_free (info->plugins);
info->cur_plugin = info->plugins = NULL;
key = get_key (info->subsys, info->name);
g_hash_table_remove (priv->supports, key);
g_free (key);
}
static void
device_added (MMManager *manager, GUdevDevice *device)
{
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (manager);
GSList *iter;
MMModem *modem = NULL;
const char *subsys, *name;
MMPlugin *best_plugin = NULL;
guint32 best_level = 0;
GError *error = NULL;
SupportsInfo *info;
char *key;
gboolean found;
g_return_if_fail (device != NULL);
if (!g_slist_length (priv->plugins))
return;
subsys = g_udev_device_get_subsystem (device);
name = g_udev_device_get_name (device);
if (find_modem_for_port (manager, subsys, name))
return;
/* Build up the list of plugins that support this port */
for (iter = priv->plugins; iter; iter = iter->next) {
MMPlugin *plugin = MM_PLUGIN (iter->data);
guint32 level;
level = mm_plugin_supports_port (plugin, subsys, name);
if (level > best_level) {
best_plugin = plugin;
best_level = level;
}
}
/* Let the best plugin handle this port */
if (!best_plugin)
return;
modem = mm_plugin_grab_port (best_plugin, subsys, name, &error);
if (modem) {
guint32 modem_type = MM_MODEM_TYPE_UNKNOWN;
const char *type_name = "UNKNOWN";
g_object_get (G_OBJECT (modem), MM_MODEM_TYPE, &modem_type, NULL);
if (modem_type == MM_MODEM_TYPE_GSM)
type_name = "GSM";
else if (modem_type == MM_MODEM_TYPE_CDMA)
type_name = "CDMA";
g_message ("(%s): %s modem %s claimed port %s",
mm_plugin_get_name (best_plugin),
type_name,
mm_modem_get_device (modem),
name);
} else {
g_warning ("%s: plugin '%s' claimed to support %s/%s but couldn't: (%d) %s",
__func__, mm_plugin_get_name (best_plugin), subsys, name,
error ? error->code : -1,
(error && error->message) ? error->message : "(unknown)");
key = get_key (subsys, name);
found = !!g_hash_table_lookup (priv->supports, key);
if (found) {
g_free (key);
return;
}
add_modem (manager, modem);
info = supports_info_new (manager, subsys, name);
g_hash_table_insert (priv->supports, key, info);
try_supports_port (manager, MM_PLUGIN (info->cur_plugin->data), subsys, name, info);
}
static void
device_removed (MMManager *manager, GUdevDevice *device)
{
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (manager);
MMModem *modem;
const char *subsys, *name;
char *key;
SupportsInfo *info;
g_return_if_fail (device != NULL);
if (!g_slist_length (priv->plugins))
return;
subsys = g_udev_device_get_subsystem (device);
name = g_udev_device_get_name (device);
modem = find_modem_for_port (manager, subsys, name);
if (modem)
if (modem) {
mm_modem_release_port (modem, subsys, name);
return;
}
/* Maybe a plugin is checking whether or not the port is supported */
key = get_key (subsys, name);
info = g_hash_table_lookup (priv->supports, key);
if (info) {
if (info->plugins)
mm_plugin_cancel_supports_port (MM_PLUGIN (info->plugins->data), subsys, name);
g_hash_table_remove (priv->supports, key);
}
g_free (key);
}
static void
@@ -391,6 +576,8 @@ mm_manager_init (MMManager *manager)
priv->modems = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_object_unref);
load_plugins (manager);
priv->supports = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify) supports_info_free);
priv->udev = g_udev_client_new (subsys);
g_assert (priv->udev);
g_signal_connect (priv->udev, "uevent", G_CALLBACK (handle_uevent), manager);
@@ -401,6 +588,7 @@ finalize (GObject *object)
{
MMManagerPrivate *priv = MM_MANAGER_GET_PRIVATE (object);
g_hash_table_destroy (priv->supports);
g_hash_table_destroy (priv->modems);
g_slist_foreach (priv->plugins, (GFunc) g_object_unref, NULL);
@@ -454,3 +642,4 @@ mm_manager_class_init (MMManagerClass *manager_class)
dbus_g_error_domain_register (MM_MODEM_CONNECT_ERROR, "org.freedesktop.ModemManager.Modem", MM_TYPE_MODEM_CONNECT_ERROR);
dbus_g_error_domain_register (MM_MOBILE_ERROR, "org.freedesktop.ModemManager.Modem.Gsm", MM_TYPE_MOBILE_ERROR);
}