7545c92c63
Ran `codespell -w .`.
1151 lines
30 KiB
C
1151 lines
30 KiB
C
/* WMiFS - a complete network monitoring dock.app
|
|
Copyright (C) 1997, 1998 Martijn Pieterse <pieterse@xs4all.nl>
|
|
Copyright (C) 1997, 1998 Antoine Nulle <warp@xs4all.nl>
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
Best viewed with vim5, using ts=4
|
|
|
|
This code was mainly put together by looking at the
|
|
following programs:
|
|
|
|
asclock
|
|
A neat piece of equip, used to display the date
|
|
and time on the screen.
|
|
Comes with every AfterStep installation.
|
|
|
|
Source used:
|
|
How do I create a not so solid window?
|
|
How do I open a window?
|
|
How do I use pixmaps?
|
|
|
|
pppstats
|
|
A program that prints the amount of data that
|
|
is transferred over a ppp-line.
|
|
|
|
Source used:
|
|
How do I read the ppp device?
|
|
------------------------------------------------------------
|
|
|
|
Author: Martijn Pieterse (pieterse@xs4all.nl)
|
|
|
|
This program was hacked together between the 7th of March
|
|
and the 14th of March 1998.
|
|
|
|
This program might be Y2K resistant. We shall see. :)
|
|
|
|
This program is distributed under the GPL license.
|
|
(as were asclock and pppstats)
|
|
|
|
Known Features: (or in non M$ talk, BUGS)
|
|
* only ppp0 will be read
|
|
use wmifs if you want to read more than one ppp connection
|
|
not sure about this.
|
|
* the leds won't be reliable with
|
|
more than 1 ppp connection
|
|
* there is an iconwin, and win variable.
|
|
I have no clue why only win shouldn't
|
|
be enough. Will check it out later.
|
|
* The afterstep what seems the shift the
|
|
pixmap a bit. Don't know how and why.
|
|
It works in the WindowManager.
|
|
|
|
Things to do:
|
|
Split up main()
|
|
|
|
----
|
|
Thanks
|
|
----
|
|
|
|
Most of the ideas, and jumpstarting it:
|
|
|
|
#linuxnl, without this irc-channel wmppp would've never seen the light!
|
|
|
|
CCC (Constructive Code Criticism):
|
|
|
|
Marcelo E. Magallon
|
|
Thanks a LOT! It takes a while to get me convinced... :)
|
|
|
|
|
|
Minor bugs and ideas:
|
|
|
|
Marc De Scheemaecker / David Mihm / Chris Soghoian /
|
|
Alessandro Usseglio Viretta
|
|
|
|
and ofcourse numerous ppl who send us bug reports.
|
|
(numerous? hmm.. todo: rephrase this :) )
|
|
|
|
|
|
----
|
|
Changes:
|
|
---
|
|
02/29/2004 (Tom Marshall, tommy@home.tig-grr.com)
|
|
* Patch to add a special interface name "auto" for the -i
|
|
option. "wmifs -i auto" will automatically select the
|
|
first up interface.
|
|
01/08/2004 (Peter Samuelson, peter@samba-tng.org)
|
|
* Patch to make make sampling and scrolling intervals
|
|
customizable, adds new options -I and -s.
|
|
01/15/2002 (Matyas Koszik, koszik@debijan.lonyay.edu.hu)
|
|
* Patch that fixes segfaults on long interface names.
|
|
08/31/2001 (Davi Leal, davileal@terra.es)
|
|
* Patch that cuts long interface names, so they look
|
|
good in wmifs. For example, "dummy0" gets displayed
|
|
as "dumm0", "vmnet10" as "vmn10", etc.
|
|
06/16/2001 (Jorge García, Jorge.Garcia@uv.es)
|
|
* Added the LockMode, so wmifs doesn't swap to another
|
|
interface if the one requested with "-i" isn't up.
|
|
05/06/2001 (Jordi Mallach, jordi@sindominio.net)
|
|
* Integrated many patches, fixing issues with suspended
|
|
wmifs.
|
|
07/21/1999 (Stephen Pitts, smpitts@midsouth.rr.com)
|
|
* Added new constant: BUFFER_SIZE to determine the size
|
|
of the buffer used in fgets() operations. Right now,
|
|
its at 512 bytes. Fixed crashing on my system when
|
|
one line of /proc/net/dev was longer than 128 bytes
|
|
04/05/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Changed the "middle of the waveform" line color
|
|
* Moved the RedrawWindow out of the main loop.
|
|
Lightens the system load
|
|
02/05/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Torn wmppp and wmifs apart.
|
|
This is gonna be wmifs
|
|
* Added parse_rcfile
|
|
* Added waitpid, to get rid of the zombies, spotteb by Alessandro Usseglio Viretta
|
|
30/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Used the DrawStats routine from wmifs in wmppp
|
|
* I decided to add a list in this source file
|
|
with name of ppl who helped me build this code better.
|
|
* I finally removed the /proc/net/route dependency
|
|
All of the connections are taken from /proc/net/dev.
|
|
/proc/net/route is still used for checking if it is on-line.
|
|
27/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* WMIFS: stats scrolled, while red led burning
|
|
* WMPPP: changed positions of line speed
|
|
25/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Changed the checknetdevs routine, a lot!
|
|
23/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added line speed detection. via separate exec. (this has to be suid root!)
|
|
Speed has to be no more than 99999
|
|
* Added speed and forcespeed in ~/.wmppprc and /etc/wmppprc
|
|
* wmifs: added on-line detection scheme, update the bitmap coordinates
|
|
* wmppp: the x-button now allways disconnects.
|
|
22/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added /etc/wmppprc support, including "forced" mode.
|
|
* Added /etc/wmifsrc support, including "forced" mode.
|
|
21/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Moved the stats one pixel down.
|
|
* Added status led in wmifs.
|
|
* Changed RX/TX leds of wmifs to resemble wmppp
|
|
* Added the "dot" between eth.0 ppp.0 etc.
|
|
* Changed to wmifs stats to match wmppp stats (only pppX changed)
|
|
* Made sure that when specified -t 1, it stayed that way, even
|
|
when longer than 100 minutes online
|
|
* With -t 1, jump from 59:59 to 01:00 instead of 99:59 to 01:40
|
|
16/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added "all" devices in wmifs
|
|
* Added "lo" support only if aked via -i
|
|
* Added on-line time detection (using /var/run/ppp0.pid)
|
|
* Added time-out for the orange led. (one minute)
|
|
15/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Another wmppp-master.xpm.
|
|
Line speed detection being the main problem here.. :(
|
|
* Moved START_COMMAND / STOP_COMMAND to ~/.wmppprc
|
|
Return 0, everything went ok.
|
|
Return 10, the command did not work.
|
|
Please note, these functions are ran in the background.
|
|
* Removed the ability to configure
|
|
* When "v" is clicked, an orange led will appear.
|
|
if the connect script fails (return value == 10)
|
|
it will become dark again. Else the on-line detection will
|
|
pick it up, and "green" it.
|
|
14/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added "-timer"
|
|
* Added "-display" support
|
|
* Changed pixmap to a no-name pixmap.
|
|
+ Changed LED positions
|
|
+ Changed Timer position
|
|
+ Changed Stats Size
|
|
05/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added ~/.wmifsrc support.
|
|
* Put some code in DrawStats
|
|
* Check devices when pressing "device change"
|
|
03/04/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added code for wmifs
|
|
28/03/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* forgot what i did.. :)
|
|
27/03/1998 (Martijn Pieterse, pieterse@xs4all.nl)
|
|
* Added on-line detection
|
|
Scan through /proc/net/route and check everye line
|
|
for "ppp".
|
|
* A bit of code clean-up.
|
|
*/
|
|
|
|
#define _DEFAULT_SOURCE
|
|
#include <X11/X.h> /* for ButtonPress, ButtonRelease, etc */
|
|
#include <X11/Xlib.h> /* for XEvent, XButtonEvent, etc */
|
|
#include <X11/xpm.h> /* for XpmColorSymbol, Pixel, etc */
|
|
#include <ctype.h> /* for toupper */
|
|
#include <linux/ppp_defs.h> /* for ppp_stats, pppstat */
|
|
#include <net/if_ppp.h> /* for ifpppstatsreq, ifr__name, etc */
|
|
#include <stddef.h> /* for size_t */
|
|
#include <stdio.h> /* for fprintf, NULL, stderr, etc */
|
|
#include <stdlib.h> /* for exit, atof, atoi, getenv */
|
|
#include <string.h> /* for strcmp, strcpy, strlen, etc */
|
|
#include <sys/ioctl.h> /* for ioctl */
|
|
#include <sys/socket.h> /* for socket, AF_INET */
|
|
#include <sys/time.h> /* for timeval, gettimeofday */
|
|
#include <sys/wait.h> /* for waitpid, WNOHANG */
|
|
#include <time.h> /* for timespec, nanosleep */
|
|
#include <libdockapp/misc.h> /* for execCommand */
|
|
#include <libdockapp/wmgeneral.h> /* for copyXPMArea, display, etc */
|
|
#include "wmifs-mask.xbm" /* for wmifs_mask_bits, etc */
|
|
#include "wmifs-master.xpm" /* for wmifs_master_xpm */
|
|
|
|
/* How often to check for new network interface, in ms */
|
|
#define CHECK_INTERFACE_INTERVAL 5000
|
|
|
|
/* How often to query the interfaces, in ms */
|
|
#define DEFAULT_SAMPLE_INTERVAL 50
|
|
|
|
/***********/
|
|
/* Defines */
|
|
/***********/
|
|
|
|
#ifndef ifr__name
|
|
#define ifr__name ifr_name
|
|
#endif
|
|
|
|
#ifndef stats_ptr
|
|
#define stats_ptr stats.p.FIXME
|
|
#endif
|
|
|
|
/* Fill in and uncomment the hardcoded actions */
|
|
/* #define LEFT_ACTION (NULL) */
|
|
/* #define MIDDLE_ACTION (NULL) */
|
|
/* #define RIGHT_ACTION (NULL) */
|
|
|
|
/* Defines voor alle coordinate */
|
|
|
|
#define LED_NET_RX (4)
|
|
#define LED_NET_TX (5)
|
|
#define LED_NET_POWER (6)
|
|
|
|
/* the size of the buffer read from /proc/net/ */
|
|
#define BUFFER_SIZE 512
|
|
|
|
/**********************/
|
|
/* External Variables */
|
|
/**********************/
|
|
|
|
extern char **environ;
|
|
|
|
/********************/
|
|
/* Global Variables */
|
|
/********************/
|
|
|
|
char *active_interface = NULL;
|
|
int TimerDivisor = 60;
|
|
int WaveForm = 0;
|
|
int LockMode = 0;
|
|
int SampleInt = DEFAULT_SAMPLE_INTERVAL;
|
|
int ScrollSpeed = CHECK_INTERFACE_INTERVAL;
|
|
XpmIcon wmgen;
|
|
char color[256];
|
|
|
|
/*****************/
|
|
/* PPP variables */
|
|
/*****************/
|
|
|
|
#define PPP_UNIT 0
|
|
int ppp_h = -1;
|
|
|
|
#define PPP_STATS_HIS 54
|
|
|
|
/***********************/
|
|
/* Function Prototypes */
|
|
/***********************/
|
|
|
|
void usage(void);
|
|
void printversion(void);
|
|
void DrawTime(int, int);
|
|
void DrawStats(int *, int, int, int, int);
|
|
|
|
void SetOnLED(int);
|
|
void SetErrLED(int);
|
|
void SetWaitLED(int);
|
|
void SetOffLED(int);
|
|
|
|
void ButtonUp(int);
|
|
void ButtonDown(int);
|
|
|
|
void wmifs_routine(int, char **);
|
|
|
|
void get_ppp_stats(struct ppp_stats *cur);
|
|
|
|
|
|
/********/
|
|
/* Main */
|
|
/********/
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
|
|
int i;
|
|
|
|
color[0] = 0;
|
|
|
|
/* Parse Command Line */
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
char *arg = argv[i];
|
|
|
|
if (*arg == '-') {
|
|
switch (arg[1]) {
|
|
case 'c' :
|
|
if (argc > i+1) {
|
|
strncpy(color, argv[i+1], sizeof(color));
|
|
i++;
|
|
}
|
|
break;
|
|
case 'd':
|
|
if (strcmp(arg+1, "display")) {
|
|
usage();
|
|
exit(1);
|
|
}
|
|
break;
|
|
case 'g':
|
|
if (strcmp(arg+1, "geometry")) {
|
|
usage();
|
|
exit(1);
|
|
}
|
|
break;
|
|
case 'i':
|
|
active_interface = argv[i+1];
|
|
i++;
|
|
break;
|
|
case 'I':
|
|
SampleInt = atof(argv[i+1]) * 1000;
|
|
i++;
|
|
break;
|
|
case 'l':
|
|
LockMode = 1;
|
|
break;
|
|
case 's':
|
|
ScrollSpeed = atof(argv[i+1]) * 1000;
|
|
i++;
|
|
break;
|
|
case 'v':
|
|
printversion();
|
|
exit(0);
|
|
break;
|
|
case 'w':
|
|
WaveForm = 1;
|
|
break;
|
|
default:
|
|
if (strcmp(argv[i-1], "-geometry")) {
|
|
usage();
|
|
exit(0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
wmifs_routine(argc, argv);
|
|
return 0;
|
|
}
|
|
|
|
Pixel scale_pixel(Pixel pixel, float scale)
|
|
{
|
|
int red, green, blue;
|
|
|
|
red = pixel / ( 1 << 16 );
|
|
green = pixel % (1 << 16) / (1 << 8);
|
|
blue = pixel % (1 << 8);
|
|
|
|
red *= scale;
|
|
green *= scale;
|
|
blue *= scale;
|
|
|
|
return red * (1 << 16) + green * (1 << 8) + blue;
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* wmifs_routine *|
|
|
\*******************************************************************************/
|
|
|
|
#define MAX_STAT_DEVICES 16
|
|
|
|
typedef struct {
|
|
|
|
char name[IFNAMSIZ];
|
|
int his[55][2];
|
|
long istatlast;
|
|
long ostatlast;
|
|
|
|
} stat_dev;
|
|
|
|
stat_dev stat_devices[MAX_STAT_DEVICES];
|
|
|
|
char *left_action = NULL;
|
|
char *right_action = NULL;
|
|
char *middle_action = NULL;
|
|
|
|
int checknetdevs(void);
|
|
int get_statistics(char *, long *, long *, long *, long *);
|
|
int stillonline(char *);
|
|
void DrawActiveIFS(char *);
|
|
|
|
void wmifs_routine(int argc, char **argv)
|
|
{
|
|
|
|
rckeys wmifs_keys[] = {
|
|
{ "left", &left_action },
|
|
{ "middle", &middle_action },
|
|
{ "right", &right_action },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
|
|
int i, j;
|
|
XEvent Event;
|
|
int but_stat = -1;
|
|
|
|
int stat_online;
|
|
int stat_current;
|
|
int first_time = 1;
|
|
|
|
unsigned int curtime;
|
|
unsigned int nexttime;
|
|
struct timeval tv, tv2;
|
|
|
|
long ipacket, opacket, istat, ostat;
|
|
|
|
char temp[BUFFER_SIZE];
|
|
char *p;
|
|
|
|
for (i = 0; i < MAX_STAT_DEVICES; i++) {
|
|
stat_devices[i].name[0] = 0;
|
|
for (j = 0; j < 48; j++) {
|
|
stat_devices[i].his[j][0] = 0;
|
|
stat_devices[i].his[j][1] = 0;
|
|
}
|
|
}
|
|
|
|
stat_online = checknetdevs();
|
|
|
|
stat_current = 0;
|
|
if (active_interface) {
|
|
int isauto = !strcmp(active_interface, "auto");
|
|
for (i = 0; i < stat_online; i++) {
|
|
if ((isauto && stillonline(stat_devices[i].name)) ||
|
|
!strcmp(stat_devices[i].name, active_interface)) {
|
|
stat_current = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef LEFT_ACTION
|
|
left_action = strdup(LEFT_ACTION);
|
|
#endif
|
|
#ifdef MIDDLE_ACTION
|
|
middle_action = strdup(MIDDLE_ACTION);
|
|
#endif
|
|
#ifdef RIGHT_ACTION
|
|
right_action = strdup(RIGHT_ACTION);
|
|
#endif
|
|
|
|
/* Scan throught the .rc files */
|
|
parse_rcfile(CONF"/wmifsrc", wmifs_keys);
|
|
|
|
p = getenv("HOME");
|
|
if (p == NULL || *p == 0) {
|
|
fprintf(stderr, "Unknown $HOME directory, please check your environment\n");
|
|
return;
|
|
}
|
|
strncpy(temp, p, BUFFER_SIZE - 10);
|
|
strcat(temp, "/.wmifsrc");
|
|
parse_rcfile(temp, wmifs_keys);
|
|
|
|
parse_rcfile(CONF"/wmifsrc.fixed", wmifs_keys);
|
|
|
|
/* set user-defined colors */
|
|
if (color[0] != 0) {
|
|
Window Root;
|
|
XColor col;
|
|
XWindowAttributes attributes;
|
|
int screen;
|
|
Pixel pixel;
|
|
#define NUMSYMBOLS 4
|
|
XpmColorSymbol user_color[NUMSYMBOLS] = {
|
|
{NULL, "#2081B2CAAEBA", 0}, /* + */
|
|
{NULL, "#28A23CF338E3", 0}, /* O */
|
|
{NULL, "#000049244103", 0}, /* @ */
|
|
{NULL, "#18618A288617", 0}, /* # */
|
|
};
|
|
|
|
|
|
/* code based on GetColor() from wmgeneral.c */
|
|
/* we need a temporary display to parse the color */
|
|
display = XOpenDisplay(NULL);
|
|
screen = DefaultScreen(display);
|
|
Root = RootWindow(display, screen);
|
|
XGetWindowAttributes(display, Root, &attributes);
|
|
|
|
col.pixel = 0;
|
|
if (!XParseColor(display, attributes.colormap, color, &col)) {
|
|
fprintf(stderr, "wmtime: can't parse %s.\n", color);
|
|
goto draw_window;
|
|
} else if (!XAllocColor(display, attributes.colormap, &col)) {
|
|
fprintf(stderr, "wmtime: can't allocate %s.\n", color);
|
|
goto draw_window;
|
|
}
|
|
|
|
pixel = col.pixel;
|
|
|
|
/* replace colors from wmtime-master.xpm */
|
|
user_color[0].pixel = pixel;
|
|
user_color[1].pixel = scale_pixel(pixel, .3);
|
|
user_color[2].pixel = scale_pixel(pixel, .4);
|
|
user_color[3].pixel = scale_pixel(pixel, .8);
|
|
|
|
wmgen.attributes.valuemask |= XpmColorSymbols;
|
|
wmgen.attributes.numsymbols = NUMSYMBOLS;
|
|
wmgen.attributes.colorsymbols = user_color;
|
|
|
|
XCloseDisplay(display);
|
|
}
|
|
|
|
draw_window:
|
|
openXwindow(argc, argv, wmifs_master_xpm, (char*)wmifs_mask_bits, wmifs_mask_width, wmifs_mask_height);
|
|
|
|
/* > Button */
|
|
AddMouseRegion(0, 5, 5, 35, 15);
|
|
AddMouseRegion(1, 5, 20, 58, 58);
|
|
|
|
gettimeofday(&tv2, NULL);
|
|
nexttime = ScrollSpeed;
|
|
|
|
DrawActiveIFS(stat_devices[stat_current].name);
|
|
|
|
while (1) {
|
|
struct timespec ts;
|
|
|
|
gettimeofday(&tv, NULL);
|
|
curtime = (tv.tv_sec - tv2.tv_sec) * 1000
|
|
+ (tv.tv_usec - tv2.tv_usec) / 1000;
|
|
|
|
waitpid(0, NULL, WNOHANG);
|
|
|
|
for (i = 0; i < stat_online; i++) {
|
|
get_statistics(stat_devices[i].name, &ipacket, &opacket, &istat, &ostat);
|
|
|
|
if (first_time) {
|
|
first_time = 0;
|
|
} else {
|
|
stat_devices[i].his[53][0] += istat - stat_devices[i].istatlast;
|
|
stat_devices[i].his[53][1] += ostat - stat_devices[i].ostatlast;
|
|
}
|
|
|
|
if (i == stat_current) {
|
|
if (!stillonline(stat_devices[i].name))
|
|
SetErrLED(LED_NET_POWER);
|
|
else
|
|
SetOnLED(LED_NET_POWER);
|
|
|
|
if (stat_devices[i].istatlast == istat)
|
|
SetOffLED(LED_NET_RX);
|
|
else
|
|
SetOnLED(LED_NET_RX);
|
|
|
|
if (stat_devices[i].ostatlast == ostat)
|
|
SetOffLED(LED_NET_TX);
|
|
else
|
|
SetOnLED(LED_NET_TX);
|
|
}
|
|
|
|
stat_devices[i].istatlast = istat;
|
|
stat_devices[i].ostatlast = ostat;
|
|
}
|
|
RedrawWindow();
|
|
|
|
if (curtime >= nexttime) {
|
|
nexttime = curtime + ScrollSpeed;
|
|
|
|
DrawStats(&stat_devices[stat_current].his[0][0], 54, 40, 5, 58);
|
|
for (i = 0; i < stat_online; i++) {
|
|
if (stillonline(stat_devices[i].name)) {
|
|
for (j = 1; j < 54; j++) {
|
|
stat_devices[i].his[j-1][0] = stat_devices[i].his[j][0];
|
|
stat_devices[i].his[j-1][1] = stat_devices[i].his[j][1];
|
|
}
|
|
stat_devices[i].his[53][0] = 0;
|
|
stat_devices[i].his[53][1] = 0;
|
|
}
|
|
}
|
|
RedrawWindow();
|
|
}
|
|
|
|
while (XPending(display)) {
|
|
XNextEvent(display, &Event);
|
|
switch (Event.type) {
|
|
case Expose:
|
|
RedrawWindow();
|
|
break;
|
|
case DestroyNotify:
|
|
XCloseDisplay(display);
|
|
exit(0);
|
|
break;
|
|
case ButtonPress:
|
|
but_stat = CheckMouseRegion(Event.xbutton.x, Event.xbutton.y);
|
|
break;
|
|
case ButtonRelease:
|
|
i = CheckMouseRegion(Event.xbutton.x, Event.xbutton.y);
|
|
|
|
if (but_stat == i && but_stat >= 0) {
|
|
switch (but_stat) {
|
|
case 0:
|
|
/* re-read the table */
|
|
strcpy(temp, stat_devices[stat_current].name);
|
|
stat_online = checknetdevs();
|
|
stat_current = 0;
|
|
for (i = 0; i < stat_online; i++) {
|
|
if (!strcmp(temp, stat_devices[i].name))
|
|
stat_current = i;
|
|
}
|
|
|
|
stat_current++;
|
|
if (stat_current == stat_online)
|
|
stat_current = 0;
|
|
|
|
DrawActiveIFS(stat_devices[stat_current].name);
|
|
|
|
DrawStats(&stat_devices[stat_current].his[0][0], 54, 40, 5, 58);
|
|
break;
|
|
case 1:
|
|
switch (Event.xbutton.button) {
|
|
case 1:
|
|
if (left_action)
|
|
execCommand(left_action);
|
|
break;
|
|
case 2:
|
|
if (middle_action)
|
|
execCommand(middle_action);
|
|
break;
|
|
case 3:
|
|
if (right_action)
|
|
execCommand(right_action);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
}
|
|
}
|
|
but_stat = -1;
|
|
RedrawWindow();
|
|
break;
|
|
}
|
|
}
|
|
ts.tv_sec = 0;
|
|
ts.tv_nsec = SampleInt * 1000000;
|
|
nanosleep(&ts, NULL);
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* void DrawActiveIFS(char *) *|
|
|
\*******************************************************************************/
|
|
|
|
void DrawActiveIFS(char *real_name)
|
|
{
|
|
|
|
/* Cijfers op: 0,65
|
|
Letters op: 0,75
|
|
Alles 9 hoog, 6 breedt
|
|
|
|
Destinatie: 5,5
|
|
*/
|
|
|
|
size_t i;
|
|
int k;
|
|
size_t len;
|
|
char name[256];
|
|
|
|
|
|
copyXPMArea(5, 84, 30, 10, 5, 5);
|
|
|
|
|
|
strcpy(name, real_name);
|
|
len = strlen(name);
|
|
if (len > 5) {
|
|
for (i = len-5; i < len && !(name[i] >= '0' && name[i] <= '9'); i++)
|
|
;
|
|
for (; i <= len; i++) /* '=' to get the '\0' character moved too \*/
|
|
name[i-(len-5)] = name[i];
|
|
}
|
|
|
|
k = 5;
|
|
for (i = 0; name[i]; i++) {
|
|
int c;
|
|
|
|
if (i == strlen(name)-1 && strlen(name) <= 4 && name[strlen(name)-1] >= '0' &&
|
|
name[strlen(name)-1] <= '9') {
|
|
copyXPMArea(61, 64, 4, 9, k, 5);
|
|
k += 4;
|
|
}
|
|
c = toupper(name[i]);
|
|
if (c >= 'A' && c <= 'Z') {
|
|
c -= 'A';
|
|
copyXPMArea(c * 6, 74, 6, 9, k, 5);
|
|
k += 6;
|
|
} else {
|
|
c -= '0';
|
|
copyXPMArea(c * 6, 64, 6, 9, k, 5);
|
|
k += 6;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* get_statistics *|
|
|
\*******************************************************************************/
|
|
|
|
int get_statistics(char *devname, long *ip, long *op, long *is, long *os)
|
|
{
|
|
|
|
FILE *fp;
|
|
char temp[BUFFER_SIZE];
|
|
char *p, *saveptr;
|
|
char *tokens = " |:\n";
|
|
int input, output;
|
|
int i;
|
|
int found;
|
|
struct ppp_stats ppp_cur;
|
|
|
|
|
|
if (!strncmp(devname, "ppp", 3)) {
|
|
static int ppp_opened;
|
|
|
|
if (!ppp_opened) {
|
|
/* Open the ppp device. */
|
|
memset(&ppp_cur, 0, sizeof(ppp_cur));
|
|
ppp_h = socket(AF_INET, SOCK_DGRAM, 0);
|
|
if (ppp_h < 0)
|
|
return -1;
|
|
get_ppp_stats(&ppp_cur);
|
|
ppp_opened = 1;
|
|
}
|
|
|
|
get_ppp_stats(&ppp_cur);
|
|
|
|
*op = ppp_cur.p.ppp_opackets;
|
|
*ip = ppp_cur.p.ppp_ipackets;
|
|
|
|
*is = ppp_cur.p.ppp_ibytes;
|
|
*os = ppp_cur.p.ppp_obytes;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Read from /proc/net/dev the stats! */
|
|
fp = fopen("/proc/net/dev", "r");
|
|
if (!fgets(temp, BUFFER_SIZE, fp)) {
|
|
fclose(fp);
|
|
return -1;
|
|
}
|
|
if (!fgets(temp, BUFFER_SIZE, fp)) {
|
|
fclose(fp);
|
|
return -1;
|
|
}
|
|
|
|
input = -1;
|
|
output = -1;
|
|
i = 0;
|
|
found = -1;
|
|
|
|
p = strtok_r(temp, tokens, &saveptr);
|
|
do {
|
|
if (!(strcmp(p, "packets"))) {
|
|
if (input == -1)
|
|
input = i;
|
|
else
|
|
output = i;
|
|
}
|
|
i++;
|
|
p = strtok_r(NULL, tokens, &saveptr);
|
|
} while (input == -1 || output == -1);
|
|
|
|
while (fgets(temp, BUFFER_SIZE, fp)) {
|
|
if (strstr(temp, devname)) {
|
|
found = 0;
|
|
p = strtok_r(temp, tokens, &saveptr);
|
|
i = 0;
|
|
do {
|
|
if (i == input) {
|
|
*ip = *is = atoi(p);
|
|
input = -1;
|
|
}
|
|
if (i == output) {
|
|
*op = *os = atoi(p);
|
|
output = -1;
|
|
}
|
|
i++;
|
|
p = strtok_r(NULL, tokens, &saveptr);
|
|
} while (input != -1 || output != -1);
|
|
}
|
|
}
|
|
fclose(fp);
|
|
|
|
return found;
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* stillonline *|
|
|
\*******************************************************************************/
|
|
|
|
int stillonline(char *ifs)
|
|
{
|
|
|
|
FILE *fp;
|
|
int i;
|
|
|
|
i = 0;
|
|
fp = fopen("/proc/net/route", "r");
|
|
if (fp) {
|
|
char temp[BUFFER_SIZE];
|
|
|
|
while (fgets(temp, BUFFER_SIZE, fp)) {
|
|
if (strstr(temp, ifs)) {
|
|
i = 1; /* Line is alive */
|
|
break;
|
|
}
|
|
}
|
|
fclose(fp);
|
|
}
|
|
return i;
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* checknetdevs *|
|
|
\*******************************************************************************/
|
|
|
|
int checknetdevs(void)
|
|
{
|
|
|
|
FILE *fd;
|
|
int i = 0, j;
|
|
int k;
|
|
int devsfound = 0;
|
|
char foundbuffer[MAX_STAT_DEVICES][IFNAMSIZ];
|
|
|
|
for (i = 0; i < MAX_STAT_DEVICES; i++)
|
|
foundbuffer[i][0] = 0;
|
|
|
|
/* foundbuffer vullen met info uit /proc/net/dev */
|
|
|
|
fd = fopen("/proc/net/dev", "r");
|
|
if (fd) {
|
|
char temp[BUFFER_SIZE];
|
|
|
|
/* Skip the first 2 lines */
|
|
if (!fgets(temp, BUFFER_SIZE, fd)) {
|
|
fclose(fd);
|
|
return -1;
|
|
}
|
|
if (!fgets(temp, BUFFER_SIZE, fd)) {
|
|
fclose(fd);
|
|
return -1;
|
|
}
|
|
while (fgets(temp, BUFFER_SIZE, fd)) {
|
|
char *p, *saveptr;
|
|
char *tokens = " :\t\n";
|
|
|
|
p = strtok_r(temp, tokens, &saveptr);
|
|
if (p == NULL) {
|
|
printf("Barfed on: %s", temp);
|
|
break;
|
|
}
|
|
/* Skip dummy code */
|
|
|
|
if (!strncmp(p, "dummy", 5))
|
|
continue;
|
|
|
|
/* If p == "lo", and active_interface (as given on the cmd line) != "lo",
|
|
skip it! */
|
|
|
|
if (strcmp(p, "lo") || (active_interface && !strcmp(active_interface, "lo"))) {
|
|
strncpy(foundbuffer[devsfound], p, IFNAMSIZ);
|
|
devsfound++;
|
|
}
|
|
if (devsfound >= MAX_STAT_DEVICES)
|
|
break;
|
|
}
|
|
fclose(fd);
|
|
}
|
|
|
|
/* Nu foundbuffer naar stat_devices[].name kopieeren */
|
|
|
|
for (i = 0; i < MAX_STAT_DEVICES; i++) {
|
|
/* Loop stat_devices na, als die naam niet voorkomt in foundbuffer, kill! */
|
|
|
|
if (stat_devices[i].name[0]) {
|
|
k = 0;
|
|
for (j = 0; j < MAX_STAT_DEVICES; j++) {
|
|
if (!strcmp(stat_devices[i].name, foundbuffer[j])) {
|
|
k = 1;
|
|
foundbuffer[j][0] = 0;
|
|
}
|
|
}
|
|
if (!k)
|
|
stat_devices[i].name[0] = 0;
|
|
}
|
|
}
|
|
|
|
for (i = 0, j = 0; j < MAX_STAT_DEVICES; i++, j++) {
|
|
|
|
while (!stat_devices[j].name[0] && j < MAX_STAT_DEVICES)
|
|
j++;
|
|
|
|
if (j < MAX_STAT_DEVICES && i != j)
|
|
stat_devices[i] = stat_devices[j];
|
|
}
|
|
i--;
|
|
|
|
for (j = 0; j < MAX_STAT_DEVICES; j++) {
|
|
if (foundbuffer[j][0]) {
|
|
|
|
strncpy(stat_devices[i].name, foundbuffer[j], IFNAMSIZ);
|
|
|
|
for (k = 0; k < 48; k++) {
|
|
stat_devices[i].his[k][0] = 0;
|
|
stat_devices[i].his[k][1] = 0;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
}
|
|
if (LockMode && active_interface != NULL) {
|
|
k = 0;
|
|
for (j = 0; j < i; j++)
|
|
if (!strcmp(stat_devices[j].name, active_interface)) {
|
|
k = 1;
|
|
break;
|
|
}
|
|
if (!k) {
|
|
strncpy(stat_devices[i].name, active_interface, IFNAMSIZ);
|
|
for (k = 0; k < 48; k++) {
|
|
stat_devices[i].his[k][0] = 0;
|
|
stat_devices[i].his[k][1] = 0;
|
|
}
|
|
devsfound++;
|
|
}
|
|
|
|
}
|
|
return devsfound;
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* DrawStats *|
|
|
\*******************************************************************************/
|
|
|
|
void DrawStats(int *his, int num, int size, int x_left, int y_bottom)
|
|
{
|
|
|
|
int pixels_per_byte;
|
|
int j, k;
|
|
int *p;
|
|
int p2, p3;
|
|
|
|
pixels_per_byte = size;
|
|
p = his;
|
|
for (j = 0; j < num; j++) {
|
|
if (p[0] + p[1] > pixels_per_byte)
|
|
pixels_per_byte = p[0] + p[1];
|
|
p += 2;
|
|
}
|
|
|
|
pixels_per_byte /= size;
|
|
p = his;
|
|
|
|
for (k = 0; k < num; k++) {
|
|
int p0, p1;
|
|
|
|
p0 = p[0];
|
|
p1 = p[1];
|
|
|
|
|
|
if (WaveForm) {
|
|
p2 = 0;
|
|
p3 = 1;
|
|
for (j = 0; j < size; j++) {
|
|
if (j < p0 / pixels_per_byte)
|
|
copyXPMArea(100+2, 68, 1, 1, k+x_left, y_bottom-size/2+p2/2);
|
|
else if (j < (p0 + p1) / pixels_per_byte)
|
|
copyXPMArea(100+1, 68, 1, 1, k+x_left, y_bottom-size/2+p2/2);
|
|
else
|
|
copyXPMArea(100+0, 68, 1, 1, k+x_left, y_bottom-size/2+p2/2);
|
|
|
|
p2 = (p2 + p3);
|
|
p3 *= -1;
|
|
p2 *= -1;
|
|
}
|
|
copyXPMArea(100+3, 68, 1, 1, k+x_left, y_bottom-size/2);
|
|
} else {
|
|
for (j = 0; j < size; j++) {
|
|
if (j < p0 / pixels_per_byte)
|
|
copyXPMArea(100+2, 68, 1, 1, k+x_left, y_bottom-j);
|
|
else if (j < (p0 + p1) / pixels_per_byte)
|
|
copyXPMArea(100+1, 68, 1, 1, k+x_left, y_bottom-j);
|
|
else
|
|
copyXPMArea(100+0, 68, 1, 1, k+x_left, y_bottom-j);
|
|
}
|
|
}
|
|
p += 2;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* usage *|
|
|
\*******************************************************************************/
|
|
|
|
void usage(void)
|
|
{
|
|
|
|
fprintf(stderr, "\nwmifs - programming: tijno, (de)bugging & design: warpstah, webhosting: bobby\n\n");
|
|
fprintf(stderr, "usage:\n");
|
|
fprintf(stderr, "\t-c <color>\t\tset color\n");
|
|
fprintf(stderr, "\t-display <display name>\tset display\n");
|
|
fprintf(stderr, "\t-geometry +x+y\t\tset window position\n");
|
|
fprintf(stderr, "\t-h\t\t\tthis help screen\n");
|
|
fprintf(stderr, "\t-i <interface name>\tdefault (as it appears in /proc/net/route)\n");
|
|
fprintf(stderr, "\t-I <interval>\t\tsampling interval, in seconds (default: 0.05)\n");
|
|
fprintf(stderr, "\t-l\t\t\tstarts in lock mode\n");
|
|
fprintf(stderr, "\t-s <interval>\t\tscrolling interval, in seconds (default: 5)\n");
|
|
fprintf(stderr, "\t-v\t\t\tprint the version number\n");
|
|
fprintf(stderr, "\t-w\t\t\twaveform load\n");
|
|
fprintf(stderr, "\n");
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* printversion *|
|
|
\*******************************************************************************/
|
|
|
|
void printversion(void)
|
|
{
|
|
|
|
fprintf(stderr, "%s\n", PACKAGE_STRING);
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* get_ppp_stats *|
|
|
\*******************************************************************************/
|
|
|
|
void get_ppp_stats(struct ppp_stats *cur)
|
|
{
|
|
|
|
struct ifpppstatsreq req;
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
|
|
req.stats_ptr = (void *) &req.stats;
|
|
|
|
sprintf(req.ifr__name, "ppp%d", PPP_UNIT);
|
|
|
|
if (ioctl(ppp_h, SIOCGPPPSTATS, &req) < 0) {
|
|
/* fprintf(stderr, "heyho!\n") */;
|
|
}
|
|
*cur = req.stats;
|
|
}
|
|
|
|
#define LED_SZE_X (4)
|
|
#define LED_SZE_Y (4)
|
|
|
|
#define LED_ON_NET_X (87)
|
|
#define LED_ON_NET_Y (66)
|
|
#define LED_OFF_NET_X (93)
|
|
#define LED_OFF_NET_Y (66)
|
|
#define LED_ERR_NET_X (81)
|
|
#define LED_ERR_NET_Y (66)
|
|
#define LED_ON_SW_NET_X (49)
|
|
#define LED_ON_SW_NET_Y (85)
|
|
#define LED_OFF_SW_NET_X (44)
|
|
#define LED_OFF_SW_NET_Y (85)
|
|
|
|
#define LED_PWR_X (53)
|
|
#define LED_PWR_Y (7)
|
|
#define LED_SND_X (47)
|
|
#define LED_SND_Y (7)
|
|
#define LED_RCV_X (41)
|
|
#define LED_RCV_Y (7)
|
|
|
|
#define LED_SW_X (38)
|
|
#define LED_SW_Y (14)
|
|
|
|
/*******************************************************************************\
|
|
|* SetOnLED *|
|
|
\*******************************************************************************/
|
|
void SetOnLED(int led)
|
|
{
|
|
|
|
switch (led) {
|
|
|
|
case LED_NET_RX:
|
|
copyXPMArea(LED_ON_NET_X, LED_ON_NET_Y, LED_SZE_X, LED_SZE_Y, LED_RCV_X, LED_RCV_Y);
|
|
break;
|
|
case LED_NET_TX:
|
|
copyXPMArea(LED_ON_NET_X, LED_ON_NET_Y, LED_SZE_X, LED_SZE_Y, LED_SND_X, LED_SND_Y);
|
|
break;
|
|
case LED_NET_POWER:
|
|
copyXPMArea(LED_ON_NET_X, LED_ON_NET_Y, LED_SZE_X, LED_SZE_Y, LED_PWR_X, LED_PWR_Y);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* SetOffLED *|
|
|
\*******************************************************************************/
|
|
void SetOffLED(int led)
|
|
{
|
|
|
|
switch (led) {
|
|
|
|
case LED_NET_RX:
|
|
copyXPMArea(LED_OFF_NET_X, LED_OFF_NET_Y, LED_SZE_X, LED_SZE_Y, LED_RCV_X, LED_RCV_Y);
|
|
break;
|
|
case LED_NET_TX:
|
|
copyXPMArea(LED_OFF_NET_X, LED_OFF_NET_Y, LED_SZE_X, LED_SZE_Y, LED_SND_X, LED_SND_Y);
|
|
break;
|
|
case LED_NET_POWER:
|
|
copyXPMArea(LED_OFF_NET_X, LED_OFF_NET_Y, LED_SZE_X, LED_SZE_Y, LED_PWR_X, LED_PWR_Y);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************\
|
|
|* SetErrLED *|
|
|
\*******************************************************************************/
|
|
void SetErrLED(int led)
|
|
{
|
|
|
|
switch (led) {
|
|
case LED_NET_POWER:
|
|
copyXPMArea(LED_ERR_NET_X, LED_ERR_NET_Y, LED_SZE_X, LED_SZE_Y, LED_PWR_X, LED_PWR_Y);
|
|
break;
|
|
}
|
|
}
|