1 /* $NetBSD: ifconfig.c,v 1.34 1997/04/21 01:17:58 lukem Exp $ */
2 /* $FreeBSD: src/sbin/ifconfig/ifmedia.c,v 1.19.2.1 2006/03/01 22:24:23 glebius Exp $ */
3 /* $DragonFly: src/sbin/ifconfig/ifmedia.c,v 1.10 2006/04/02 05:33:59 sephe Exp $ */
6 * Copyright (c) 1997 Jason R. Thorpe.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 * must display the following acknowledgement:
19 * This product includes software developed for the NetBSD Project
21 * 4. The name of the author may not be used to endorse or promote products
22 * derived from this software without specific prior written permission.
24 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
25 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
26 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
27 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
28 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
29 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
30 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
31 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
32 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
38 * Copyright (c) 1983, 1993
39 * The Regents of the University of California. All rights reserved.
41 * Redistribution and use in source and binary forms, with or without
42 * modification, are permitted provided that the following conditions
44 * 1. Redistributions of source code must retain the above copyright
45 * notice, this list of conditions and the following disclaimer.
46 * 2. Redistributions in binary form must reproduce the above copyright
47 * notice, this list of conditions and the following disclaimer in the
48 * documentation and/or other materials provided with the distribution.
49 * 3. All advertising materials mentioning features or use of this software
50 * must display the following acknowledgement:
51 * This product includes software developed by the University of
52 * California, Berkeley and its contributors.
53 * 4. Neither the name of the University nor the names of its contributors
54 * may be used to endorse or promote products derived from this software
55 * without specific prior written permission.
57 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
58 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
59 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
60 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
61 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
62 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
63 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
64 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
65 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
66 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
70 #include <sys/param.h>
71 #include <sys/ioctl.h>
72 #include <sys/socket.h>
73 #include <sys/sysctl.h>
77 #include <net/if_dl.h>
78 #include <net/if_types.h>
79 #include <net/if_media.h>
80 #include <net/route.h>
93 static void domediaopt(const char *, int, int);
94 static int get_media_subtype(int, const char *);
95 static int get_media_mode(int, const char *);
96 static int get_media_options(int, const char *);
97 static int lookup_media_word(struct ifmedia_description
*, const char *);
98 static void print_media_word(int, int);
99 static void print_media_word_ifconfig(int);
101 static struct ifmedia_description
*get_toptype_desc(int);
102 static struct ifmedia_type_to_subtype
*get_toptype_ttos(int);
103 static struct ifmedia_description
*get_subtype_desc(int,
104 struct ifmedia_type_to_subtype
*ttos
);
109 struct ifmediareq ifmr
;
112 memset(&ifmr
, 0, sizeof(ifmr
));
113 strncpy(ifmr
.ifm_name
, name
, sizeof(ifmr
.ifm_name
));
115 if (ioctl(s
, SIOCGIFMEDIA
, &ifmr
) < 0) {
117 * Interface doesn't support SIOC{G,S}IFMEDIA.
122 if (ifmr
.ifm_count
== 0) {
123 warnx("%s: no media types?", name
);
127 media_list
= malloc(ifmr
.ifm_count
* sizeof(int));
128 if (media_list
== NULL
)
130 ifmr
.ifm_ulist
= media_list
;
132 if (ioctl(s
, SIOCGIFMEDIA
, &ifmr
) < 0)
133 err(1, "SIOCGIFMEDIA");
136 print_media_word(ifmr
.ifm_current
, 1);
137 if (ifmr
.ifm_active
!= ifmr
.ifm_current
) {
140 print_media_word(ifmr
.ifm_active
, 0);
146 if (ifmr
.ifm_status
& IFM_AVALID
) {
147 printf("\tstatus: ");
148 switch (IFM_TYPE(ifmr
.ifm_active
)) {
150 if (ifmr
.ifm_status
& IFM_ACTIVE
)
153 printf("no carrier");
158 if (ifmr
.ifm_status
& IFM_ACTIVE
)
165 if (ifmr
.ifm_status
& IFM_ACTIVE
)
168 printf("no carrier");
172 /* XXX: Different value for adhoc? */
173 if (ifmr
.ifm_status
& IFM_ACTIVE
)
174 printf("associated");
176 printf("no carrier");
182 if (ifmr
.ifm_count
> 0 && supmedia
) {
183 printf("\tsupported media:\n");
184 for (i
= 0; i
< ifmr
.ifm_count
; i
++) {
186 print_media_word_ifconfig(media_list
[i
]);
194 static struct ifmediareq
*
195 getifmediastate(int s
)
197 static struct ifmediareq
*ifmr
= NULL
;
201 ifmr
= (struct ifmediareq
*)malloc(sizeof(struct ifmediareq
));
205 (void) memset(ifmr
, 0, sizeof(struct ifmediareq
));
206 (void) strncpy(ifmr
->ifm_name
, name
,
207 sizeof(ifmr
->ifm_name
));
210 ifmr
->ifm_ulist
= NULL
;
213 * We must go through the motions of reading all
214 * supported media because we need to know both
215 * the current media type and the top-level type.
218 if (ioctl(s
, SIOCGIFMEDIA
, (caddr_t
)ifmr
) < 0) {
219 err(1, "SIOCGIFMEDIA");
222 if (ifmr
->ifm_count
== 0)
223 errx(1, "%s: no media types?", name
);
225 mwords
= (int *)malloc(ifmr
->ifm_count
* sizeof(int));
229 ifmr
->ifm_ulist
= mwords
;
230 if (ioctl(s
, SIOCGIFMEDIA
, (caddr_t
)ifmr
) < 0)
231 err(1, "SIOCGIFMEDIA");
238 setifmediacallback(int s
, void *arg
)
240 struct ifmediareq
*ifmr
= (struct ifmediareq
*)arg
;
241 static int did_it
= 0;
244 ifr
.ifr_media
= ifmr
->ifm_current
;
245 if (ioctl(s
, SIOCSIFMEDIA
, (caddr_t
)&ifr
) < 0)
246 err(1, "SIOCSIFMEDIA (media)");
247 free(ifmr
->ifm_ulist
);
254 setmedia(const char *val
, int d
, int s
, const struct afswtch
*afp
)
256 struct ifmediareq
*ifmr
;
260 ifmr
= getifmediastate(s
);
263 * We are primarily concerned with the top-level type.
264 * However, "current" may be only IFM_NONE, so we just look
265 * for the top-level type in the first "supported type"
268 * (I'm assuming that all supported media types for a given
269 * interface will be the same top-level type..)
271 subtype
= get_media_subtype(IFM_TYPE(ifmr
->ifm_ulist
[0]), val
);
273 strncpy(ifr
.ifr_name
, name
, sizeof(ifr
.ifr_name
));
274 ifr
.ifr_media
= (ifmr
->ifm_current
& ~(IFM_NMASK
|IFM_TMASK
)) |
275 IFM_TYPE(ifmr
->ifm_ulist
[0]) | subtype
;
277 if ((ifr
.ifr_media
& IFM_TMASK
) == 0) {
278 ifr
.ifr_media
&= ~IFM_GMASK
;
281 ifmr
->ifm_current
= ifr
.ifr_media
;
282 callback_register(setifmediacallback
, (void *)ifmr
);
286 setmediaopt(const char *val
, int d
, int s
, const struct afswtch
*afp
)
289 domediaopt(val
, 0, s
);
293 unsetmediaopt(const char *val
, int d
, int s
, const struct afswtch
*afp
)
296 domediaopt(val
, 1, s
);
300 domediaopt(const char *val
, int clear
, int s
)
302 struct ifmediareq
*ifmr
;
305 ifmr
= getifmediastate(s
);
307 options
= get_media_options(IFM_TYPE(ifmr
->ifm_ulist
[0]), val
);
309 strncpy(ifr
.ifr_name
, name
, sizeof(ifr
.ifr_name
));
310 ifr
.ifr_media
= ifmr
->ifm_current
;
312 ifr
.ifr_media
&= ~options
;
314 ifr
.ifr_media
|= options
;
316 ifmr
->ifm_current
= ifr
.ifr_media
;
317 callback_register(setifmediacallback
, (void *)ifmr
);
322 setmediamode(const char *val
, int d
, int s
, const struct afswtch
*afp
)
324 struct ifmediareq
*ifmr
;
327 ifmr
= getifmediastate(s
);
329 mode
= get_media_mode(IFM_TYPE(ifmr
->ifm_ulist
[0]), val
);
331 strncpy(ifr
.ifr_name
, name
, sizeof(ifr
.ifr_name
));
332 ifr
.ifr_media
= (ifmr
->ifm_current
& ~IFM_MMASK
) | mode
;
334 ifmr
->ifm_current
= ifr
.ifr_media
;
335 callback_register(setifmediacallback
, (void *)ifmr
);
338 /**********************************************************************
339 * A good chunk of this is duplicated from sys/net/ifmedia.c
340 **********************************************************************/
342 static struct ifmedia_description ifm_type_descriptions
[] =
343 IFM_TYPE_DESCRIPTIONS
;
345 static struct ifmedia_description ifm_subtype_ethernet_descriptions
[] =
346 IFM_SUBTYPE_ETHERNET_DESCRIPTIONS
;
348 static struct ifmedia_description ifm_subtype_ethernet_aliases
[] =
349 IFM_SUBTYPE_ETHERNET_ALIASES
;
351 static struct ifmedia_description ifm_subtype_ethernet_option_descriptions
[] =
352 IFM_SUBTYPE_ETHERNET_OPTION_DESCRIPTIONS
;
354 static struct ifmedia_description ifm_subtype_tokenring_descriptions
[] =
355 IFM_SUBTYPE_TOKENRING_DESCRIPTIONS
;
357 static struct ifmedia_description ifm_subtype_tokenring_aliases
[] =
358 IFM_SUBTYPE_TOKENRING_ALIASES
;
360 static struct ifmedia_description ifm_subtype_tokenring_option_descriptions
[] =
361 IFM_SUBTYPE_TOKENRING_OPTION_DESCRIPTIONS
;
363 static struct ifmedia_description ifm_subtype_fddi_descriptions
[] =
364 IFM_SUBTYPE_FDDI_DESCRIPTIONS
;
366 static struct ifmedia_description ifm_subtype_fddi_aliases
[] =
367 IFM_SUBTYPE_FDDI_ALIASES
;
369 static struct ifmedia_description ifm_subtype_fddi_option_descriptions
[] =
370 IFM_SUBTYPE_FDDI_OPTION_DESCRIPTIONS
;
372 static struct ifmedia_description ifm_subtype_ieee80211_descriptions
[] =
373 IFM_SUBTYPE_IEEE80211_DESCRIPTIONS
;
375 static struct ifmedia_description ifm_subtype_ieee80211_aliases
[] =
376 IFM_SUBTYPE_IEEE80211_ALIASES
;
378 static struct ifmedia_description ifm_subtype_ieee80211_option_descriptions
[] =
379 IFM_SUBTYPE_IEEE80211_OPTION_DESCRIPTIONS
;
381 struct ifmedia_description ifm_subtype_ieee80211_mode_descriptions
[] =
382 IFM_SUBTYPE_IEEE80211_MODE_DESCRIPTIONS
;
384 struct ifmedia_description ifm_subtype_ieee80211_mode_aliases
[] =
385 IFM_SUBTYPE_IEEE80211_MODE_ALIASES
;
387 static struct ifmedia_description ifm_subtype_atm_descriptions
[] =
388 IFM_SUBTYPE_ATM_DESCRIPTIONS
;
390 static struct ifmedia_description ifm_subtype_atm_aliases
[] =
391 IFM_SUBTYPE_ATM_ALIASES
;
393 static struct ifmedia_description ifm_subtype_atm_option_descriptions
[] =
394 IFM_SUBTYPE_ATM_OPTION_DESCRIPTIONS
;
396 static struct ifmedia_description ifm_subtype_shared_descriptions
[] =
397 IFM_SUBTYPE_SHARED_DESCRIPTIONS
;
399 static struct ifmedia_description ifm_subtype_shared_aliases
[] =
400 IFM_SUBTYPE_SHARED_ALIASES
;
402 static struct ifmedia_description ifm_shared_option_descriptions
[] =
403 IFM_SHARED_OPTION_DESCRIPTIONS
;
405 struct ifmedia_type_to_subtype
{
407 struct ifmedia_description
*desc
;
411 struct ifmedia_description
*desc
;
415 struct ifmedia_description
*desc
;
420 /* must be in the same order as IFM_TYPE_DESCRIPTIONS */
421 static struct ifmedia_type_to_subtype ifmedia_types_to_subtypes
[] = {
424 { &ifm_subtype_shared_descriptions
[0], 0 },
425 { &ifm_subtype_shared_aliases
[0], 1 },
426 { &ifm_subtype_ethernet_descriptions
[0], 0 },
427 { &ifm_subtype_ethernet_aliases
[0], 1 },
431 { &ifm_shared_option_descriptions
[0], 0 },
432 { &ifm_subtype_ethernet_option_descriptions
[0], 0 },
441 { &ifm_subtype_shared_descriptions
[0], 0 },
442 { &ifm_subtype_shared_aliases
[0], 1 },
443 { &ifm_subtype_tokenring_descriptions
[0], 0 },
444 { &ifm_subtype_tokenring_aliases
[0], 1 },
448 { &ifm_shared_option_descriptions
[0], 0 },
449 { &ifm_subtype_tokenring_option_descriptions
[0], 0 },
458 { &ifm_subtype_shared_descriptions
[0], 0 },
459 { &ifm_subtype_shared_aliases
[0], 1 },
460 { &ifm_subtype_fddi_descriptions
[0], 0 },
461 { &ifm_subtype_fddi_aliases
[0], 1 },
465 { &ifm_shared_option_descriptions
[0], 0 },
466 { &ifm_subtype_fddi_option_descriptions
[0], 0 },
475 { &ifm_subtype_shared_descriptions
[0], 0 },
476 { &ifm_subtype_shared_aliases
[0], 1 },
477 { &ifm_subtype_ieee80211_descriptions
[0], 0 },
478 { &ifm_subtype_ieee80211_aliases
[0], 1 },
482 { &ifm_shared_option_descriptions
[0], 0 },
483 { &ifm_subtype_ieee80211_option_descriptions
[0], 0 },
487 { &ifm_subtype_ieee80211_mode_descriptions
[0], 0 },
488 { &ifm_subtype_ieee80211_mode_aliases
[0], 0 },
494 { &ifm_subtype_shared_descriptions
[0], 0 },
495 { &ifm_subtype_shared_aliases
[0], 1 },
496 { &ifm_subtype_atm_descriptions
[0], 0 },
497 { &ifm_subtype_atm_aliases
[0], 1 },
501 { &ifm_shared_option_descriptions
[0], 0 },
502 { &ifm_subtype_atm_option_descriptions
[0], 0 },
512 get_media_subtype(int type
, const char *val
)
514 struct ifmedia_description
*desc
;
515 struct ifmedia_type_to_subtype
*ttos
;
518 /* Find the top-level interface type. */
519 for (desc
= ifm_type_descriptions
, ttos
= ifmedia_types_to_subtypes
;
520 desc
->ifmt_string
!= NULL
; desc
++, ttos
++)
521 if (type
== desc
->ifmt_word
)
523 if (desc
->ifmt_string
== NULL
)
524 errx(1, "unknown media type 0x%x", type
);
526 for (i
= 0; ttos
->subtypes
[i
].desc
!= NULL
; i
++) {
527 rval
= lookup_media_word(ttos
->subtypes
[i
].desc
, val
);
531 errx(1, "unknown media subtype: %s", val
);
536 get_media_mode(int type
, const char *val
)
538 struct ifmedia_description
*desc
;
539 struct ifmedia_type_to_subtype
*ttos
;
542 /* Find the top-level interface type. */
543 for (desc
= ifm_type_descriptions
, ttos
= ifmedia_types_to_subtypes
;
544 desc
->ifmt_string
!= NULL
; desc
++, ttos
++)
545 if (type
== desc
->ifmt_word
)
547 if (desc
->ifmt_string
== NULL
)
548 errx(1, "unknown media mode 0x%x", type
);
550 for (i
= 0; ttos
->modes
[i
].desc
!= NULL
; i
++) {
551 rval
= lookup_media_word(ttos
->modes
[i
].desc
, val
);
559 get_media_options(int type
, const char *val
)
561 struct ifmedia_description
*desc
;
562 struct ifmedia_type_to_subtype
*ttos
;
563 char *optlist
, *optptr
;
564 int option
= 0, i
, rval
= 0;
566 /* We muck with the string, so copy it. */
567 optlist
= strdup(val
);
571 /* Find the top-level interface type. */
572 for (desc
= ifm_type_descriptions
, ttos
= ifmedia_types_to_subtypes
;
573 desc
->ifmt_string
!= NULL
; desc
++, ttos
++)
574 if (type
== desc
->ifmt_word
)
576 if (desc
->ifmt_string
== NULL
)
577 errx(1, "unknown media type 0x%x", type
);
580 * Look up the options in the user-provided comma-separated
584 for (; (optptr
= strtok(optptr
, ",")) != NULL
; optptr
= NULL
) {
585 for (i
= 0; ttos
->options
[i
].desc
!= NULL
; i
++) {
586 option
= lookup_media_word(ttos
->options
[i
].desc
, optptr
);
591 errx(1, "unknown option: %s", optptr
);
600 lookup_media_word(struct ifmedia_description
*desc
, const char *val
)
603 for (; desc
->ifmt_string
!= NULL
; desc
++)
604 if (strcasecmp(desc
->ifmt_string
, val
) == 0)
605 return (desc
->ifmt_word
);
610 static struct ifmedia_description
*get_toptype_desc(int ifmw
)
612 struct ifmedia_description
*desc
;
614 for (desc
= ifm_type_descriptions
; desc
->ifmt_string
!= NULL
; desc
++)
615 if (IFM_TYPE(ifmw
) == desc
->ifmt_word
)
621 static struct ifmedia_type_to_subtype
*get_toptype_ttos(int ifmw
)
623 struct ifmedia_description
*desc
;
624 struct ifmedia_type_to_subtype
*ttos
;
626 for (desc
= ifm_type_descriptions
, ttos
= ifmedia_types_to_subtypes
;
627 desc
->ifmt_string
!= NULL
; desc
++, ttos
++)
628 if (IFM_TYPE(ifmw
) == desc
->ifmt_word
)
634 static struct ifmedia_description
*get_subtype_desc(int ifmw
,
635 struct ifmedia_type_to_subtype
*ttos
)
638 struct ifmedia_description
*desc
;
640 for (i
= 0; ttos
->subtypes
[i
].desc
!= NULL
; i
++) {
641 if (ttos
->subtypes
[i
].alias
)
643 for (desc
= ttos
->subtypes
[i
].desc
;
644 desc
->ifmt_string
!= NULL
; desc
++) {
645 if (IFM_SUBTYPE(ifmw
) == desc
->ifmt_word
)
653 static struct ifmedia_description
*get_mode_desc(int ifmw
,
654 struct ifmedia_type_to_subtype
*ttos
)
657 struct ifmedia_description
*desc
;
659 for (i
= 0; ttos
->modes
[i
].desc
!= NULL
; i
++) {
660 if (ttos
->modes
[i
].alias
)
662 for (desc
= ttos
->modes
[i
].desc
;
663 desc
->ifmt_string
!= NULL
; desc
++) {
664 if (IFM_MODE(ifmw
) == desc
->ifmt_word
)
673 print_media_word(int ifmw
, int print_toptype
)
675 struct ifmedia_description
*desc
;
676 struct ifmedia_type_to_subtype
*ttos
;
677 int seen_option
= 0, i
;
679 /* Find the top-level interface type. */
680 desc
= get_toptype_desc(ifmw
);
681 ttos
= get_toptype_ttos(ifmw
);
682 if (desc
->ifmt_string
== NULL
) {
683 printf("<unknown type>");
685 } else if (print_toptype
) {
686 printf("%s", desc
->ifmt_string
);
690 * Don't print the top-level type; it's not like we can
691 * change it, or anything.
695 desc
= get_subtype_desc(ifmw
, ttos
);
699 /* Falling to here means unknown subtype. */
700 printf("<unknown subtype>");
707 printf("%s", desc
->ifmt_string
);
710 desc
= get_mode_desc(ifmw
, ttos
);
711 if (desc
!= NULL
&& strcasecmp("autoselect", desc
->ifmt_string
))
712 printf(" mode %s", desc
->ifmt_string
);
716 for (i
= 0; ttos
->options
[i
].desc
!= NULL
; i
++) {
717 if (ttos
->options
[i
].alias
)
719 for (desc
= ttos
->options
[i
].desc
;
720 desc
->ifmt_string
!= NULL
; desc
++) {
721 if (ifmw
& desc
->ifmt_word
) {
722 if (seen_option
== 0)
724 printf("%s%s", seen_option
++ ? "," : "",
729 printf("%s", seen_option
? ">" : "");
733 print_media_word_ifconfig(int ifmw
)
735 struct ifmedia_description
*desc
;
736 struct ifmedia_type_to_subtype
*ttos
;
739 /* Find the top-level interface type. */
740 desc
= get_toptype_desc(ifmw
);
741 ttos
= get_toptype_ttos(ifmw
);
742 if (desc
->ifmt_string
== NULL
) {
743 printf("<unknown type>");
748 * Don't print the top-level type; it's not like we can
749 * change it, or anything.
753 desc
= get_subtype_desc(ifmw
, ttos
);
757 /* Falling to here means unknown subtype. */
758 printf("<unknown subtype>");
762 printf("media %s", desc
->ifmt_string
);
764 desc
= get_mode_desc(ifmw
, ttos
);
766 printf(" mode %s", desc
->ifmt_string
);
769 for (i
= 0; ttos
->options
[i
].desc
!= NULL
; i
++) {
770 if (ttos
->options
[i
].alias
)
772 for (desc
= ttos
->options
[i
].desc
;
773 desc
->ifmt_string
!= NULL
; desc
++) {
774 if (ifmw
& desc
->ifmt_word
) {
775 printf(" mediaopt %s", desc
->ifmt_string
);
781 /**********************************************************************
783 **********************************************************************/
785 static struct cmd media_cmds
[] = {
786 DEF_CMD_ARG("media", setmedia
),
787 DEF_CMD_ARG("mode", setmediamode
),
788 DEF_CMD_ARG("mediaopt", setmediaopt
),
789 DEF_CMD_ARG("-mediaopt",unsetmediaopt
),
791 static struct afswtch af_media
= {
792 .af_name
= "af_media",
794 .af_other_status
= media_status
,
797 static __constructor
void
800 #define N(a) (sizeof(a) / sizeof(a[0]))
803 for (i
= 0; i
< N(media_cmds
); i
++)
804 cmd_register(&media_cmds
[i
]);
805 af_register(&af_media
);