/* * $Id: wmifinfo.c,v 1.4 2004/07/11 12:00:46 ico Exp $ */ #include <stdio.h> #include <unistd.h> #ifdef linux #include <getopt.h> #endif #include <stdlib.h> #include <string.h> #include <ctype.h> #include <time.h> #include <X11/X.h> #include <X11/xpm.h> #include <netinet/in.h> #include <arpa/inet.h> #include <sys/ioctl.h> #include <sys/socket.h> #include <net/if.h> #if defined(__OpenBSD__) #include <net/if_dl.h> #include <net/if_types.h> #include <net/route.h> #include <sys/param.h> #include <sys/sysctl.h> #include <netinet/if_ether.h> #include <net/if_ieee80211.h> #include <dev/ic/if_wi_ieee.h> #include <dev/ic/if_wireg.h> #include <dev/ic/if_wi_hostap.h> #define ROUNDUP(a) \ ((a) > 0 ? (1 + (((a) - 1) | (sizeof(long) - 1))) : sizeof(long)) #endif #include <signal.h> #include <sys/wait.h> #include <sys/types.h> #include <errno.h> #ifdef ENABLE_NWN_SUPPORT #include "nwn.h" #endif #include "xutils.h" #include "bitmaps/wmifinfo_led.xbm" #include "bitmaps/wmifinfo_led.xpm" #include "bitmaps/wmifinfo_lcd.xbm" #include "bitmaps/wmifinfo_lcd.xpm" #define MAXIFS 10 #ifdef linux #define DELAY 1000000L #define USEC_PER_SEC 1000000L #else #define DELAY 100000L #endif #define MODE_LED 1 #define MODE_LCD 2 struct ifinfo_t { char id[16]; int state; unsigned char hw[6]; uint32_t ip; uint32_t nm; uint32_t gw; int sl; int bytes; }; struct font_t { char *chars; int sx; int sy; int dx; int dy; int charspline; }; char LabelColor[30] = "#79bdbf"; char WindGustColor[30] = "#ff0000"; char DataColor[30] = "#ffbf50"; char BackColor[30] = "#181818"; char StationTimeColor[30] = "#c5a6ff"; struct font_t font1 = { " 0123456789ABCDEF", 64, 74, 4, 5, 17}; struct font_t font2 = { "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789: ", 1, 65, 5, 7, 26}; char *exec_dflt = "echo Use the -u and -d options for setting ifup/ifdown commands."; char *exec_up; char *exec_down; int mode = MODE_LED; char startif[16] = ""; char ifname[MAXIFS][16]; int ifaces; int ifno = 0; struct ifinfo_t ifinfo; int fd = 0; struct ifconf ifc; volatile int exec_busy=0; void parse_cmdline(int argc, char *argv[]); void ButtonPressEvent(XButtonEvent *); char *strupper(char *str); void getifnames(void); int getifinfo(char *ifname, struct ifinfo_t *info); void drawtext(char *str, struct font_t *font, int x, int y) { int i = 0; int ix, iy; char *p; while(str[i]) { p = strchr(font->chars, str[i]); ix = (p) ? (p - font->chars) : 0; iy = (ix / font->charspline); ix = (ix % font->charspline); copyXPMArea( font->sx + ix * font->dx, font->sy + iy * font->dy, font->dx, font->dy, x + font->dx * i, y); i++; } } void drawipaddr(uint32_t a, int linenum) { char buf[4]; int i; uint32_t addr = ntohl(a); for(i=0; i<4; i++) { snprintf(buf, 4, "%3d", (addr >> ((3-i)*8)) & 255); drawtext(buf, &font1, 5 + i*14, 19 + linenum*9); } } void drawhwaddr(unsigned char *addr) { char buf[4]; int i; for(i=0; i<6; i++) { snprintf(buf, 4, "%02X", addr[i]); drawtext(buf, &font1, 6 + i*9, 46); } } int main(int argc, char *argv[]) { XEvent event; char buf[16]; int d=0; int pifno=-1; int lastbytes=0; struct timeval tv; fd_set fds; int x; int prev_exec_busy=0; exec_up = exec_dflt; exec_down = exec_dflt; parse_cmdline(argc, argv); initXwindow(argc, argv); if(mode == MODE_LED) { openXwindow(argc, argv, (char **) wmifinfo_led_xpm, (char*) wmifinfo_led_bits, wmifinfo_led_width, wmifinfo_led_height, BackColor, LabelColor, WindGustColor, DataColor, StationTimeColor); } else { openXwindow(argc, argv, (char **) wmifinfo_lcd_xpm, (char*) wmifinfo_lcd_bits, wmifinfo_lcd_width, wmifinfo_lcd_height, BackColor, LabelColor, WindGustColor, DataColor, StationTimeColor); } /* Initialize global variables */ fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_IP); ifc.ifc_len = sizeof(struct ifreq) * 10; ifc.ifc_buf = malloc(ifc.ifc_len); while(1) { while (XPending(display)) { XNextEvent(display, &event); switch (event.type) { case Expose: RedrawWindow(); break; case ButtonPress: ButtonPressEvent(&event.xbutton); break; case ButtonRelease: break; } } if ((d++ == 3) || (ifno != pifno) || (exec_busy != prev_exec_busy)) { d=0; copyXPMArea(64, 0, 64, 64, 0, 0); getifnames(); if(ifaces>0) { ifno = ifno % ifaces; getifinfo(ifname[ifno], &ifinfo); if(ifno != pifno) lastbytes = ifinfo.bytes; sprintf(buf, "%-7s", ifinfo.id); strupper(buf); drawtext(buf, &font2, 6, 4); if(memcmp(ifinfo.hw, "\x00\x00\x00\x00\x00\x00", 6) != 0) { drawhwaddr(ifinfo.hw); } if(ifinfo.ip) drawipaddr(ifinfo.ip, 0); if(ifinfo.nm) drawipaddr(ifinfo.nm, 1); if(ifinfo.gw) drawipaddr(ifinfo.gw, 2); /* WLAN signal level */ #ifdef linux x = ifinfo.sl/4; #elif defined(__OpenBSD__) x = ifinfo.sl/7; #endif if(x>13) x=13; copyXPMArea(4, 82, x*4, 4, 6, 53); /* LED */ x=0; if(exec_busy) { x=0; } else { if(ifinfo.state) x += 8; if(lastbytes == ifinfo.bytes) x+= 16; } copyXPMArea(64 + x, 81, 8, 8, 50, 4); lastbytes = ifinfo.bytes; } RedrawWindow(); prev_exec_busy = exec_busy; pifno = ifno; } #ifdef linux tv.tv_sec = DELAY / USEC_PER_SEC; tv.tv_usec = DELAY - (tv.tv_sec * USEC_PER_SEC); FD_ZERO(&fds); FD_SET(ConnectionNumber(display), &fds); select(ConnectionNumber(display)+1, &fds, NULL, NULL, &tv); #elif defined(__OpenBSD__) usleep(DELAY); #endif } } void print_usage() { printf( "usage: wmifinfo [-lh] [-i interface]\n" " -d <cmd> Command to exec for iface-down\n" " -i <interface> Start with given interface, if available\n" " -l LCD display mode\n" " -h Print this help\n" " -u <cmd> Command to exec for iface-up\n" " -v Show version info and exit\n" ); } void print_version() { printf("%s version %s, compile-time options: ", NAME, VERSION); #ifdef ENABLE_NWN_SUPPORT printf("ENABLE NWN SUPPORT "); #endif printf("\n"); } void parse_cmdline(int argc, char *argv[]) { int c; while((c = getopt(argc, argv, "d:i:lhu:v")) != EOF) { switch(c) { case 'd' : exec_down = strdup(optarg); break; case 'i' : strncpy(startif, optarg, sizeof(startif)); break; case 'l' : mode = MODE_LCD; break; case 'h' : print_usage(); exit(0); case 'u' : exec_up = strdup(optarg); break; case 'v' : print_version(); exit(0); } } } void sigchldhandler(int a) { wait(NULL); exec_busy = 0; } void exec_cmd(char *cmd) { int pid; signal(SIGCHLD, sigchldhandler); if(exec_busy) return; exec_busy=1; pid=fork(); if(pid == 0) { if (system(cmd) == -1) { fprintf(stderr, "%s failed\n", cmd); exit(1); } exit(0); } return; } void ButtonPressEvent(XButtonEvent * xev) { char buf[256]; if (xev->type != ButtonPress) return; switch (xev->button) { case Button1: ifno++; break; case Button2: case Button3: if(ifinfo.state == 0) { sprintf(buf, exec_up, ifinfo.id); } else { sprintf(buf, exec_down, ifinfo.id); } exec_cmd(buf); break; } } char *strupper(char *str) { int i; for (i = 0; i < strlen(str); i++) str[i] = toupper(str[i]); return str; } int getifinfo(char *ifname, struct ifinfo_t *info) { struct ifreq ifr; struct sockaddr_in *sa; #ifdef linux static FILE *froute = NULL; static FILE *fwireless = NULL; static FILE *fdev = NULL; #elif defined(__OpenBSD__) struct ifreq ibuf[32]; struct ifconf ifc; struct ifreq *ifrp, *ifend; #endif char parent[16]; char buf[1024]; char *p; char a[16]; int b,c,d; #ifdef linux if(froute == NULL) froute = fopen("/proc/net/route", "r"); if(fwireless == NULL) fwireless = fopen("/proc/net/wireless", "r"); if(fdev == NULL) fdev = fopen("/proc/net/dev", "r"); #endif strcpy(parent, ifname); p=strchr(parent, ':'); if(p) *p=0; strcpy(info->id, ifname); strcpy(ifr.ifr_name, ifname); /* Get status (UP/DOWN) */ if(ioctl(fd, SIOCGIFFLAGS, &ifr) != -1) { sa = (struct sockaddr_in *)&(ifr.ifr_addr); info->state = (ifr.ifr_flags & 1) ? 1 : 0; } else { info->state = 0; } /* Get mac address */ #ifdef linux if(ioctl(fd, SIOCGIFHWADDR, &ifr) != -1) { memcpy(info->hw, ifr.ifr_hwaddr.sa_data, 6); } else { memset(info->hw, 0, 6); } #elif defined(__OpenBSD__) ifc.ifc_len = sizeof(ibuf); ifc.ifc_buf = (caddr_t) ibuf; if (ioctl(fd, SIOCGIFCONF, (char *) &ifc) == -1 || ifc.ifc_len < sizeof(struct ifreq)) { memset(info->hw, 0, 6); } else { /* Search interface configuration list for link layer address. */ ifrp = ibuf; ifend = (struct ifreq *) ((char *) ibuf + ifc.ifc_len); while (ifrp < ifend) { /* Look for interface */ if (strcmp(ifname, ifrp->ifr_name) == 0 && ifrp->ifr_addr.sa_family == AF_LINK && ((struct sockaddr_dl *) &ifrp->ifr_addr)->sdl_type == IFT_ETHER) { memcpy(info->hw, LLADDR((struct sockaddr_dl *) &ifrp->ifr_addr), 6); break; } /* Bump interface config pointer */ r = ifrp->ifr_addr.sa_len + sizeof(ifrp->ifr_name); if (r < sizeof(*ifrp)) r = sizeof(*ifrp); ifrp = (struct ifreq *) ((char *) ifrp + r); } } #endif /* Get IP address */ if(ioctl(fd, SIOCGIFADDR, &ifr) != -1) { sa = (struct sockaddr_in *)&(ifr.ifr_addr); info->ip = sa->sin_addr.s_addr; } else { info->ip = 0; } /* Get netmask */ if(ioctl(fd, SIOCGIFNETMASK, &ifr) != -1) { sa = (struct sockaddr_in *)&(ifr.ifr_addr); info->nm = sa->sin_addr.s_addr; } else { info->nm = 0; } /* Get default gateway if on this interface */ info->gw = 0; #ifdef linux if(froute != NULL) { fseek(froute, 0, 0); while(fgets(buf, sizeof(buf), froute)) { sscanf(buf, "%s %x %x", a, (unsigned int *) &b, (unsigned int *) &c); if((strcmp(a, info->id) == 0) && (b == 0)) { info->gw = c; } } } #elif defined(__OpenBSD__) { struct rt_msghdr *rtm = NULL; char *buf = NULL, *next, *lim = NULL; size_t needed; int mib[6]; struct sockaddr *sa; struct sockaddr_in *sin; mib[0] = CTL_NET; mib[1] = PF_ROUTE; mib[2] = 0; mib[3] = AF_INET; mib[4] = NET_RT_DUMP; mib[5] = 0; if (sysctl(mib, 6, NULL, &needed, NULL, 0) == -1) { perror("route-sysctl-estimate"); exit(1); } if (needed > 0) { if ((buf = malloc(needed)) == 0) { printf("out of space\n"); exit(1); } if (sysctl(mib, 6, buf, &needed, NULL, 0) == -1) { perror("sysctl of routing table"); exit(1); } lim = buf + needed; } if (buf) { for (next = buf; next < lim; next += rtm->rtm_msglen) { rtm = (struct rt_msghdr *)next; sa = (struct sockaddr *)(rtm + 1); sin = (struct sockaddr_in *)sa; if (sin->sin_addr.s_addr == 0) { sa = (struct sockaddr *)(ROUNDUP(sa->sa_len) + (char *)sa); sin = (struct sockaddr_in *)sa; info->gw = sin->sin_addr.s_addr; break; } } free(buf); } } #endif /* Get wireless link status if wireless */ info->sl = 0; #ifdef linux if(fwireless != NULL) { fseek(fwireless, 0, 0); while(fgets(buf, sizeof(buf), fwireless)) { sscanf(buf, "%s %d %d ", a, &b, &c); if(strchr(a, ':')) *(strchr(a, ':')) = 0; if(strcmp(a, parent) == 0) { info->sl = c; } } } #ifdef ENABLE_NWN_SUPPORT if (info->sl == 0) { info->sl = nwn_get_link(parent); } #endif #elif defined(__OpenBSD__) { struct wi_req wreq; struct ifreq ifr; wreq.wi_len = WI_MAX_DATALEN; wreq.wi_type = WI_RID_COMMS_QUALITY; strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name)); ifr.ifr_data = (caddr_t)&wreq; if (ioctl(fd, SIOCGWAVELAN, &ifr) != -1) info->sl = letoh16(wreq.wi_val[0]); } #endif /* Get Total tx/rx bytes */ #ifdef linux if(fdev != NULL) { fseek(fdev, 0, 0); while(fgets(buf, sizeof(buf), fdev)) { sscanf(buf, "%s %d %d %d %d %d %d %d %d %d", a, &b, &d,&d,&d,&d,&d,&d,&d, &c); if(strchr(a, ':')) *(strchr(a, ':')) = 0; if(strcmp(a, parent) == 0) { info->bytes = b + c; } } } #endif return(0); } void addifname(char *name) { int i; if(strcmp(name, "lo") == 0) return; if(strncmp(name, "vlan", 4) == 0) return; for(i=0; i<ifaces; i++) { if(strcmp(ifname[i], name) == 0) return; } strcpy(ifname[ifaces], name); ifaces++; return; } /* * get list of interfaces. First read /proc/net/dev, then do a SIOCGIFCONF */ void getifnames(void) { char pifname[MAXIFS][16]; int pifaces; int i,j; int isnew; #ifdef linux FILE *f; char buf[128]; char *p1, *p2; int ifcount; #endif /* * Copy list of interface names and clean the old list */ for(i=0; i<ifaces; i++) strncpy(pifname[i], ifname[i], sizeof(pifname[i])); pifaces = ifaces; ifaces = 0; #ifdef linux f = fopen("/proc/net/dev", "r"); if(f == NULL) { fprintf(stderr, "Can't open /proc/net/dev\n"); exit(1); } while(fgets(buf, sizeof(buf), f)) { p1=buf; while(*p1 == ' ') p1++; p2=p1; while(*p2 && (*p2 != ':')) p2++; if(*p2 == ':') { *p2=0; addifname(p1); } } fclose(f); ifc.ifc_len = sizeof(struct ifreq) * 10; if(ioctl(fd, SIOCGIFCONF, &ifc) == -1) { fprintf(stderr, "SIOCGIFCONF : Can't get list of interfaces : %s\n", strerror(errno)); exit(1); } ifcount = ifc.ifc_len / sizeof(struct ifreq); for(i=0; i<ifcount; i++) { addifname(ifc.ifc_req[i].ifr_name); } #endif #ifdef __OpenBSD__ struct ifreq ibuf[32]; struct ifconf ifc; struct ifreq *ifrp, *ifend; int r; ifc.ifc_len = sizeof(ibuf); ifc.ifc_buf = (caddr_t) ibuf; if (ioctl(fd, SIOCGIFCONF, (char *) &ifc) == -1 || ifc.ifc_len < sizeof(struct ifreq)) { fprintf(stderr, "SIOCGIFCONF : Can't get list of interfaces : %s\n", strerror(errno)); exit(1); } /* Search interface configuration list for link layer address. */ ifrp = ibuf; ifend = (struct ifreq *) ((char *) ibuf + ifc.ifc_len); while (ifrp < ifend) { if (ifrp->ifr_addr.sa_family == AF_LINK && ((struct sockaddr_dl *) &ifrp->ifr_addr)->sdl_type == IFT_ETHER) { addifname(ifrp->ifr_name); } /* Bump interface config pointer */ r = ifrp->ifr_addr.sa_len + sizeof(ifrp->ifr_name); if (r < sizeof(*ifrp)) r = sizeof(*ifrp); ifrp = (struct ifreq *) ((char *) ifrp + r); } #endif /* * Check if the new list contains interfaces that were not in the old list. If a new * interface is found, make it the current one to display. (-i will override) */ for(i=0; i<ifaces; i++) { isnew = 1; for(j=0; j<pifaces; j++) if(strcmp(ifname[i], pifname[j]) == 0) isnew = 0; if(isnew) ifno = i; } for(i=0; i<ifaces; i++) { if(strcasecmp(ifname[i], startif) == 0) { ifno = i; startif[0] = 0; } } }