123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977 |
- /*
- * Wireless Tools
- *
- * Jean II - HPLB 97->99 - HPL 99->07
- *
- * Main code for "iwconfig". This is the generic tool for most
- * manipulations...
- * You need to link this code against "iwlib.c" and "-lm".
- *
- * This file is released under the GPL license.
- * Copyright (c) 1997-2007 Jean Tourrilhes <jt@hpl.hp.com>
- */
- #include "iwlib.h" /* Header */
- /************************** DOCUMENTATION **************************/
- /*
- * BASIC PRINCIPLE
- * ---------------
- * Wireless Extension recognise that each wireless device has some
- * specific features not covered by the standard wireless extensions.
- * Private wireless ioctls/requests allow a device to export the control
- * of those device specific features, and allow users to directly interact
- * with your driver.
- * There are many other ways you can implement such functionality :
- * o module parameters
- * o netlink socket
- * o file system (/proc/ or /sysfs/)
- * o extra character device (/dev/)
- * Private wireless ioctls is one of the simplest implementation,
- * however it is limited, so you may want to check the alternatives.
- *
- * Like for standard Wireless Extensions, each private wireless
- * request is identified by an IOCTL NUMBER and carry a certain number
- * of arguments (SET or GET).
- * The driver exports a description of those requests (ioctl number,
- * request name, set and get arguments). Then, iwpriv uses those request
- * descriptions to call the appropriate request and handle the
- * arguments.
- *
- * IOCTL RANGES :
- * ------------
- * The initial implementation of iwpriv was using the SIOCDEVPRIVATE
- * ioctl range (up to 16 ioctls - driver specific). However, this was
- * causing some compatibility problems with other usages of those
- * ioctls, and those ioctls are supposed to be removed.
- * Therefore, I created a new ioctl range, at SIOCIWFIRSTPRIV. Those
- * ioctls are specific to Wireless Extensions, so you don't have to
- * worry about collisions with other usages. On the other hand, in the
- * new range, the SET convention is enforced (see below).
- * The differences are : SIOCDEVPRIVATE SIOCIWFIRSTPRIV
- * o availability <= 2.5.X WE > 11 (>= 2.4.13)
- * o collisions yes no
- * o SET convention optional enforced
- * o number 16 32
- *
- * NEW DRIVER API :
- * --------------
- * Wireless Extension 13 introduces a new driver API. Wireless
- * Extensions requests can be handled via a iw_handler table instead
- * of through the regular ioctl handler.
- * The new driver API can be handled only with the new ioctl range
- * and enforces the GET convention (see below).
- * The differences are : old API new API
- * o handler do_ioctl() struct iw_handler_def
- * o SIOCIWFIRSTPRIV WE > 11 yes
- * o SIOCDEVPRIVATE yes no
- * o GET convention optional enforced
- * Note that the new API before Wireless Extension 15 contains bugs
- * when handling sub-ioctls and addr/float data types.
- *
- * INLINING vs. POINTER :
- * --------------------
- * One of the tricky aspect of the old driver API is how the data
- * is handled, which is how the driver is supposed to extract the data
- * passed to it by iwpriv.
- * 1) If the data has a fixed size (private ioctl definition
- * has the flag IW_PRIV_SIZE_FIXED) and the byte size of the data is
- * lower than 16 bytes, the data will be inlined. The driver can extract
- * data in the field 'u.name' of the struct iwreq.
- * 2) If the if the data doesn't have a fixed size or is larger than
- * 16 bytes, the data is passed by pointer. struct iwreq contains a
- * struct iwpoint with a user space pointer to the data. Appropriate
- * copy_from/to_user() function should be used.
- *
- * With the new API, this is handled transparently, the data is
- * always available as the fourth argument of the request handler
- * (usually called 'extra').
- *
- * SET/GET CONVENTION :
- * ------------------
- * Simplistic summary :
- * o even numbered ioctls are SET, restricted to root, and should not
- * return arguments (get_args = 0).
- * o odd numbered ioctls are GET, authorised to anybody, and should
- * not expect any arguments (set_args = 0).
- *
- * The regular Wireless Extensions use the SET/GET convention, where
- * the low order bit identify a SET (0) or a GET (1) request. The private
- * Wireless Extension is not as restrictive, but still has some
- * limitations.
- * The new ioctl range enforces the SET convention : SET request will
- * be available to root only and can't return any arguments. If you don't
- * like that, just use every other two ioctl.
- * The new driver API enforce the GET convention : GET request won't
- * be able to accept any arguments (except if its fits within (union
- * iwreq_data)). If you don't like that, you can either use the Token Index
- * support or the old API (aka the ioctl handler).
- * In any case, it's a good idea to not have ioctl with both SET
- * and GET arguments. If the GET arguments doesn't fit within
- * (union iwreq_data) and SET do, or vice versa, the current code in iwpriv
- * won't work. One exception is if both SET and GET arguments fit within
- * (union iwreq_data), this case should be handled safely in a GET
- * request.
- * If you don't fully understand those limitations, just follow the
- * rules of the simplistic summary ;-)
- *
- * SUB-IOCTLS :
- * ----------
- * Wireless Extension 15 introduces sub-ioctls. For some applications,
- * 32 ioctls is not enough, and this simple mechanism allows to increase
- * the number of ioctls by adding a sub-ioctl index to some of the ioctls
- * (so basically it's a two level addressing).
- * One might argue that at the point, some other mechanisms might be
- * better, like using a real filesystem abstraction (/proc, driverfs, ...),
- * but sub-ioctls are simple enough and don't have much drawbacks (which
- * means that it's a quick and dirty hack ;-).
- *
- * There are two slightly different variations of the sub-ioctl scheme :
- * 1) If the payload fits within (union iwreq_data), the first int
- * (4 bytes) is reserved as the sub-ioctl number and the regular payload
- * shifted by 4 bytes. The handler must extract the sub-ioctl number,
- * increment the data pointer and then use it in the usual way.
- * 2) If the ioctl uses (struct iw_point), the sub-ioctl number is
- * set in the flags member of the structure. In this case, the handler
- * should simply get the sub-ioctl number from the flags and process the
- * data in the usual way.
- *
- * Sub-ioctls are declared normally in the private definition table,
- * with cmd (first arg) being the sub-ioctl number. Then, you should
- * declare the real ioctl, which will process the sub-ioctls, with
- * the SAME ARGUMENTS and a EMPTY NAME.
- * Here's an example of how it could look like :
- * --------------------------------------------
- // --- sub-ioctls handlers ---
- { 0x8BE0, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, 0, "" },
- { 0x8BE1, 0, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, "" },
- // --- sub-ioctls definitions ---
- { 1, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, 0, "set_param1" },
- { 1, 0, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, "get_param1" },
- { 2, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, 0, "set_param2" },
- { 2, 0, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, "get_param2" },
- // --- Raw access to sub-ioctl handlers ---
- { 0x8BE0, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 2, 0, "set_paramN" },
- { 0x8BE1, IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1,
- IW_PRIV_TYPE_INT | IW_PRIV_SIZE_FIXED | 1, "get_paramN" },
- * --------------------------------------------
- * And iwpriv should do the rest for you ;-)
- *
- * Note that versions of iwpriv up to v24 (included) expect at most
- * 16 ioctls definitions and will likely crash when given more.
- * There is no fix that I can see, apart from recommending your users
- * to upgrade their Wireless Tools. Wireless Extensions 15 will check this
- * condition, so another workaround is restricting those extra definitions
- * to WE-15.
- *
- * Another problem is that the new API before Wireless Extension 15
- * has a bug when passing fixed arguments of 12-15 bytes. It will
- * try to get them inline instead of by pointer. You can fool the new API
- * to do the right thing using fake ioctl definitions (but remember that
- * you will be more likely to hit the limit of 16 ioctl definitions).
- * To play safe, use the old-style ioctl handler before v15.
- *
- * NEW DATA TYPES (ADDR/FLOAT) :
- * ---------------------------
- * Wireless Tools 25 introduce two new data types, addr and float,
- * corresponding to struct sockaddr and struct iwfreq.
- * Those types are properly handled with Wireless Extensions 15.
- * However, the new API before v15 won't handle them properly.
- *
- * The first problem is that the new API won't know their size, so
- * it won't copy them. This can be workaround with a fake ioctl definition.
- * The second problem is that a fixed single addr won't be inlined
- * in struct iwreq and will be passed as a pointer. This is due to an
- * off-by-one error, where all fixed data of 16 bytes is considered too
- * big to fit in struct iwreq.
- *
- * For those reasons, I would recommend to use the ioctl handler
- * before v15 when manipulating those data.
- *
- * TOKEN INDEX :
- * -----------
- * Token index is very similar to sub-ioctl. It allows the user
- * to specify an integer index in front of a bunch of other arguments
- * (addresses, strings, ...). It's specified in square brackets on the
- * iwpriv command line before other arguments.
- * > iwpriv eth0 [index] args...
- * Token index works only when the data is passed as pointer, and
- * is otherwise ignored. If your data would fit within struct iwreq, you
- * should declare the command *without* IW_PRIV_SIZE_FIXED to force
- * this to happen (and check arg number yourself).
- * --------------------------------------------
- // --- Commands that would fit in struct iwreq ---
- { 0x8BE0, IW_PRIV_TYPE_ADDR | 1, 0, "set_param_with_token" },
- // --- No problem here (bigger than struct iwreq) ---
- { 0x8BE1, IW_PRIV_TYPE_ADDR | IW_PRIV_SIZE_FIXED | 2, 0, "again" },
- * --------------------------------------------
- * The token index feature is pretty transparent, the token index
- * will just be in the flags member of (struct iw_point). Default value
- * (if the user doesn't specify it) will be 0. Token index itself will
- * work with any version of Wireless Extensions.
- * Token index is not compatible with sub-ioctl (both use the same
- * field of struct iw_point). However, the token index can be used to offer
- * raw access to the sub-ioctl handlers (if it uses struct iw_point) :
- * --------------------------------------------
- // --- sub-ioctls handler ---
- { 0x8BE0, IW_PRIV_TYPE_ADDR | IW_PRIV_SIZE_FIXED | 1, 0, "" },
- // --- sub-ioctls definitions ---
- { 0, IW_PRIV_TYPE_ADDR | IW_PRIV_SIZE_FIXED | 1, 0, "setaddr" },
- { 1, IW_PRIV_TYPE_ADDR | IW_PRIV_SIZE_FIXED | 1, 0, "deladdr" },
- // --- raw access with token index (+ iwreq workaround) ---
- { 0x8BE0, IW_PRIV_TYPE_ADDR | 1, 0, "rawaddr" },
- * --------------------------------------------
- *
- * Jean II
- */
- /**************************** CONSTANTS ****************************/
- static const char * argtype[] = {
- " ", "byte ", "char ", "", "int ", "float", "addr " };
- /************************* MISC SUBROUTINES **************************/
- /*------------------------------------------------------------------*/
- /*
- * Print usage string
- */
- static void
- iw_usage(void)
- {
- fprintf(stderr, "Usage: iwpriv interface [private-command [private-arguments]]\n");
- }
- /************************* SETTING ROUTINES **************************/
- /*------------------------------------------------------------------*/
- /*
- * Execute a private command on the interface
- */
- static int
- set_private_cmd(int skfd, /* Socket */
- char * args[], /* Command line args */
- int count, /* Args count */
- char * ifname, /* Dev name */
- char * cmdname, /* Command name */
- iwprivargs * priv, /* Private ioctl description */
- int priv_num) /* Number of descriptions */
- {
- struct iwreq wrq;
- u_char buffer[4096]; /* Only that big in v25 and later */
- int i = 0; /* Start with first command arg */
- int k; /* Index in private description table */
- int temp;
- int subcmd = 0; /* sub-ioctl index */
- int offset = 0; /* Space for sub-ioctl index */
- /* Check if we have a token index.
- * Do it now so that sub-ioctl takes precedence, and so that we
- * don't have to bother with it later on... */
- if((count >= 1) && (sscanf(args[0], "[%i]", &temp) == 1))
- {
- subcmd = temp;
- args++;
- count--;
- }
- /* Search the correct ioctl */
- k = -1;
- while((++k < priv_num) && strcmp(priv[k].name, cmdname));
- /* If not found... */
- if(k == priv_num)
- {
- fprintf(stderr, "Invalid command : %s\n", cmdname);
- return(-1);
- }
-
- /* Watch out for sub-ioctls ! */
- if(priv[k].cmd < SIOCDEVPRIVATE)
- {
- int j = -1;
- /* Find the matching *real* ioctl */
- while((++j < priv_num) && ((priv[j].name[0] != '\0') ||
- (priv[j].set_args != priv[k].set_args) ||
- (priv[j].get_args != priv[k].get_args)));
- /* If not found... */
- if(j == priv_num)
- {
- fprintf(stderr, "Invalid private ioctl definition for : %s\n",
- cmdname);
- return(-1);
- }
- /* Save sub-ioctl number */
- subcmd = priv[k].cmd;
- /* Reserve one int (simplify alignment issues) */
- offset = sizeof(__u32);
- /* Use real ioctl definition from now on */
- k = j;
- #if 0
- printf("<mapping sub-ioctl %s to cmd 0x%X-%d>\n", cmdname,
- priv[k].cmd, subcmd);
- #endif
- }
- /* If we have to set some data */
- if((priv[k].set_args & IW_PRIV_TYPE_MASK) &&
- (priv[k].set_args & IW_PRIV_SIZE_MASK))
- {
- switch(priv[k].set_args & IW_PRIV_TYPE_MASK)
- {
- case IW_PRIV_TYPE_BYTE:
- /* Number of args to fetch */
- wrq.u.data.length = count;
- if(wrq.u.data.length > (priv[k].set_args & IW_PRIV_SIZE_MASK))
- wrq.u.data.length = priv[k].set_args & IW_PRIV_SIZE_MASK;
- /* Fetch args */
- for(; i < wrq.u.data.length; i++) {
- sscanf(args[i], "%i", &temp);
- buffer[i] = (char) temp;
- }
- break;
- case IW_PRIV_TYPE_INT:
- /* Number of args to fetch */
- wrq.u.data.length = count;
- if(wrq.u.data.length > (priv[k].set_args & IW_PRIV_SIZE_MASK))
- wrq.u.data.length = priv[k].set_args & IW_PRIV_SIZE_MASK;
- /* Fetch args */
- for(; i < wrq.u.data.length; i++) {
- sscanf(args[i], "%i", &temp);
- ((__s32 *) buffer)[i] = (__s32) temp;
- }
- break;
- case IW_PRIV_TYPE_CHAR:
- if(i < count)
- {
- /* Size of the string to fetch */
- wrq.u.data.length = strlen(args[i]) + 1;
- if(wrq.u.data.length > (priv[k].set_args & IW_PRIV_SIZE_MASK))
- wrq.u.data.length = priv[k].set_args & IW_PRIV_SIZE_MASK;
- /* Fetch string */
- memcpy(buffer, args[i], wrq.u.data.length);
- buffer[sizeof(buffer) - 1] = '\0';
- i++;
- }
- else
- {
- wrq.u.data.length = 1;
- buffer[0] = '\0';
- }
- break;
- case IW_PRIV_TYPE_FLOAT:
- /* Number of args to fetch */
- wrq.u.data.length = count;
- if(wrq.u.data.length > (priv[k].set_args & IW_PRIV_SIZE_MASK))
- wrq.u.data.length = priv[k].set_args & IW_PRIV_SIZE_MASK;
- /* Fetch args */
- for(; i < wrq.u.data.length; i++) {
- double freq;
- if(sscanf(args[i], "%lg", &(freq)) != 1)
- {
- printf("Invalid float [%s]...\n", args[i]);
- return(-1);
- }
- if(strchr(args[i], 'G')) freq *= GIGA;
- if(strchr(args[i], 'M')) freq *= MEGA;
- if(strchr(args[i], 'k')) freq *= KILO;
- sscanf(args[i], "%i", &temp);
- iw_float2freq(freq, ((struct iw_freq *) buffer) + i);
- }
- break;
- case IW_PRIV_TYPE_ADDR:
- /* Number of args to fetch */
- wrq.u.data.length = count;
- if(wrq.u.data.length > (priv[k].set_args & IW_PRIV_SIZE_MASK))
- wrq.u.data.length = priv[k].set_args & IW_PRIV_SIZE_MASK;
- /* Fetch args */
- for(; i < wrq.u.data.length; i++) {
- if(iw_in_addr(skfd, ifname, args[i],
- ((struct sockaddr *) buffer) + i) < 0)
- {
- printf("Invalid address [%s]...\n", args[i]);
- return(-1);
- }
- }
- break;
- default:
- fprintf(stderr, "Not implemented...\n");
- return(-1);
- }
-
- if((priv[k].set_args & IW_PRIV_SIZE_FIXED) &&
- (wrq.u.data.length != (priv[k].set_args & IW_PRIV_SIZE_MASK)))
- {
- printf("The command %s needs exactly %d argument(s)...\n",
- cmdname, priv[k].set_args & IW_PRIV_SIZE_MASK);
- return(-1);
- }
- } /* if args to set */
- else
- {
- wrq.u.data.length = 0L;
- }
- strncpy(wrq.ifr_name, ifname, IFNAMSIZ);
- /* Those two tests are important. They define how the driver
- * will have to handle the data */
- if((priv[k].set_args & IW_PRIV_SIZE_FIXED) &&
- ((iw_get_priv_size(priv[k].set_args) + offset) <= IFNAMSIZ))
- {
- /* First case : all SET args fit within wrq */
- if(offset)
- wrq.u.mode = subcmd;
- memcpy(wrq.u.name + offset, buffer, IFNAMSIZ - offset);
- }
- else
- {
- if((priv[k].set_args == 0) &&
- (priv[k].get_args & IW_PRIV_SIZE_FIXED) &&
- (iw_get_priv_size(priv[k].get_args) <= IFNAMSIZ))
- {
- /* Second case : no SET args, GET args fit within wrq */
- if(offset)
- wrq.u.mode = subcmd;
- }
- else
- {
- /* Third case : args won't fit in wrq, or variable number of args */
- wrq.u.data.pointer = (caddr_t) buffer;
- wrq.u.data.flags = subcmd;
- }
- }
- /* Perform the private ioctl */
- if(ioctl(skfd, priv[k].cmd, &wrq) < 0)
- {
- fprintf(stderr, "Interface doesn't accept private ioctl...\n");
- fprintf(stderr, "%s (%X): %s\n", cmdname, priv[k].cmd, strerror(errno));
- return(-1);
- }
- /* If we have to get some data */
- if((priv[k].get_args & IW_PRIV_TYPE_MASK) &&
- (priv[k].get_args & IW_PRIV_SIZE_MASK))
- {
- int j;
- int n = 0; /* number of args */
- printf("%-8.16s %s:", ifname, cmdname);
- /* Check where is the returned data */
- if((priv[k].get_args & IW_PRIV_SIZE_FIXED) &&
- (iw_get_priv_size(priv[k].get_args) <= IFNAMSIZ))
- {
- memcpy(buffer, wrq.u.name, IFNAMSIZ);
- n = priv[k].get_args & IW_PRIV_SIZE_MASK;
- }
- else
- n = wrq.u.data.length;
- switch(priv[k].get_args & IW_PRIV_TYPE_MASK)
- {
- case IW_PRIV_TYPE_BYTE:
- /* Display args */
- for(j = 0; j < n; j++)
- printf("%d ", buffer[j]);
- printf("\n");
- break;
- case IW_PRIV_TYPE_INT:
- /* Display args */
- for(j = 0; j < n; j++)
- printf("%d ", ((__s32 *) buffer)[j]);
- printf("\n");
- break;
- case IW_PRIV_TYPE_CHAR:
- /* Display args */
- buffer[n] = '\0';
- printf("%s\n", buffer);
- break;
- case IW_PRIV_TYPE_FLOAT:
- {
- double freq;
- /* Display args */
- for(j = 0; j < n; j++)
- {
- freq = iw_freq2float(((struct iw_freq *) buffer) + j);
- if(freq >= GIGA)
- printf("%gG ", freq / GIGA);
- else
- if(freq >= MEGA)
- printf("%gM ", freq / MEGA);
- else
- printf("%gk ", freq / KILO);
- }
- printf("\n");
- }
- break;
- case IW_PRIV_TYPE_ADDR:
- {
- char scratch[128];
- struct sockaddr * hwa;
- /* Display args */
- for(j = 0; j < n; j++)
- {
- hwa = ((struct sockaddr *) buffer) + j;
- if(j)
- printf(" %.*s",
- (int) strlen(cmdname), " ");
- printf("%s\n", iw_saether_ntop(hwa, scratch));
- }
- }
- break;
- default:
- fprintf(stderr, "Not yet implemented...\n");
- return(-1);
- }
- } /* if args to set */
- return(0);
- }
- /*------------------------------------------------------------------*/
- /*
- * Execute a private command on the interface
- */
- static inline int
- set_private(int skfd, /* Socket */
- char * args[], /* Command line args */
- int count, /* Args count */
- char * ifname) /* Dev name */
- {
- iwprivargs * priv;
- int number; /* Max of private ioctl */
- int ret;
- /* Read the private ioctls */
- number = iw_get_priv_info(skfd, ifname, &priv);
- /* Is there any ? */
- if(number <= 0)
- {
- /* Should I skip this message ? */
- fprintf(stderr, "%-8.16s no private ioctls.\n\n",
- ifname);
- if(priv)
- free(priv);
- return(-1);
- }
- /* Do it */
- ret = set_private_cmd(skfd, args + 1, count - 1, ifname, args[0],
- priv, number);
- free(priv);
- return(ret);
- }
- /************************ CATALOG FUNCTIONS ************************/
- /*------------------------------------------------------------------*/
- /*
- * Print on the screen in a neat fashion the list of private ioctls
- * for the device.
- */
- static int
- print_priv_info(int skfd,
- char * ifname,
- char * args[],
- int count)
- {
- int k;
- iwprivargs * priv;
- int n;
- /* Avoid "Unused parameter" warning */
- args = args; count = count;
- /* Read the private ioctls */
- n = iw_get_priv_info(skfd, ifname, &priv);
- /* Is there any ? */
- if(n <= 0)
- {
- /* Should I skip this message ? */
- fprintf(stderr, "%-8.16s no private ioctls.\n\n",
- ifname);
- }
- else
- {
- printf("%-8.16s Available private ioctls :\n", ifname);
- /* Print them all */
- for(k = 0; k < n; k++)
- if(priv[k].name[0] != '\0')
- printf(" %-16.16s (%.4X) : set %3d %s & get %3d %s\n",
- priv[k].name, priv[k].cmd,
- priv[k].set_args & IW_PRIV_SIZE_MASK,
- argtype[(priv[k].set_args & IW_PRIV_TYPE_MASK) >> 12],
- priv[k].get_args & IW_PRIV_SIZE_MASK,
- argtype[(priv[k].get_args & IW_PRIV_TYPE_MASK) >> 12]);
- printf("\n");
- }
- /* Cleanup */
- if(priv)
- free(priv);
- return(0);
- }
- /*------------------------------------------------------------------*/
- /*
- * Print on the screen in a neat fashion the list of private GET ioctl
- * data for the device and data returned by those.
- */
- static int
- print_priv_all(int skfd,
- char * ifname,
- char * args[],
- int count)
- {
- int k;
- iwprivargs * priv;
- int n;
- /* Avoid "Unused parameter" warning */
- args = args; count = count;
- /* Read the private ioctls */
- n = iw_get_priv_info(skfd, ifname, &priv);
- /* Is there any ? */
- if(n <= 0)
- {
- /* Should I skip this message ? */
- fprintf(stderr, "%-8.16s no private ioctls.\n\n",
- ifname);
- }
- else
- {
- printf("%-8.16s Available read-only private ioctl :\n", ifname);
- /* Print them all */
- for(k = 0; k < n; k++)
- /* We call all ioctls that don't have a null name, don't require
- * args and return some (avoid triggering "reset" commands) */
- if((priv[k].name[0] != '\0') && (priv[k].set_args == 0) &&
- (priv[k].get_args != 0))
- set_private_cmd(skfd, NULL, 0, ifname, priv[k].name,
- priv, n);
- printf("\n");
- }
- /* Cleanup */
- if(priv)
- free(priv);
- return(0);
- }
- /********************** PRIVATE IOCTLS MANIPS ***********************/
- /*
- * Convenient access to some private ioctls of some devices
- */
- #if 0
- /*------------------------------------------------------------------*/
- /*
- * Set roaming mode on and off
- * Found in wavelan_cs driver
- * Note : this is obsolete, most 802.11 devices should use the
- * SIOCSIWAP request.
- */
- static int
- set_roaming(int skfd, /* Socket */
- char * args[], /* Command line args */
- int count, /* Args count */
- char * ifname) /* Dev name */
- {
- u_char buffer[1024];
- struct iwreq wrq;
- int i = 0; /* Start with first arg */
- int k;
- iwprivargs * priv;
- int number;
- int roamcmd;
- char RoamState; /* buffer to hold new roam state */
- char ChangeRoamState=0; /* whether or not we are going to
- change roam states */
- /* Read the private ioctls */
- number = iw_get_priv_info(skfd, ifname, &priv);
- /* Is there any ? */
- if(number <= 0)
- {
- /* Should I skip this message ? */
- fprintf(stderr, "%-8.16s no private ioctls.\n\n",
- ifname);
- if(priv)
- free(priv);
- return(-1);
- }
- /* Get the ioctl number */
- k = -1;
- while((++k < number) && strcmp(priv[k].name, "setroam"));
- if(k == number)
- {
- fprintf(stderr, "This device doesn't support roaming\n");
- free(priv);
- return(-1);
- }
- roamcmd = priv[k].cmd;
- /* Cleanup */
- free(priv);
- if(count != 1)
- {
- iw_usage();
- return(-1);
- }
- if(!strcasecmp(args[i], "on"))
- {
- printf("%-8.16s enable roaming\n", ifname);
- if(!number)
- {
- fprintf(stderr, "This device doesn't support roaming\n");
- return(-1);
- }
- ChangeRoamState=1;
- RoamState=1;
- }
- else
- if(!strcasecmp(args[i], "off"))
- {
- i++;
- printf("%-8.16s disable roaming\n", ifname);
- if(!number)
- {
- fprintf(stderr, "This device doesn't support roaming\n");
- return(-1);
- }
- ChangeRoamState=1;
- RoamState=0;
- }
- else
- {
- iw_usage();
- return(-1);
- }
- if(ChangeRoamState)
- {
- strncpy(wrq.ifr_name, ifname, IFNAMSIZ);
- buffer[0]=RoamState;
- memcpy(wrq.u.name, &buffer, IFNAMSIZ);
- if(ioctl(skfd, roamcmd, &wrq) < 0)
- {
- fprintf(stderr, "Roaming support is broken.\n");
- return(-1);
- }
- }
- return(0);
- }
- /*------------------------------------------------------------------*/
- /*
- * Get and set the port type
- * Found in wavelan2_cs and wvlan_cs drivers
- * TODO : Add support for HostAP ?
- */
- static int
- port_type(int skfd, /* Socket */
- char * args[], /* Command line args */
- int count, /* Args count */
- char * ifname) /* Dev name */
- {
- struct iwreq wrq;
- int i = 0; /* Start with first arg */
- int k;
- iwprivargs * priv;
- int number;
- char ptype = 0;
- char * modes[] = { "invalid", "managed (BSS)", "reserved", "ad-hoc" };
- /* Read the private ioctls */
- number = iw_get_priv_info(skfd, ifname, &priv);
- /* Is there any ? */
- if(number <= 0)
- {
- /* Should I skip this message ? */
- fprintf(stderr, "%-8.16s no private ioctls.\n\n", ifname);
- if(priv)
- free(priv);
- return(-1);
- }
- /* Arguments ? */
- if(count == 0)
- {
- /* So, we just want to see the current value... */
- k = -1;
- while((++k < number) && strcmp(priv[k].name, "gport_type") &&
- strcmp(priv[k].name, "get_port"));
- if(k == number)
- {
- fprintf(stderr, "This device doesn't support getting port type\n");
- goto err;
- }
- strncpy(wrq.ifr_name, ifname, IFNAMSIZ);
- /* Get it */
- if(ioctl(skfd, priv[k].cmd, &wrq) < 0)
- {
- fprintf(stderr, "Port type support is broken.\n");
- goto err;
- }
- ptype = *wrq.u.name;
- /* Display it */
- printf("%-8.16s Current port mode is %s <port type is %d>.\n\n",
- ifname, modes[(int) ptype], ptype);
- free(priv);
- return(0);
- }
- if(count != 1)
- {
- iw_usage();
- goto err;
- }
- /* Read it */
- /* As a string... */
- k = 0;
- while((k < 4) && strncasecmp(args[i], modes[k], 2))
- k++;
- if(k < 4)
- ptype = k;
- else
- /* ...or as an integer */
- if(sscanf(args[i], "%i", (int *) &ptype) != 1)
- {
- iw_usage();
- goto err;
- }
-
- k = -1;
- while((++k < number) && strcmp(priv[k].name, "sport_type") &&
- strcmp(priv[k].name, "set_port"));
- if(k == number)
- {
- fprintf(stderr, "This device doesn't support setting port type\n");
- goto err;
- }
- strncpy(wrq.ifr_name, ifname, IFNAMSIZ);
- *(wrq.u.name) = ptype;
- if(ioctl(skfd, priv[k].cmd, &wrq) < 0)
- {
- fprintf(stderr, "Invalid port type (or setting not allowed)\n");
- goto err;
- }
- free(priv);
- return(0);
- err:
- free(priv);
- return(-1);
- }
- #endif
- /******************************* MAIN ********************************/
- /*------------------------------------------------------------------*/
- /*
- * The main !
- */
- int
- main(int argc,
- char ** argv)
- {
- int skfd; /* generic raw socket desc. */
- int goterr = 0;
- /* Create a channel to the NET kernel. */
- if((skfd = iw_sockets_open()) < 0)
- {
- perror("socket");
- return(-1);
- }
- /* No argument : show the list of all devices + ioctl list */
- if(argc == 1)
- iw_enum_devices(skfd, &print_priv_info, NULL, 0);
- else
- /* Special cases take one... */
- /* All */
- if((!strncmp(argv[1], "-a", 2)) || (!strcmp(argv[1], "--all")))
- iw_enum_devices(skfd, &print_priv_all, NULL, 0);
- else
- /* Help */
- if((!strncmp(argv[1], "-h", 2)) || (!strcmp(argv[1], "--help")))
- iw_usage();
- else
- /* Version */
- if (!strcmp(argv[1], "-v") || !strcmp(argv[1], "--version"))
- goterr = iw_print_version_info("iwpriv");
- else
- /* The device name must be the first argument */
- /* Name only : show for that device only */
- if(argc == 2)
- print_priv_info(skfd, argv[1], NULL, 0);
- else
- /* Special cases take two... */
- /* All */
- if((!strncmp(argv[2], "-a", 2)) ||
- (!strcmp(argv[2], "--all")))
- print_priv_all(skfd, argv[1], NULL, 0);
- else
- #if 0
- /* Roaming */
- if(!strncmp(argv[2], "roam", 4))
- goterr = set_roaming(skfd, argv + 3, argc - 3, argv[1]);
- else
- /* Port type */
- if(!strncmp(argv[2], "port", 4))
- goterr = port_type(skfd, argv + 3, argc - 3, argv[1]);
- else
- #endif
- /*-------------*/
- /* Otherwise, it's a private ioctl */
- goterr = set_private(skfd, argv + 2, argc - 2, argv[1]);
- /* Close the socket. */
- iw_sockets_close(skfd);
- return(goterr);
- }
|