Johannes Berg | 4855d25 | 2006-01-12 21:12:59 +0100 | [diff] [blame] | 1 | /* |
| 2 | * Event system |
| 3 | * Also see comments in public header file and longer explanation below. |
| 4 | * |
Johannes Berg | 7985905 | 2006-01-31 19:31:41 +0100 | [diff] [blame] | 5 | * Copyright (c) 2005, 2006 Johannes Berg <johannes@sipsolutions.net> |
| 6 | * Joseph Jezak <josejx@gentoo.org> |
| 7 | * Larry Finger <Larry.Finger@lwfinger.net> |
| 8 | * Danny van Dyk <kugelfang@gentoo.org> |
| 9 | * Michael Buesch <mbuesch@freenet.de> |
Johannes Berg | 4855d25 | 2006-01-12 21:12:59 +0100 | [diff] [blame] | 10 | * |
| 11 | * This program is free software; you can redistribute it and/or modify it |
| 12 | * under the terms of version 2 of the GNU General Public License as |
| 13 | * published by the Free Software Foundation. |
| 14 | * |
| 15 | * This program is distributed in the hope that it will be useful, but WITHOUT |
| 16 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| 17 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for |
| 18 | * more details. |
| 19 | * |
| 20 | * You should have received a copy of the GNU General Public License |
| 21 | * along with this program; if not, write to the Free Software |
| 22 | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
| 23 | * |
| 24 | * The full GNU General Public License is included in this distribution in the |
| 25 | * file called COPYING. |
| 26 | */ |
| 27 | |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 28 | #include "ieee80211softmac_priv.h" |
| 29 | |
| 30 | /* |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 31 | * Each event has associated to it |
| 32 | * - an event type (see constants in public header) |
| 33 | * - an event context (see below) |
| 34 | * - the function to be called |
| 35 | * - a context (extra parameter to call the function with) |
| 36 | * - and the softmac struct |
| 37 | * |
| 38 | * The event context is private and can only be used from |
| 39 | * within this module. Its meaning varies with the event |
| 40 | * type: |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 41 | * SCAN_FINISHED, |
| 42 | * DISASSOCIATED: NULL |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 43 | * ASSOCIATED, |
| 44 | * ASSOCIATE_FAILED, |
| 45 | * ASSOCIATE_TIMEOUT, |
| 46 | * AUTHENTICATED, |
| 47 | * AUTH_FAILED, |
| 48 | * AUTH_TIMEOUT: a pointer to the network struct |
| 49 | * ... |
| 50 | * Code within this module can use the event context to be only |
| 51 | * called when the event is true for that specific context |
| 52 | * as per above table. |
| 53 | * If the event context is NULL, then the notification is always called, |
| 54 | * regardless of the event context. The event context is not passed to |
| 55 | * the callback, it is assumed that the context suffices. |
| 56 | * |
| 57 | * You can also use the event context only by setting the event type |
| 58 | * to -1 (private use only), in which case you'll be notified |
| 59 | * whenever the event context matches. |
| 60 | */ |
| 61 | |
| 62 | static char *event_descriptions[IEEE80211SOFTMAC_EVENT_LAST+1] = { |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 63 | NULL, /* scan finished */ |
| 64 | NULL, /* associated */ |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 65 | "associating failed", |
| 66 | "associating timed out", |
| 67 | "authenticated", |
| 68 | "authenticating failed", |
| 69 | "authenticating timed out", |
| 70 | "associating failed because no suitable network was found", |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 71 | NULL, /* disassociated */ |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 72 | }; |
| 73 | |
| 74 | |
| 75 | static void |
| 76 | ieee80211softmac_notify_callback(void *d) |
| 77 | { |
| 78 | struct ieee80211softmac_event event = *(struct ieee80211softmac_event*) d; |
| 79 | kfree(d); |
| 80 | |
| 81 | event.fun(event.mac->dev, event.context); |
| 82 | } |
| 83 | |
| 84 | int |
| 85 | ieee80211softmac_notify_internal(struct ieee80211softmac_device *mac, |
| 86 | int event, void *event_context, notify_function_ptr fun, void *context, gfp_t gfp_mask) |
| 87 | { |
| 88 | struct ieee80211softmac_event *eventptr; |
| 89 | unsigned long flags; |
| 90 | |
| 91 | if (event < -1 || event > IEEE80211SOFTMAC_EVENT_LAST) |
| 92 | return -ENOSYS; |
| 93 | |
| 94 | if (!fun) |
| 95 | return -EINVAL; |
| 96 | |
| 97 | eventptr = kmalloc(sizeof(struct ieee80211softmac_event), gfp_mask); |
| 98 | if (!eventptr) |
| 99 | return -ENOMEM; |
| 100 | |
| 101 | eventptr->event_type = event; |
| 102 | INIT_WORK(&eventptr->work, ieee80211softmac_notify_callback, eventptr); |
| 103 | eventptr->fun = fun; |
| 104 | eventptr->context = context; |
| 105 | eventptr->mac = mac; |
| 106 | eventptr->event_context = event_context; |
| 107 | |
| 108 | spin_lock_irqsave(&mac->lock, flags); |
| 109 | list_add(&eventptr->list, &mac->events); |
| 110 | spin_unlock_irqrestore(&mac->lock, flags); |
| 111 | |
| 112 | return 0; |
| 113 | } |
| 114 | |
| 115 | int |
| 116 | ieee80211softmac_notify_gfp(struct net_device *dev, |
| 117 | int event, notify_function_ptr fun, void *context, gfp_t gfp_mask) |
| 118 | { |
| 119 | struct ieee80211softmac_device *mac = ieee80211_priv(dev); |
| 120 | |
| 121 | if (event < 0 || event > IEEE80211SOFTMAC_EVENT_LAST) |
| 122 | return -ENOSYS; |
| 123 | |
| 124 | return ieee80211softmac_notify_internal(mac, event, NULL, fun, context, gfp_mask); |
| 125 | } |
| 126 | EXPORT_SYMBOL_GPL(ieee80211softmac_notify_gfp); |
| 127 | |
| 128 | /* private -- calling all callbacks that were specified */ |
| 129 | void |
| 130 | ieee80211softmac_call_events_locked(struct ieee80211softmac_device *mac, int event, void *event_ctx) |
| 131 | { |
| 132 | struct ieee80211softmac_event *eventptr, *tmp; |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 133 | struct ieee80211softmac_network *network; |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 134 | |
| 135 | if (event >= 0) { |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 136 | union iwreq_data wrqu; |
| 137 | int we_event; |
| 138 | char *msg = NULL; |
| 139 | |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 140 | memset(&wrqu, '\0', sizeof (union iwreq_data)); |
| 141 | |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 142 | switch(event) { |
| 143 | case IEEE80211SOFTMAC_EVENT_ASSOCIATED: |
| 144 | network = (struct ieee80211softmac_network *)event_ctx; |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 145 | memcpy(wrqu.ap_addr.sa_data, &network->bssid[0], ETH_ALEN); |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 146 | /* fall through */ |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 147 | case IEEE80211SOFTMAC_EVENT_DISASSOCIATED: |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 148 | wrqu.ap_addr.sa_family = ARPHRD_ETHER; |
| 149 | we_event = SIOCGIWAP; |
| 150 | break; |
Johannes Berg | 6788a07 | 2006-04-13 11:41:28 +0200 | [diff] [blame] | 151 | case IEEE80211SOFTMAC_EVENT_SCAN_FINISHED: |
Johannes Berg | 6788a07 | 2006-04-13 11:41:28 +0200 | [diff] [blame] | 152 | we_event = SIOCGIWSCAN; |
| 153 | break; |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 154 | default: |
| 155 | msg = event_descriptions[event]; |
Johannes Berg | 921a91e | 2006-04-20 20:02:04 +0200 | [diff] [blame^] | 156 | if (!msg) |
| 157 | msg = "SOFTMAC EVENT BUG"; |
Johannes Berg | feeeaa8 | 2006-04-13 02:42:42 +0200 | [diff] [blame] | 158 | wrqu.data.length = strlen(msg); |
| 159 | we_event = IWEVCUSTOM; |
| 160 | break; |
| 161 | } |
| 162 | wireless_send_event(mac->dev, we_event, &wrqu, msg); |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 163 | } |
| 164 | |
| 165 | if (!list_empty(&mac->events)) |
| 166 | list_for_each_entry_safe(eventptr, tmp, &mac->events, list) { |
| 167 | if ((eventptr->event_type == event || eventptr->event_type == -1) |
| 168 | && (eventptr->event_context == NULL || eventptr->event_context == event_ctx)) { |
| 169 | list_del(&eventptr->list); |
Johannes Berg | 5c4df6d | 2006-01-06 01:43:45 +0100 | [diff] [blame] | 170 | schedule_work(&eventptr->work); |
Johannes Berg | 370121e | 2006-01-04 16:32:16 +0100 | [diff] [blame] | 171 | } |
| 172 | } |
| 173 | } |
| 174 | |
| 175 | void |
| 176 | ieee80211softmac_call_events(struct ieee80211softmac_device *mac, int event, void *event_ctx) |
| 177 | { |
| 178 | unsigned long flags; |
| 179 | |
| 180 | spin_lock_irqsave(&mac->lock, flags); |
| 181 | ieee80211softmac_call_events_locked(mac, event, event_ctx); |
| 182 | |
| 183 | spin_unlock_irqrestore(&mac->lock, flags); |
| 184 | } |