2 * cfg80211 - wext compat code
4 * This is temporary code until all wireless functionality is migrated
5 * into cfg80211, when that happens all the exports here go away and
6 * we directly assign the wireless handlers of wireless interfaces.
8 * Copyright 2008 Johannes Berg <johannes@sipsolutions.net>
11 #include <linux/wireless.h>
12 #include <linux/nl80211.h>
13 #include <linux/if_arp.h>
14 #include <net/iw_handler.h>
15 #include <net/wireless.h>
16 #include <net/cfg80211.h>
19 int cfg80211_wext_giwname(struct net_device
*dev
,
20 struct iw_request_info
*info
,
21 char *name
, char *extra
)
23 struct wireless_dev
*wdev
= dev
->ieee80211_ptr
;
24 struct ieee80211_supported_band
*sband
;
25 bool is_ht
= false, is_a
= false, is_b
= false, is_g
= false;
30 sband
= wdev
->wiphy
->bands
[IEEE80211_BAND_5GHZ
];
33 is_ht
|= sband
->ht_cap
.ht_supported
;
36 sband
= wdev
->wiphy
->bands
[IEEE80211_BAND_2GHZ
];
39 /* Check for mandatory rates */
40 for (i
= 0; i
< sband
->n_bitrates
; i
++) {
41 if (sband
->bitrates
[i
].bitrate
== 10)
43 if (sband
->bitrates
[i
].bitrate
== 60)
46 is_ht
|= sband
->ht_cap
.ht_supported
;
49 strcpy(name
, "IEEE 802.11");
61 EXPORT_SYMBOL(cfg80211_wext_giwname
);
63 int cfg80211_wext_siwmode(struct net_device
*dev
, struct iw_request_info
*info
,
64 u32
*mode
, char *extra
)
66 struct wireless_dev
*wdev
= dev
->ieee80211_ptr
;
67 struct cfg80211_registered_device
*rdev
;
68 struct vif_params vifparams
;
69 enum nl80211_iftype type
;
75 rdev
= wiphy_to_dev(wdev
->wiphy
);
77 if (!rdev
->ops
->change_virtual_intf
)
80 /* don't support changing VLANs, you just re-create them */
81 if (wdev
->iftype
== NL80211_IFTYPE_AP_VLAN
)
86 type
= NL80211_IFTYPE_STATION
;
89 type
= NL80211_IFTYPE_ADHOC
;
92 type
= NL80211_IFTYPE_WDS
;
95 type
= NL80211_IFTYPE_MONITOR
;
101 if (type
== wdev
->iftype
)
104 memset(&vifparams
, 0, sizeof(vifparams
));
106 ret
= rdev
->ops
->change_virtual_intf(wdev
->wiphy
, dev
->ifindex
, type
,
108 WARN_ON(!ret
&& wdev
->iftype
!= type
);
112 EXPORT_SYMBOL(cfg80211_wext_siwmode
);
114 int cfg80211_wext_giwmode(struct net_device
*dev
, struct iw_request_info
*info
,
115 u32
*mode
, char *extra
)
117 struct wireless_dev
*wdev
= dev
->ieee80211_ptr
;
122 switch (wdev
->iftype
) {
123 case NL80211_IFTYPE_AP
:
124 *mode
= IW_MODE_MASTER
;
126 case NL80211_IFTYPE_STATION
:
127 *mode
= IW_MODE_INFRA
;
129 case NL80211_IFTYPE_ADHOC
:
130 *mode
= IW_MODE_ADHOC
;
132 case NL80211_IFTYPE_MONITOR
:
133 *mode
= IW_MODE_MONITOR
;
135 case NL80211_IFTYPE_WDS
:
136 *mode
= IW_MODE_REPEAT
;
138 case NL80211_IFTYPE_AP_VLAN
:
139 *mode
= IW_MODE_SECOND
; /* FIXME */
142 *mode
= IW_MODE_AUTO
;
147 EXPORT_SYMBOL(cfg80211_wext_giwmode
);
150 int cfg80211_wext_giwrange(struct net_device
*dev
,
151 struct iw_request_info
*info
,
152 struct iw_point
*data
, char *extra
)
154 struct wireless_dev
*wdev
= dev
->ieee80211_ptr
;
155 struct iw_range
*range
= (struct iw_range
*) extra
;
156 enum ieee80211_band band
;
162 data
->length
= sizeof(struct iw_range
);
163 memset(range
, 0, sizeof(struct iw_range
));
165 range
->we_version_compiled
= WIRELESS_EXT
;
166 range
->we_version_source
= 21;
167 range
->retry_capa
= IW_RETRY_LIMIT
;
168 range
->retry_flags
= IW_RETRY_LIMIT
;
169 range
->min_retry
= 0;
170 range
->max_retry
= 255;
172 range
->max_rts
= 2347;
173 range
->min_frag
= 256;
174 range
->max_frag
= 2346;
176 range
->encoding_size
[0] = 5;
177 range
->encoding_size
[1] = 13;
178 range
->num_encoding_sizes
= 2;
179 range
->max_encoding_tokens
= 4;
181 range
->max_qual
.updated
= IW_QUAL_NOISE_INVALID
;
183 switch (wdev
->wiphy
->signal_type
) {
184 case CFG80211_SIGNAL_TYPE_NONE
:
186 case CFG80211_SIGNAL_TYPE_MBM
:
187 range
->max_qual
.level
= -110;
188 range
->max_qual
.qual
= 70;
189 range
->avg_qual
.qual
= 35;
190 range
->max_qual
.updated
|= IW_QUAL_DBM
;
191 range
->max_qual
.updated
|= IW_QUAL_QUAL_UPDATED
;
192 range
->max_qual
.updated
|= IW_QUAL_LEVEL_UPDATED
;
194 case CFG80211_SIGNAL_TYPE_UNSPEC
:
195 range
->max_qual
.level
= 100;
196 range
->max_qual
.qual
= 100;
197 range
->avg_qual
.qual
= 50;
198 range
->max_qual
.updated
|= IW_QUAL_QUAL_UPDATED
;
199 range
->max_qual
.updated
|= IW_QUAL_LEVEL_UPDATED
;
203 range
->avg_qual
.level
= range
->max_qual
.level
/ 2;
204 range
->avg_qual
.noise
= range
->max_qual
.noise
/ 2;
205 range
->avg_qual
.updated
= range
->max_qual
.updated
;
207 range
->enc_capa
= IW_ENC_CAPA_WPA
| IW_ENC_CAPA_WPA2
|
208 IW_ENC_CAPA_CIPHER_TKIP
| IW_ENC_CAPA_CIPHER_CCMP
;
210 for (band
= 0; band
< IEEE80211_NUM_BANDS
; band
++) {
212 struct ieee80211_supported_band
*sband
;
214 sband
= wdev
->wiphy
->bands
[band
];
219 for (i
= 0; i
< sband
->n_channels
&& c
< IW_MAX_FREQUENCIES
; i
++) {
220 struct ieee80211_channel
*chan
= &sband
->channels
[i
];
222 if (!(chan
->flags
& IEEE80211_CHAN_DISABLED
)) {
224 ieee80211_frequency_to_channel(
226 range
->freq
[c
].m
= chan
->center_freq
;
227 range
->freq
[c
].e
= 6;
232 range
->num_channels
= c
;
233 range
->num_frequency
= c
;
235 IW_EVENT_CAPA_SET_KERNEL(range
->event_capa
);
236 IW_EVENT_CAPA_SET(range
->event_capa
, SIOCGIWAP
);
237 IW_EVENT_CAPA_SET(range
->event_capa
, SIOCGIWSCAN
);
239 range
->scan_capa
|= IW_SCAN_CAPA_ESSID
;
243 EXPORT_SYMBOL(cfg80211_wext_giwrange
);
245 int cfg80211_wext_siwmlme(struct net_device
*dev
,
246 struct iw_request_info
*info
,
247 struct iw_point
*data
, char *extra
)
249 struct wireless_dev
*wdev
= dev
->ieee80211_ptr
;
250 struct iw_mlme
*mlme
= (struct iw_mlme
*)extra
;
251 struct cfg80211_registered_device
*rdev
;
253 struct cfg80211_disassoc_request disassoc
;
254 struct cfg80211_deauth_request deauth
;
260 rdev
= wiphy_to_dev(wdev
->wiphy
);
262 if (wdev
->iftype
!= NL80211_IFTYPE_STATION
)
265 if (mlme
->addr
.sa_family
!= ARPHRD_ETHER
)
268 memset(&cmd
, 0, sizeof(cmd
));
272 if (!rdev
->ops
->deauth
)
274 cmd
.deauth
.peer_addr
= mlme
->addr
.sa_data
;
275 cmd
.deauth
.reason_code
= mlme
->reason_code
;
276 return rdev
->ops
->deauth(wdev
->wiphy
, dev
, &cmd
.deauth
);
277 case IW_MLME_DISASSOC
:
278 if (!rdev
->ops
->disassoc
)
280 cmd
.disassoc
.peer_addr
= mlme
->addr
.sa_data
;
281 cmd
.disassoc
.reason_code
= mlme
->reason_code
;
282 return rdev
->ops
->disassoc(wdev
->wiphy
, dev
, &cmd
.disassoc
);
287 EXPORT_SYMBOL(cfg80211_wext_siwmlme
);
291 * cfg80211_wext_freq - get wext frequency for non-"auto"
293 * @freq: the wext freq encoding
295 * Returns a channel, %NULL for auto, or an ERR_PTR for errors!
297 struct ieee80211_channel
*cfg80211_wext_freq(struct wiphy
*wiphy
,
298 struct iw_freq
*freq
)
304 return ieee80211_get_channel(wiphy
,
305 ieee80211_channel_to_frequency(freq
->m
));
307 int i
, div
= 1000000;
308 for (i
= 0; i
< freq
->e
; i
++)
311 return ieee80211_get_channel(wiphy
, freq
->m
/ div
);
313 return ERR_PTR(-EINVAL
);
317 EXPORT_SYMBOL(cfg80211_wext_freq
);