Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src ifconfig: Report link state even if media is not supported



details:   https://anonhg.NetBSD.org/src/rev/d6915ffdf548
branches:  trunk
changeset: 1014426:d6915ffdf548
user:      roy <roy%NetBSD.org@localhost>
date:      Tue Sep 22 14:14:17 2020 +0000

description:
ifconfig: Report link state even if media is not supported

For AF_LINK addrs from getifaddrs(2), ifa_data is struct if_data.
This in turn holds ifi_link_state which we can use to report
link status if the interface does not support media where it's normally
reported.

Based on OpenBSD.

diffstat:

 sbin/ifconfig/ifconfig.c |  22 +++++-----
 sbin/ifconfig/media.c    |  89 ++++++++++++++++++++++++++++++-----------------
 sbin/ifconfig/media.h    |   4 +-
 sys/net/if.h             |  27 ++++++++++++++-
 4 files changed, 96 insertions(+), 46 deletions(-)

diffs (270 lines):

diff -r 20ac9100a9a3 -r d6915ffdf548 sbin/ifconfig/ifconfig.c
--- a/sbin/ifconfig/ifconfig.c  Tue Sep 22 12:21:11 2020 +0000
+++ b/sbin/ifconfig/ifconfig.c  Tue Sep 22 14:14:17 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: ifconfig.c,v 1.242 2020/06/07 06:02:58 thorpej Exp $   */
+/*     $NetBSD: ifconfig.c,v 1.243 2020/09/22 14:14:17 roy Exp $       */
 
 /*-
  * Copyright (c) 1997, 1998, 2000 The NetBSD Foundation, Inc.
@@ -63,7 +63,7 @@
 #ifndef lint
 __COPYRIGHT("@(#) Copyright (c) 1983, 1993\
  The Regents of the University of California.  All rights reserved.");
-__RCSID("$NetBSD: ifconfig.c,v 1.242 2020/06/07 06:02:58 thorpej Exp $");
+__RCSID("$NetBSD: ifconfig.c,v 1.243 2020/09/22 14:14:17 roy Exp $");
 #endif /* not lint */
 
 #include <sys/param.h>
@@ -135,7 +135,7 @@
 static int unsetlinkstr(prop_dictionary_t, prop_dictionary_t);
 static int setifdescr(prop_dictionary_t, prop_dictionary_t);
 static int unsetifdescr(prop_dictionary_t, prop_dictionary_t);
-static void status(const struct sockaddr *, prop_dictionary_t,
+static void status(const struct sockaddr_dl *, prop_dictionary_t,
     prop_dictionary_t);
 __dead static void usage(void);
 
@@ -851,7 +851,7 @@
 {
        struct ifaddrs *ifap, *ifa;
        struct ifreq ifr;
-       const struct sockaddr *sdl = NULL;
+       const struct sockaddr_dl *sdl = NULL;
        prop_dictionary_t env, oenv;
        int idx;
        char *p;
@@ -881,7 +881,7 @@
                if (ifname != NULL && strcmp(ifname, ifa->ifa_name) != 0)
                        continue;
                if (ifa->ifa_addr->sa_family == AF_LINK)
-                       sdl = ifa->ifa_addr;
+                       sdl = (const struct sockaddr_dl *)ifa->ifa_addr;
                if (p && strcmp(p, ifa->ifa_name) == 0)
                        continue;
                if (!prop_dictionary_set_string(env, "if", ifa->ifa_name))
@@ -1265,7 +1265,7 @@
 #define MAX_PRINT_LEN 58       /* XXX need a better way to determine this! */
 
 void
-status(const struct sockaddr *sdl, prop_dictionary_t env,
+status(const struct sockaddr_dl *sdl, prop_dictionary_t env,
     prop_dictionary_t oenv)
 {
        const struct if_data *ifi;
@@ -1360,11 +1360,6 @@
                free(p);
        }
 
-       media_status(env, oenv);
-
-       if (!vflag && !zflag)
-               goto proto_status;
-
        estrlcpy(ifdr.ifdr_name, ifname, sizeof(ifdr.ifdr_name));
 
        if (prog_ioctl(s, zflag ? SIOCZIFDATA : SIOCGIFDATA, &ifdr) == -1)
@@ -1372,6 +1367,11 @@
 
        ifi = &ifdr.ifdr_data;
 
+       media_status(sdl->sdl_type, ifi->ifi_link_state, env, oenv);
+
+       if (!vflag && !zflag)
+               goto proto_status;
+
        print_plural("\tinput: ", ifi->ifi_ipackets, "packet");
        print_human_bytes(hflag, ifi->ifi_ibytes);
        if (ifi->ifi_imcasts)
diff -r 20ac9100a9a3 -r d6915ffdf548 sbin/ifconfig/media.c
--- a/sbin/ifconfig/media.c     Tue Sep 22 12:21:11 2020 +0000
+++ b/sbin/ifconfig/media.c     Tue Sep 22 14:14:17 2020 +0000
@@ -1,6 +1,6 @@
 #include <sys/cdefs.h>
 #ifndef lint
-__RCSID("$NetBSD: media.c,v 1.9 2020/06/07 06:02:58 thorpej Exp $");
+__RCSID("$NetBSD: media.c,v 1.10 2020/09/22 14:14:17 roy Exp $");
 #endif /* not lint */
 
 #include <assert.h>
@@ -16,6 +16,7 @@
 #include <net/if.h>
 #include <net/if_dl.h>
 #include <net/if_media.h>
+#include <net/if_types.h>
 
 #include <prop/proplib.h>
 
@@ -52,6 +53,9 @@
 static const struct ifmedia_status_description ifm_status_descriptions[] =
     IFM_STATUS_DESCRIPTIONS;
 
+const struct if_status_description if_status_descriptions[] =
+       LINK_STATE_DESCRIPTIONS;
+
 static struct pstr mediamode = PSTR_INITIALIZER1(&mediamode, "mediamode",
     setmediamode, "mediamode", false, &command_root.pb_parser);
 
@@ -344,8 +348,55 @@
                printf(" instance %d", IFM_INST(ifmw));
 }
 
+static void
+print_link_status(int media_type, int link_state)
+{
+       const struct if_status_description *p;
+
+       printf("\tstatus: ");
+       for (p = if_status_descriptions; p->ifs_string != NULL; p++) {
+               if (LINK_STATE_DESC_MATCH(p, media_type, link_state)) {
+                       printf("%s\n", p->ifs_string);
+                       return;
+               }
+       }
+       printf("[#%d]\n", link_state);
+}
+
+static void
+print_media_status(int media_type, int media_status)
+{
+       const struct ifmedia_status_description *ifms;
+       int bitno, found = 0;
+
+       printf("\tstatus: ");
+       for (bitno = 0; ifm_status_valid_list[bitno] != 0; bitno++) {
+               for (ifms = ifm_status_descriptions;
+                    ifms->ifms_valid != 0; ifms++) {
+                       if (ifms->ifms_type != media_type ||
+                           ifms->ifms_valid != ifm_status_valid_list[bitno])
+                               continue;
+                       printf("%s%s", found ? ", " : "",
+                           IFM_STATUS_DESC(ifms, media_status));
+                       found = 1;
+
+                       /*
+                        * For each valid indicator bit, there's
+                        * only one entry for each media type, so
+                        * terminate the inner loop now.
+                        */
+                       break;
+               }
+       }
+
+       if (found == 0)
+               printf("unknown");
+       printf("\n");
+}
+
 void
-media_status(prop_dictionary_t env, prop_dictionary_t oenv)
+media_status(int media_type, int link_state,
+    prop_dictionary_t env, prop_dictionary_t oenv)
 {
        struct ifmediareq ifmr;
        int af, i, s;
@@ -368,6 +419,8 @@
                /*
                 * Interface doesn't support SIOC{G,S}IFMEDIA.
                 */
+               if (link_state != LINK_STATE_UNKNOWN)
+                       print_link_status(media_type, link_state);
                return;
        }
 
@@ -393,36 +446,8 @@
        } else
                media_list = NULL;
 
-       if (ifmr.ifm_status & IFM_STATUS_VALID) {
-               const struct ifmedia_status_description *ifms;
-               int bitno, found = 0;
-
-               printf("\tstatus: ");
-               for (bitno = 0; ifm_status_valid_list[bitno] != 0; bitno++) {
-                       for (ifms = ifm_status_descriptions;
-                            ifms->ifms_valid != 0; ifms++) {
-                               if (ifms->ifms_type !=
-                                     IFM_TYPE(ifmr.ifm_current) ||
-                                   ifms->ifms_valid !=
-                                     ifm_status_valid_list[bitno])
-                                       continue;
-                               printf("%s%s", found ? ", " : "",
-                                   IFM_STATUS_DESC(ifms, ifmr.ifm_status));
-                               found = 1;
-
-                               /*
-                                * For each valid indicator bit, there's
-                                * only one entry for each media type, so
-                                * terminate the inner loop now.
-                                */
-                               break;
-                       }
-               }
-
-               if (found == 0)
-                       printf("unknown");
-               printf("\n");
-       }
+       if (ifmr.ifm_status & IFM_STATUS_VALID)
+               print_media_status(IFM_TYPE(ifmr.ifm_current), ifmr.ifm_status);
 
        if (get_flag('m')) {
                int type, printed_type;
diff -r 20ac9100a9a3 -r d6915ffdf548 sbin/ifconfig/media.h
--- a/sbin/ifconfig/media.h     Tue Sep 22 12:21:11 2020 +0000
+++ b/sbin/ifconfig/media.h     Tue Sep 22 14:14:17 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: media.h,v 1.1 2008/07/02 07:44:15 dyoung Exp $ */
+/*     $NetBSD: media.h,v 1.2 2020/09/22 14:14:17 roy Exp $    */
 
 #ifndef        _IFCONFIG_MEDIA_H
 #define        _IFCONFIG_MEDIA_H
@@ -11,6 +11,6 @@
 
 void   print_media_word(int, const char *);
 void   process_media_commands(prop_dictionary_t);
-void   media_status(prop_dictionary_t, prop_dictionary_t);
+void   media_status(int, int, prop_dictionary_t, prop_dictionary_t);
 
 #endif /* _IFCONFIG_MEDIA_H */
diff -r 20ac9100a9a3 -r d6915ffdf548 sys/net/if.h
--- a/sys/net/if.h      Tue Sep 22 12:21:11 2020 +0000
+++ b/sys/net/if.h      Tue Sep 22 14:14:17 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if.h,v 1.284 2020/08/28 06:23:42 ozaki-r Exp $ */
+/*     $NetBSD: if.h,v 1.285 2020/09/22 14:14:17 roy Exp $     */
 
 /*-
  * Copyright (c) 1999, 2000, 2001 The NetBSD Foundation, Inc.
@@ -208,6 +208,31 @@
 #define        LINK_STATE_UP           2       /* link is up */
 
 /*
+ * Status bit descriptions for the various interface types.
+ */
+struct if_status_description {
+       unsigned char   ifs_type;
+       unsigned char   ifs_state;
+       const char      *ifs_string;
+};
+
+#define LINK_STATE_DESC_MATCH(_ifs, _t, _s)                            \
+       (((_ifs)->ifs_type == (_t) || (_ifs)->ifs_type == 0) &&         \
+           (_ifs)->ifs_state == (_s))
+
+#define LINK_STATE_DESCRIPTIONS {                                      \
+       { IFT_ETHER, LINK_STATE_DOWN, "no carrier" },                   \
+       { IFT_IEEE80211, LINK_STATE_DOWN, "no network" },               \
+       { IFT_PPP, LINK_STATE_DOWN, "no carrier" },                     \
+       { IFT_CARP, LINK_STATE_DOWN, "backup" },                        \
+       { IFT_CARP, LINK_STATE_UP, "master" },                          \
+       { 0, LINK_STATE_UP, "active" },                                 \
+       { 0, LINK_STATE_UNKNOWN, "unknown" },                           \
+       { 0, LINK_STATE_DOWN, "down" },                                 \
+       { 0, 0, NULL }                                                  \
+}
+
+/*
  * Structure defining a queue for a network interface.
  */
 struct ifqueue {



Home | Main Index | Thread Index | Old Index