Blame view

include/net/iw_handler.h 20.9 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  /* SPDX-License-Identifier: GPL-2.0 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
  /*
   * This file define the new driver API for Wireless Extensions
   *
c2805fbb8   Jean Tourrilhes   [PATCH] WE-22 : p...
5
   * Version :	8	16.3.07
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
6
7
   *
   * Authors :	Jean Tourrilhes - HPL - <jt@hpl.hp.com>
c2805fbb8   Jean Tourrilhes   [PATCH] WE-22 : p...
8
   * Copyright (c) 2001-2007 Jean Tourrilhes, All Rights Reserved.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
   */
  
  #ifndef _IW_HANDLER_H
  #define _IW_HANDLER_H
  
  /************************** DOCUMENTATION **************************/
  /*
   * Initial driver API (1996 -> onward) :
   * -----------------------------------
   * The initial API just sends the IOCTL request received from user space
   * to the driver (via the driver ioctl handler). The driver has to
   * handle all the rest...
   *
   * The initial API also defines a specific handler in struct net_device
   * to handle wireless statistics.
   *
   * The initial APIs served us well and has proven a reasonably good design.
   * However, there is a few shortcommings :
   *	o No events, everything is a request to the driver.
   *	o Large ioctl function in driver with gigantic switch statement
   *	  (i.e. spaghetti code).
   *	o Driver has to mess up with copy_to/from_user, and in many cases
   *	  does it unproperly. Common mistakes are :
   *		* buffer overflows (no checks or off by one checks)
   *		* call copy_to/from_user with irq disabled
   *	o The user space interface is tied to ioctl because of the use
   *	  copy_to/from_user.
   *
   * New driver API (2002 -> onward) :
   * -------------------------------
   * The new driver API is just a bunch of standard functions (handlers),
   * each handling a specific Wireless Extension. The driver just export
   * the list of handler it supports, and those will be called apropriately.
   *
   * I tried to keep the main advantage of the previous API (simplicity,
   * efficiency and light weight), and also I provide a good dose of backward
   * compatibility (most structures are the same, driver can use both API
   * simultaneously, ...).
   * Hopefully, I've also addressed the shortcomming of the initial API.
   *
   * The advantage of the new API are :
   *	o Handling of Extensions in driver broken in small contained functions
   *	o Tighter checks of ioctl before calling the driver
   *	o Flexible commit strategy (at least, the start of it)
   *	o Backward compatibility (can be mixed with old API)
   *	o Driver doesn't have to worry about memory and user-space issues
   * The last point is important for the following reasons :
   *	o You are now able to call the new driver API from any API you
   *		want (including from within other parts of the kernel).
   *	o Common mistakes are avoided (buffer overflow, user space copy
   *		with irq disabled and so on).
   *
   * The Drawback of the new API are :
   *	o bloat (especially kernel)
   *	o need to migrate existing drivers to new API
   * My initial testing shows that the new API adds around 3kB to the kernel
   * and save between 0 and 5kB from a typical driver.
   * Also, as all structures and data types are unchanged, the migration is
   * quite straightforward (but tedious).
   *
   * ---
   *
   * The new driver API is defined below in this file. User space should
   * not be aware of what's happening down there...
   *
   * A new kernel wrapper is in charge of validating the IOCTLs and calling
   * the appropriate driver handler. This is implemented in :
   *	# net/core/wireless.c
   *
   * The driver export the list of handlers in :
   *	# include/linux/netdevice.h (one place)
   *
   * The new driver API is available for WIRELESS_EXT >= 13.
   * Good luck with migration to the new API ;-)
   */
  
  /* ---------------------- THE IMPLEMENTATION ---------------------- */
  /*
   * Some of the choice I've made are pretty controversials. Defining an
   * API is very much weighting compromises. This goes into some of the
   * details and the thinking behind the implementation.
   *
   * Implementation goals :
   * --------------------
   * The implementation goals were as follow :
   *	o Obvious : you should not need a PhD to understand what's happening,
25985edce   Lucas De Marchi   Fix common misspe...
95
   *		the benefit is easier maintenance.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
   *	o Flexible : it should accommodate a wide variety of driver
   *		implementations and be as flexible as the old API.
   *	o Lean : it should be efficient memory wise to minimise the impact
   *		on kernel footprint.
   *	o Transparent to user space : the large number of user space
   *		applications that use Wireless Extensions should not need
   *		any modifications.
   *
   * Array of functions versus Struct of functions
   * ---------------------------------------------
   * 1) Having an array of functions allow the kernel code to access the
   * handler in a single lookup, which is much more efficient (think hash
   * table here).
   * 2) The only drawback is that driver writer may put their handler in
   * the wrong slot. This is trivial to test (I set the frequency, the
   * bitrate changes). Once the handler is in the proper slot, it will be
   * there forever, because the array is only extended at the end.
   * 3) Backward/forward compatibility : adding new handler just require
   * extending the array, so you can put newer driver in older kernel
   * without having to patch the kernel code (and vice versa).
   *
   * All handler are of the same generic type
   * ----------------------------------------
   * That's a feature !!!
   * 1) Having a generic handler allow to have generic code, which is more
   * efficient. If each of the handler was individually typed I would need
   * to add a big switch in the kernel (== more bloat). This solution is
   * more scalable, adding new Wireless Extensions doesn't add new code.
   * 2) You can use the same handler in different slots of the array. For
   * hardware, it may be more efficient or logical to handle multiple
   * Wireless Extensions with a single function, and the API allow you to
   * do that. (An example would be a single record on the card to control
   * both bitrate and frequency, the handler would read the old record,
   * modify it according to info->cmd and rewrite it).
   *
   * Functions prototype uses union iwreq_data
   * -----------------------------------------
25985edce   Lucas De Marchi   Fix common misspe...
133
   * Some would have preferred functions defined this way :
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
   *	static int mydriver_ioctl_setrate(struct net_device *dev, 
   *					  long rate, int auto)
   * 1) The kernel code doesn't "validate" the content of iwreq_data, and
   * can't do it (different hardware may have different notion of what a
   * valid frequency is), so we don't pretend that we do it.
   * 2) The above form is not extendable. If I want to add a flag (for
   * example to distinguish setting max rate and basic rate), I would
   * break the prototype. Using iwreq_data is more flexible.
   * 3) Also, the above form is not generic (see above).
   * 4) I don't expect driver developper using the wrong field of the
   * union (Doh !), so static typechecking doesn't add much value.
   * 5) Lastly, you can skip the union by doing :
   *	static int mydriver_ioctl_setrate(struct net_device *dev,
   *					  struct iw_request_info *info,
   *					  struct iw_param *rrq,
   *					  char *extra)
   * And then adding the handler in the array like this :
   *        (iw_handler) mydriver_ioctl_setrate,             // SIOCSIWRATE
   *
   * Using functions and not a registry
   * ----------------------------------
   * Another implementation option would have been for every instance to
   * define a registry (a struct containing all the Wireless Extensions)
   * and only have a function to commit the registry to the hardware.
   * 1) This approach can be emulated by the current code, but not
   * vice versa.
   * 2) Some drivers don't keep any configuration in the driver, for them
   * adding such a registry would be a significant bloat.
   * 3) The code to translate from Wireless Extension to native format is
   * needed anyway, so it would not reduce significantely the amount of code.
   * 4) The current approach only selectively translate Wireless Extensions
   * to native format and only selectively set, whereas the registry approach
   * would require to translate all WE and set all parameters for any single
   * change.
   * 5) For many Wireless Extensions, the GET operation return the current
   * dynamic value, not the value that was set.
   *
   * This header is <net/iw_handler.h>
   * ---------------------------------
   * 1) This header is kernel space only and should not be exported to
   * user space. Headers in "include/linux/" are exported, headers in
   * "include/net/" are not.
   *
   * Mixed 32/64 bit issues
   * ----------------------
   * The Wireless Extensions are designed to be 64 bit clean, by using only
   * datatypes with explicit storage size.
   * There are some issues related to kernel and user space using different
   * memory model, and in particular 64bit kernel with 32bit user space.
   * The problem is related to struct iw_point, that contains a pointer
   * that *may* need to be translated.
   * This is quite messy. The new API doesn't solve this problem (it can't),
   * but is a step in the right direction :
   * 1) Meta data about each ioctl is easily available, so we know what type
   * of translation is needed.
   * 2) The move of data between kernel and user space is only done in a single
   * place in the kernel, so adding specific hooks in there is possible.
   * 3) In the long term, it allows to move away from using ioctl as the
   * user space API.
   *
   * So many comments and so few code
   * --------------------------------
   * That's a feature. Comments won't bloat the resulting kernel binary.
   */
  
  /***************************** INCLUDES *****************************/
  
  #include <linux/wireless.h>		/* IOCTL user space API */
  #include <linux/if_ether.h>
  
  /***************************** VERSION *****************************/
  /*
   * This constant is used to know which version of the driver API is
   * available. Hopefully, this will be pretty stable and no changes
   * will be needed...
   * I just plan to increment with each new version.
   */
c2805fbb8   Jean Tourrilhes   [PATCH] WE-22 : p...
211
  #define IW_HANDLER_VERSION	8
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
  
  /*
   * Changes :
   *
   * V2 to V3
   * --------
   *	- Move event definition in <linux/wireless.h>
   *	- Add Wireless Event support :
   *		o wireless_send_event() prototype
   *		o iwe_stream_add_event/point() inline functions
   * V3 to V4
   * --------
   *	- Reshuffle IW_HEADER_TYPE_XXX to map IW_PRIV_TYPE_XXX changes
   *
   * V4 to V5
   * --------
   *	- Add new spy support : struct iw_spy_data & prototypes
   *
   * V5 to V6
   * --------
   *	- Change the way we get to spy_data method for added safety
   *	- Remove spy #ifdef, they are always on -> cleaner code
   *	- Add IW_DESCR_FLAG_NOMAX flag for very large requests
   *	- Start migrating get_wireless_stats to struct iw_handler_def
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
236
237
238
239
240
241
242
   *
   * V6 to V7
   * --------
   *	- Add struct ieee80211_device pointer in struct iw_public_data
   *	- Remove (struct iw_point *)->pointer from events and streams
   *	- Remove spy_offset from struct iw_handler_def
   *	- Add "check" version of event macros for ieee802.11 stack
c2805fbb8   Jean Tourrilhes   [PATCH] WE-22 : p...
243
244
245
246
   *
   * V7 to V8
   * ----------
   *	- Prevent leaking of kernel space in stream on 64 bits.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
247
248
249
250
251
252
253
254
255
256
257
258
259
   */
  
  /**************************** CONSTANTS ****************************/
  
  /* Enhanced spy support available */
  #define IW_WIRELESS_SPY
  #define IW_WIRELESS_THRSPY
  
  /* Special error message for the driver to indicate that we
   * should do a commit after return from the iw_handler */
  #define EIWCOMMIT	EINPROGRESS
  
  /* Flags available in struct iw_request_info */
0f5cabba4   David S. Miller   wext: Create IW_R...
260
  #define IW_REQUEST_FLAG_COMPAT	0x0001	/* Compat ioctl call */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
  
  /* Type of headers we know about (basically union iwreq_data) */
  #define IW_HEADER_TYPE_NULL	0	/* Not available */
  #define IW_HEADER_TYPE_CHAR	2	/* char [IFNAMSIZ] */
  #define IW_HEADER_TYPE_UINT	4	/* __u32 */
  #define IW_HEADER_TYPE_FREQ	5	/* struct iw_freq */
  #define IW_HEADER_TYPE_ADDR	6	/* struct sockaddr */
  #define IW_HEADER_TYPE_POINT	8	/* struct iw_point */
  #define IW_HEADER_TYPE_PARAM	9	/* struct iw_param */
  #define IW_HEADER_TYPE_QUAL	10	/* struct iw_quality */
  
  /* Handling flags */
  /* Most are not implemented. I just use them as a reminder of some
   * cool features we might need one day ;-) */
  #define IW_DESCR_FLAG_NONE	0x0000	/* Obvious */
  /* Wrapper level flags */
  #define IW_DESCR_FLAG_DUMP	0x0001	/* Not part of the dump command */
  #define IW_DESCR_FLAG_EVENT	0x0002	/* Generate an event on SET */
  #define IW_DESCR_FLAG_RESTRICT	0x0004	/* GET : request is ROOT only */
  				/* SET : Omit payload from generated iwevent */
  #define IW_DESCR_FLAG_NOMAX	0x0008	/* GET : no limit on request size */
  /* Driver level flags */
  #define IW_DESCR_FLAG_WAIT	0x0100	/* Wait for driver event */
  
  /****************************** TYPES ******************************/
  
  /* ----------------------- WIRELESS HANDLER ----------------------- */
  /*
   * A wireless handler is just a standard function, that looks like the
   * ioctl handler.
   * We also define there how a handler list look like... As the Wireless
   * Extension space is quite dense, we use a simple array, which is faster
   * (that's the perfect hash table ;-).
   */
  
  /*
   * Meta data about the request passed to the iw_handler.
   * Most handlers can safely ignore what's in there.
   * The 'cmd' field might come handy if you want to use the same handler
   * for multiple command...
   * This struct is also my long term insurance. I can add new fields here
   * without breaking the prototype of iw_handler...
   */
fd2c3ef76   Eric Dumazet   net: cleanup incl...
304
  struct iw_request_info {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
  	__u16		cmd;		/* Wireless Extension command */
  	__u16		flags;		/* More to come ;-) */
  };
  
  struct net_device;
  
  /*
   * This is how a function handling a Wireless Extension should look
   * like (both get and set, standard and private).
   */
  typedef int (*iw_handler)(struct net_device *dev, struct iw_request_info *info,
  			  union iwreq_data *wrqu, char *extra);
  
  /*
   * This define all the handler that the driver export.
   * As you need only one per driver type, please use a static const
   * shared by all driver instances... Same for the members...
   * This will be linked from net_device in <linux/netdevice.h>
   */
fd2c3ef76   Eric Dumazet   net: cleanup incl...
324
  struct iw_handler_def {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
325
326
  
  	/* Array of handlers for standard ioctls
f9ea3eb44   Joe Perches   include/net/iw_ha...
327
  	 * We will call dev->wireless_handlers->standard[ioctl - SIOCIWFIRST]
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
328
329
  	 */
  	const iw_handler *	standard;
3d23e349d   Johannes Berg   wext: refactor
330
331
332
  	/* Number of handlers defined (more precisely, index of the
  	 * last defined handler + 1) */
  	__u16			num_standard;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
333

3d23e349d   Johannes Berg   wext: refactor
334
335
336
337
  #ifdef CONFIG_WEXT_PRIV
  	__u16			num_private;
  	/* Number of private arg description */
  	__u16			num_private_args;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
338
339
340
341
342
343
344
345
346
  	/* Array of handlers for private ioctls
  	 * Will call dev->wireless_handlers->private[ioctl - SIOCIWFIRSTPRIV]
  	 */
  	const iw_handler *	private;
  
  	/* Arguments of private handler. This one is just a list, so you
  	 * can put it in any order you want and should not leave holes...
  	 * We will automatically export that to user space... */
  	const struct iw_priv_args *	private_args;
3d23e349d   Johannes Berg   wext: refactor
347
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
348

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
  	/* New location of get_wireless_stats, to de-bloat struct net_device.
  	 * The old pointer in struct net_device will be gradually phased
  	 * out, and drivers are encouraged to use this one... */
  	struct iw_statistics*	(*get_wireless_stats)(struct net_device *dev);
  };
  
  /* ---------------------- IOCTL DESCRIPTION ---------------------- */
  /*
   * One of the main goal of the new interface is to deal entirely with
   * user space/kernel space memory move.
   * For that, we need to know :
   *	o if iwreq is a pointer or contain the full data
   *	o what is the size of the data to copy
   *
   * For private IOCTLs, we use the same rules as used by iwpriv and
   * defined in struct iw_priv_args.
   *
   * For standard IOCTLs, things are quite different and we need to
   * use the stuctures below. Actually, this struct is also more
   * efficient, but that's another story...
   */
  
  /*
   * Describe how a standard IOCTL looks like.
   */
fd2c3ef76   Eric Dumazet   net: cleanup incl...
374
  struct iw_ioctl_description {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
  	__u8	header_type;		/* NULL, iw_point or other */
  	__u8	token_type;		/* Future */
  	__u16	token_size;		/* Granularity of payload */
  	__u16	min_tokens;		/* Min acceptable token number */
  	__u16	max_tokens;		/* Max acceptable token number */
  	__u32	flags;			/* Special handling of the request */
  };
  
  /* Need to think of short header translation table. Later. */
  
  /* --------------------- ENHANCED SPY SUPPORT --------------------- */
  /*
   * In the old days, the driver was handling spy support all by itself.
   * Now, the driver can delegate this task to Wireless Extensions.
   * It needs to include this struct in its private part and use the
   * standard spy iw_handler.
   */
  
  /*
   * Instance specific spy data, i.e. addresses spied and quality for them.
   */
fd2c3ef76   Eric Dumazet   net: cleanup incl...
396
  struct iw_spy_data {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
397
398
399
400
401
402
403
404
405
406
407
408
409
  	/* --- Standard spy support --- */
  	int			spy_number;
  	u_char			spy_address[IW_MAX_SPY][ETH_ALEN];
  	struct iw_quality	spy_stat[IW_MAX_SPY];
  	/* --- Enhanced spy support (event) */
  	struct iw_quality	spy_thr_low;	/* Low threshold */
  	struct iw_quality	spy_thr_high;	/* High threshold */
  	u_char			spy_thr_under[IW_MAX_SPY];
  };
  
  /* --------------------- DEVICE WIRELESS DATA --------------------- */
  /*
   * This is all the wireless data specific to a device instance that
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
410
   * is managed by the core of Wireless Extensions or the 802.11 layer.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
411
412
413
414
415
416
   * We only keep pointer to those structures, so that a driver is free
   * to share them between instances.
   * This structure should be initialised before registering the device.
   * Access to this data follow the same rules as any other struct net_device
   * data (i.e. valid as long as struct net_device exist, same locking rules).
   */
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
417
  /* Forward declaration */
b0a4e7d8a   John W. Linville   libipw: switch fr...
418
  struct libipw_device;
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
419
  /* The struct */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
420
421
  struct iw_public_data {
  	/* Driver enhanced spy support */
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
422
  	struct iw_spy_data *		spy_data;
b0a4e7d8a   John W. Linville   libipw: switch fr...
423
424
  	/* Legacy structure managed by the ipw2x00-specific IEEE 802.11 layer */
  	struct libipw_device *		libipw;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
425
426
427
428
429
430
431
  };
  
  /**************************** PROTOTYPES ****************************/
  /*
   * Functions part of the Wireless Extensions (defined in net/core/wireless.c).
   * Those may be called only within the kernel.
   */
556829657   Johannes Berg   [NL80211]: add ne...
432
433
434
  /* First : function strictly used inside the kernel */
  
  /* Handle /proc/net/wireless, called in net/code/dev.c */
70a3926f4   Joe Perches   iw_handler.h: Rem...
435
  int dev_get_wireless_info(char *buffer, char **start, off_t offset, int length);
556829657   Johannes Berg   [NL80211]: add ne...
436
437
  
  /* Second : functions that may be called by driver modules */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
438
439
  
  /* Send a single event to user space */
70a3926f4   Joe Perches   iw_handler.h: Rem...
440
441
  void wireless_send_event(struct net_device *dev, unsigned int cmd,
  			 union iwreq_data *wrqu, const char *extra);
cb150b9d2   Johannes Berg   cfg80211/wext: fi...
442
443
444
445
446
447
  #ifdef CONFIG_WEXT_CORE
  /* flush all previous wext events - if work is done from netdev notifiers */
  void wireless_nlevent_flush(void);
  #else
  static inline void wireless_nlevent_flush(void) {}
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
448
449
450
451
452
  
  /* We may need a function to send a stream of events to user space.
   * More on that later... */
  
  /* Standard handler for SIOCSIWSPY */
70a3926f4   Joe Perches   iw_handler.h: Rem...
453
454
  int iw_handler_set_spy(struct net_device *dev, struct iw_request_info *info,
  		       union iwreq_data *wrqu, char *extra);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
455
  /* Standard handler for SIOCGIWSPY */
70a3926f4   Joe Perches   iw_handler.h: Rem...
456
457
  int iw_handler_get_spy(struct net_device *dev, struct iw_request_info *info,
  		       union iwreq_data *wrqu, char *extra);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
458
  /* Standard handler for SIOCSIWTHRSPY */
70a3926f4   Joe Perches   iw_handler.h: Rem...
459
460
  int iw_handler_set_thrspy(struct net_device *dev, struct iw_request_info *info,
  			  union iwreq_data *wrqu, char *extra);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
461
  /* Standard handler for SIOCGIWTHRSPY */
70a3926f4   Joe Perches   iw_handler.h: Rem...
462
463
  int iw_handler_get_thrspy(struct net_device *dev, struct iw_request_info *info,
  			  union iwreq_data *wrqu, char *extra);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
464
  /* Driver call to update spy records */
70a3926f4   Joe Perches   iw_handler.h: Rem...
465
466
  void wireless_spy_update(struct net_device *dev, unsigned char *address,
  			 struct iw_quality *wstats);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
467
468
469
470
471
  
  /************************* INLINE FUNTIONS *************************/
  /*
   * Function that are so simple that it's more efficient inlining them
   */
ccc580571   David S. Miller   wext: Emit event ...
472
  static inline int iwe_stream_lcp_len(struct iw_request_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
473
  {
ccc580571   David S. Miller   wext: Emit event ...
474
475
476
477
478
  #ifdef CONFIG_COMPAT
  	if (info->flags & IW_REQUEST_FLAG_COMPAT)
  		return IW_EV_COMPAT_LCP_LEN;
  #endif
  	return IW_EV_LCP_LEN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
479
  }
ccc580571   David S. Miller   wext: Emit event ...
480
  static inline int iwe_stream_point_len(struct iw_request_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
481
  {
ccc580571   David S. Miller   wext: Emit event ...
482
483
484
485
486
  #ifdef CONFIG_COMPAT
  	if (info->flags & IW_REQUEST_FLAG_COMPAT)
  		return IW_EV_COMPAT_POINT_LEN;
  #endif
  	return IW_EV_POINT_LEN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
487
  }
ccc580571   David S. Miller   wext: Emit event ...
488
489
  static inline int iwe_stream_event_len_adjust(struct iw_request_info *info,
  					      int event_len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
490
  {
ccc580571   David S. Miller   wext: Emit event ...
491
492
493
494
  #ifdef CONFIG_COMPAT
  	if (info->flags & IW_REQUEST_FLAG_COMPAT) {
  		event_len -= IW_EV_LCP_LEN;
  		event_len += IW_EV_COMPAT_LCP_LEN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
495
  	}
ccc580571   David S. Miller   wext: Emit event ...
496
497
498
  #endif
  
  	return event_len;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
499
  }
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
500
501
502
  /*------------------------------------------------------------------*/
  /*
   * Wrapper to add an Wireless Event to a stream of events.
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
503
   */
10b2eb694   Johannes Berg   wext: uninline st...
504
505
  char *iwe_stream_add_event(struct iw_request_info *info, char *stream,
  			   char *ends, struct iw_event *iwe, int event_len);
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
506

36ef906ee   Johannes Berg   wext: add checked...
507
508
509
510
511
512
513
514
515
516
  static inline char *
  iwe_stream_add_event_check(struct iw_request_info *info, char *stream,
  			   char *ends, struct iw_event *iwe, int event_len)
  {
  	char *res = iwe_stream_add_event(info, stream, ends, iwe, event_len);
  
  	if (res == stream)
  		return ERR_PTR(-E2BIG);
  	return res;
  }
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
517
518
519
520
  /*------------------------------------------------------------------*/
  /*
   * Wrapper to add an short Wireless Event containing a pointer to a
   * stream of events.
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
521
   */
10b2eb694   Johannes Berg   wext: uninline st...
522
523
  char *iwe_stream_add_point(struct iw_request_info *info, char *stream,
  			   char *ends, struct iw_event *iwe, char *extra);
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
524

36ef906ee   Johannes Berg   wext: add checked...
525
526
527
528
529
530
531
532
533
534
  static inline char *
  iwe_stream_add_point_check(struct iw_request_info *info, char *stream,
  			   char *ends, struct iw_event *iwe, char *extra)
  {
  	char *res = iwe_stream_add_point(info, stream, ends, iwe, extra);
  
  	if (res == stream)
  		return ERR_PTR(-E2BIG);
  	return res;
  }
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
535
536
537
538
539
  /*------------------------------------------------------------------*/
  /*
   * Wrapper to add a value to a Wireless Event in a stream of events.
   * Be careful, this one is tricky to use properly :
   * At the first run, you need to have (value = event + IW_EV_LCP_LEN).
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
540
   */
10b2eb694   Johannes Berg   wext: uninline st...
541
542
543
  char *iwe_stream_add_value(struct iw_request_info *info, char *event,
  			   char *value, char *ends, struct iw_event *iwe,
  			   int event_len);
6582c164f   Jean Tourrilhes   [PATCH] WE-19 for...
544

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
545
  #endif	/* _IW_HANDLER_H */