/* * Event system * Also see comments in public header file and longer explanation below. * * Copyright (c) 2005, 2006 Johannes Berg <johannes@sipsolutions.net> * Joseph Jezak <josejx@gentoo.org> * Larry Finger <Larry.Finger@lwfinger.net> * Danny van Dyk <kugelfang@gentoo.org> * Michael Buesch <mbuesch@freenet.de> * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * 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 St, Fifth Floor, Boston, MA 02110-1301 USA * * The full GNU General Public License is included in this distribution in the * file called COPYING. */ #include "ieee80211softmac_priv.h" /* * Each event has associated to it * - an event type (see constants in public header) * - an event context (see below) * - the function to be called * - a context (extra parameter to call the function with) * - and the softmac struct * * The event context is private and can only be used from * within this module. Its meaning varies with the event * type: * SCAN_FINISHED, * DISASSOCIATED: NULL * ASSOCIATED, * ASSOCIATE_FAILED, * ASSOCIATE_TIMEOUT, * AUTHENTICATED, * AUTH_FAILED, * AUTH_TIMEOUT: a pointer to the network struct * ... * Code within this module can use the event context to be only * called when the event is true for that specific context * as per above table. * If the event context is NULL, then the notification is always called, * regardless of the event context. The event context is not passed to * the callback, it is assumed that the context suffices. * * You can also use the event context only by setting the event type * to -1 (private use only), in which case you'll be notified * whenever the event context matches. */ static char *event_descriptions[IEEE80211SOFTMAC_EVENT_LAST+1] = { NULL, /* scan finished */ NULL, /* associated */ "associating failed", "associating timed out", "authenticated", "authenticating failed", "authenticating timed out", "associating failed because no suitable network was found", NULL, /* disassociated */ }; static void ieee80211softmac_notify_callback(struct work_struct *work) { struct ieee80211softmac_event *pevent = container_of(work, struct ieee80211softmac_event, work.work); struct ieee80211softmac_event event = *pevent; kfree(pevent); event.fun(event.mac->dev, event.event_type, event.context); } int ieee80211softmac_notify_internal(struct ieee80211softmac_device *mac, int event, void *event_context, notify_function_ptr fun, void *context, gfp_t gfp_mask) { struct ieee80211softmac_event *eventptr; unsigned long flags; if (event < -1 || event > IEEE80211SOFTMAC_EVENT_LAST) return -ENOSYS; if (!fun) return -EINVAL; eventptr = kmalloc(sizeof(struct ieee80211softmac_event), gfp_mask); if (!eventptr) return -ENOMEM; eventptr->event_type = event; INIT_DELAYED_WORK(&eventptr->work, ieee80211softmac_notify_callback); eventptr->fun = fun; eventptr->context = context; eventptr->mac = mac; eventptr->event_context = event_context; spin_lock_irqsave(&mac->lock, flags); list_add(&eventptr->list, &mac->events); spin_unlock_irqrestore(&mac->lock, flags); return 0; } int ieee80211softmac_notify_gfp(struct net_device *dev, int event, notify_function_ptr fun, void *context, gfp_t gfp_mask) { struct ieee80211softmac_device *mac = ieee80211_priv(dev); if (event < 0 || event > IEEE80211SOFTMAC_EVENT_LAST) return -ENOSYS; return ieee80211softmac_notify_internal(mac, event, NULL, fun, context, gfp_mask); } EXPORT_SYMBOL_GPL(ieee80211softmac_notify_gfp); /* private -- calling all callbacks that were specified */ void ieee80211softmac_call_events_locked(struct ieee80211softmac_device *mac, int event, void *event_ctx) { struct ieee80211softmac_event *eventptr, *tmp; struct ieee80211softmac_network *network; if (event >= 0) { union iwreq_data wrqu; int we_event; char *msg = NULL; memset(&wrqu, '\0', sizeof (union iwreq_data)); switch(event) { case IEEE80211SOFTMAC_EVENT_ASSOCIATED: network = (struct ieee80211softmac_network *)event_ctx; memcpy(wrqu.ap_addr.sa_data, &network->bssid[0], ETH_ALEN); /* fall through */ case IEEE80211SOFTMAC_EVENT_DISASSOCIATED: wrqu.ap_addr.sa_family = ARPHRD_ETHER; we_event = SIOCGIWAP; break; case IEEE80211SOFTMAC_EVENT_SCAN_FINISHED: we_event = SIOCGIWSCAN; break; default: msg = event_descriptions[event]; if (!msg) msg = "SOFTMAC EVENT BUG"; wrqu.data.length = strlen(msg); we_event = IWEVCUSTOM; break; } wireless_send_event(mac->dev, we_event, &wrqu, msg); } if (!list_empty(&mac->events)) list_for_each_entry_safe(eventptr, tmp, &mac->events, list) { if ((eventptr->event_type == event || eventptr->event_type == -1) && (eventptr->event_context == NULL || eventptr->event_context == event_ctx)) { list_del(&eventptr->list); /* User may have subscribed to ANY event, so * we tell them which event triggered it. */ eventptr->event_type = event; queue_delayed_work(mac->wq, &eventptr->work, 0); } } } void ieee80211softmac_call_events(struct ieee80211softmac_device *mac, int event, void *event_ctx) { unsigned long flags; spin_lock_irqsave(&mac->lock, flags); ieee80211softmac_call_events_locked(mac, event, event_ctx); spin_unlock_irqrestore(&mac->lock, flags); }