2be3432290
In particular, if HOME is undefined, then a segmentation fault will occur. Also, if HOME is at least 245 characters, then a buffer overflow will occur. We check for these conditions and exit with an error message instead. Patch obtained from Debian package [1]. [1] https://sources.debian.net/src/wmfsm/0.35-1/debian/patches/handle_HOME_errors.patch/
650 lines
14 KiB
C
650 lines
14 KiB
C
/*
|
|
*
|
|
* wmfsm (C) 1999 Stefan Eilemann (Stefan.Eilemann@dlr.de)
|
|
*
|
|
* - Shows file system usage ala mfsm
|
|
*
|
|
*
|
|
*
|
|
*
|
|
* 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, 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 (see the file COPYING); if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA
|
|
*
|
|
*
|
|
*
|
|
* Changes: #include "../ChangeLog"
|
|
*
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
* Includes
|
|
*/
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
#include <X11/X.h>
|
|
#include <X11/xpm.h>
|
|
#include <libdockapp/wmgeneral.h>
|
|
#include "wmfsm_master.xpm"
|
|
#include "wmfsm_mask.xbm"
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#ifdef SVR4
|
|
#include <netdb.h>
|
|
#include <sys/systeminfo.h>
|
|
#endif /* SVR4 */
|
|
|
|
#ifdef HAVE_SYS_VFS_H
|
|
#include <sys/vfs.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_STATFS_H
|
|
#include <sys/statfs.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_MOUNT_H
|
|
#include <sys/mount.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_PARAM_H
|
|
#include <sys/param.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
|
|
#ifdef STATFS_2_ARGUMENTS
|
|
# define STATFS(a,b) statfs(a,b)
|
|
#elif defined STATFS_4_ARGUMENTS
|
|
# define STATFS(a,b) statfs(a,b,sizeof(struct statfs),0)
|
|
#else
|
|
# define STATFS(a,b) statfs(a,b) /* Maybe configure got messed up */
|
|
#endif
|
|
|
|
/*
|
|
#if defined IRIX64
|
|
# include <sys/types.h>
|
|
# define STATFS(a,b) statfs(a,b,sizeof(struct statfs),0)
|
|
#elif defined linux
|
|
# define STATFS(a,b) statfs(a,b)
|
|
#elif defined(SunOS)
|
|
# define STATFS(a,b) statfs(a,b,sizeof(struct statfs),0)
|
|
#elif defined(__OpenBSD__) || defined(__FreeBSD__)
|
|
# define STATFS(a,b) statfs(a,b)
|
|
# include <sys/param.h>
|
|
#endif
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#include <getopt.h>
|
|
/*
|
|
* Delay between refreshes (in microseconds), uses the 10*DELAY_10
|
|
* coz irix has max of 100000L :(.
|
|
*/
|
|
#define DELAY_10 99999L
|
|
#define LENMP 5 /*max 10, number char for mountpoint */
|
|
|
|
/*modes for drawing*/
|
|
#define NORMAL 0
|
|
#define FIRE 1
|
|
|
|
|
|
void ParseCMDLine(int argc, char *argv[]);
|
|
void pressEvent(XButtonEvent * xev);
|
|
void readFileSystems(void);
|
|
void excludeFileSystems(void);
|
|
void print_usage(void);
|
|
|
|
int UpToDate = 0;
|
|
int ForceUpdate = 1;
|
|
char *mp[100];
|
|
int per[9];
|
|
int numberfs;
|
|
int mode = FIRE;
|
|
int blink = 1;
|
|
int blinkper = 95;
|
|
char dummy[4096];
|
|
char *myName;
|
|
int delay = DELAY_10;
|
|
char appearance[256] = "";
|
|
|
|
int xpos[] = { 66, 71, 76, 81, 86, 91, /* A B C D E F */
|
|
66, 71, 76, 81, 86, 91, /* G H I J K L */
|
|
66, 71, 76, 81, 86, 91, /* M N O P Q R */
|
|
66, 71, 76, 81, 86, 91, /* S T U V W X */
|
|
66, 71, 76, 81, 86, 91, /* Y Z / _ - . */
|
|
96, 101, /* 0 1 */
|
|
96, 101, /* 2 3 */
|
|
96, 101, /* 4 5 */
|
|
96, 101, /* 6 7 */
|
|
96, 101
|
|
}; /* 8 9 */
|
|
|
|
int ypos[] = { 4, 4, 4, 4, 4, 4,
|
|
9, 9, 9, 9, 9, 9,
|
|
14, 14, 14, 14, 14, 14,
|
|
19, 19, 19, 19, 19, 19,
|
|
24, 24, 24, 24, 24, 24,
|
|
4, 4,
|
|
9, 9,
|
|
14, 14,
|
|
19, 19,
|
|
24, 24
|
|
};
|
|
|
|
|
|
/*
|
|
* main
|
|
*/
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
|
|
XEvent event;
|
|
int dx, dy;
|
|
char hostname[100];
|
|
int i, j, k;
|
|
int xpm_free = 0;
|
|
int c, on[9] = { 1, 1, 1, 1, 1, 1, 1, 1, 1 };
|
|
struct statfs buffer;
|
|
char **pixmap;
|
|
|
|
/*
|
|
* Parse any command line arguments.
|
|
*/
|
|
myName = strdup(argv[0]);
|
|
ParseCMDLine(argc, argv);
|
|
|
|
if (appearance[0] == 0) {
|
|
pixmap = wmfsm_master_xpm;
|
|
} else {
|
|
if (XpmReadFileToData(appearance, &pixmap) != XpmSuccess) {
|
|
fprintf(stderr, "warning: could not read appearance file; using default.\n");
|
|
pixmap = wmfsm_master_xpm;
|
|
} else {
|
|
xpm_free = 1;
|
|
}
|
|
}
|
|
|
|
openXwindow(argc, argv, pixmap, wmfsm_mask_bits, wmfsm_mask_width, wmfsm_mask_height);
|
|
|
|
if (xpm_free)
|
|
XpmFree(pixmap);
|
|
|
|
#ifndef SVR4
|
|
if (gethostname(hostname, 100) != 0) {
|
|
perror("gethostname");
|
|
exit(10);
|
|
}
|
|
#else
|
|
if (sysinfo(SI_HOSTNAME, hostname, 100) < 0) {
|
|
perror("sysinfo(SI_HOSTNAME)");
|
|
exit(20);
|
|
}
|
|
#endif
|
|
while (1) {
|
|
|
|
readFileSystems ();
|
|
usleep (100);
|
|
|
|
/*
|
|
* Process any pending X events.
|
|
*/
|
|
while (XPending(display)) {
|
|
XNextEvent(display, &event);
|
|
switch (event.type) {
|
|
case Expose:
|
|
RedrawWindow();
|
|
break;
|
|
case ButtonPress:
|
|
pressEvent(&event.xbutton);
|
|
mode = (mode + 1) % 2;
|
|
break;
|
|
case ButtonRelease:
|
|
break;
|
|
}
|
|
}
|
|
/*! Here read data
|
|
*/
|
|
for (i = 0; i < numberfs; i++) {
|
|
if (STATFS(mp[i], &buffer) != -1) {
|
|
per[i] = 100 - (int) ((float) buffer.f_bfree / (float) buffer.f_blocks * 100.0);
|
|
}
|
|
}
|
|
/*!
|
|
* Draw window.
|
|
*/
|
|
/*
|
|
* Clear window.
|
|
*/
|
|
copyXPMArea(5, 69, 54, 54, 5, 5);
|
|
/*! here draw values
|
|
* void copyXPMArea(int x, int y, int sx, int sy, int dx, int dy)
|
|
*/
|
|
|
|
if (numberfs > 4) {
|
|
for (i = 0, dy = 0; i < numberfs; i++) {
|
|
for (j = 0, dx = 0; j < LENMP && (unsigned int)j < strlen(mp[i]); j++) {
|
|
k = j + (strlen(mp[i]) > LENMP ? strlen(mp[i]) - LENMP : 0);
|
|
c = (int) mp[i][k];
|
|
switch (c) {
|
|
case '/':
|
|
c = 123;
|
|
break;
|
|
case '_':
|
|
c = 124;
|
|
break;
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
case '8':
|
|
case '9':
|
|
c += 79;
|
|
break;
|
|
default:
|
|
c = tolower(c);
|
|
break;
|
|
}
|
|
copyXPMArea(xpos[c - 97], ypos[c - 97], 5, 5, 6 + dx, 5 + dy);
|
|
dx += 5;
|
|
}
|
|
c = (int) ((float) per[i] * 0.52 + 66.0);
|
|
k = (10 - LENMP) * 5 + 2;
|
|
k = (float) per[i] / 100.0 * k;
|
|
k = LENMP * 5 + k;
|
|
k = k < LENMP * 5 + 1 ? LENMP * 5 + 1 : k;
|
|
c = c > 66 ? c : 66;
|
|
c = c > 117 ? 117 : c;
|
|
if (blink) {
|
|
if ((int) per[i] >= blinkper)
|
|
on[i] = !on[i];
|
|
else
|
|
on[i] = 1;
|
|
}
|
|
else
|
|
on[i] = 1;
|
|
|
|
for (j = LENMP * 5; j < k; j++) {
|
|
if (mode == FIRE && on[i])
|
|
copyXPMArea(66 + (float) (j - LENMP * 5) * 52.0 / (float) ((10 - LENMP) * 5), 29, 1, 5, 6 + j, 5 + dy);
|
|
else if (mode == NORMAL && on[i])
|
|
copyXPMArea(c, 29, 1, 5, 6 + j, 5 + dy);
|
|
}
|
|
dy += 6;
|
|
}
|
|
}
|
|
else { /*one fs in two lines */
|
|
for (i = 0, dy = 0; i < numberfs; i++) {
|
|
for (j = 0, dx = 0; j < 10 && (unsigned int)j < strlen(mp[i]); j++) {
|
|
c = (int) mp[i][j + (strlen(mp[i]) > 10 ? strlen(mp[i]) - 10 : 0)];
|
|
switch (c) {
|
|
case '/':
|
|
c = 123;
|
|
break;
|
|
case '_':
|
|
c = 124;
|
|
break;
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
case '8':
|
|
case '9':
|
|
c += 79;
|
|
break;
|
|
default:
|
|
c = tolower(c);
|
|
break;
|
|
}
|
|
copyXPMArea(xpos[c - 97], ypos[c - 97], 5, 5, 6 + dx, 5 + dy);
|
|
dx += 5;
|
|
}
|
|
dy += 6;
|
|
c = (int) ((float) per[i] * 0.52 + 66.0);
|
|
k = (float) per[i] / 100.0 * 52;
|
|
k = k < 1 ? 1 : k;
|
|
c = c > 66 ? c : 66;
|
|
c = c > 117 ? 117 : c;
|
|
if (blink) {
|
|
if ((int) per[i] >= blinkper)
|
|
on[i] = !on[i];
|
|
else
|
|
on[i] = 1;
|
|
}
|
|
else
|
|
on[i] = 1;
|
|
|
|
for (j = 0; j < k; j++) {
|
|
if (mode == FIRE && on[i])
|
|
copyXPMArea(66 + (float) (j) * 52.0 / 50.0, 29, 1, 5, 6 + j, 5 + dy);
|
|
else if (mode == NORMAL && on[i])
|
|
copyXPMArea(c, 29, 1, 5, 6 + j, 5 + dy);
|
|
}
|
|
dy += 6;
|
|
}
|
|
}
|
|
if (numberfs < 9) {
|
|
for (j = 0, dx = 0, dy = 47; j < 10 && (unsigned int)j < strlen(hostname); j++) {
|
|
c = (int) hostname[j];
|
|
switch (c) {
|
|
case '/':
|
|
c = 123;
|
|
break;
|
|
case '_':
|
|
c = 124;
|
|
break;
|
|
case '-':
|
|
c = 125;
|
|
break;
|
|
case '.':
|
|
c = 126;
|
|
break;
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
case '8':
|
|
case '9':
|
|
c += 79;
|
|
break;
|
|
default:
|
|
c = tolower(c);
|
|
break;
|
|
}
|
|
copyXPMArea(xpos[c - 97], ypos[c - 97], 5, 5, 6 + dx, 6 + dy);
|
|
dx += 5;
|
|
}
|
|
}
|
|
|
|
RedrawWindow();
|
|
/*
|
|
* Wait for next update
|
|
*/
|
|
i = 0;
|
|
while (i < 10) {
|
|
usleep(delay);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* ParseCMDLine()
|
|
*/
|
|
void
|
|
ParseCMDLine(int argc, char *argv[])
|
|
{
|
|
|
|
int c, option_index;
|
|
static struct option long_options[] = {
|
|
{"fire", no_argument, 0, 'f'},
|
|
{"normal", no_argument, 0, 'n'},
|
|
{"blink", no_argument, &blink, 1},
|
|
{"noblink", no_argument, &blink, 0},
|
|
{"delay", required_argument, 0, 'd'},
|
|
{"appearance", required_argument, 0, 'a'},
|
|
{"help", no_argument, 0, 'h'},
|
|
{"version", no_argument, 0, 'v'},
|
|
{0, 0, 0, 0}
|
|
};
|
|
while (1) {
|
|
c = getopt_long(argc, argv, "a:bd:fhnv", long_options, &option_index);
|
|
if (c == -1)
|
|
break;
|
|
switch (c) {
|
|
case 0: /* If blink or noblink was toggled */
|
|
break;
|
|
case 'a':
|
|
strcpy(appearance, optarg);
|
|
break;
|
|
case 'b':
|
|
blink = 1;
|
|
break;
|
|
case 'd':
|
|
delay = atoi(optarg);
|
|
break;
|
|
case 'f':
|
|
mode = FIRE;
|
|
break;
|
|
case 'h':
|
|
print_usage();
|
|
exit(0);
|
|
case 'n':
|
|
mode = NORMAL;
|
|
break;
|
|
case 'v':
|
|
printf("wmfsm version: %s\n", PACKAGE_VERSION);
|
|
exit(0);
|
|
case '?':
|
|
break;
|
|
default:
|
|
print_usage();
|
|
exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
print_usage()
|
|
{
|
|
|
|
printf("usage: wmfsm \n");
|
|
printf("\t--normal, -n\t\tDraw bars in normal mode.\n");
|
|
printf("\t--fire, -f\t\tDraw bars in fire mode.\n");
|
|
printf("\t--[no]blink\t\tBlinks if a filesystem is 95 percent full.\n");
|
|
printf("\t-display <Display>\tUse alternate X display.\n");
|
|
printf("\t--delay <number>, -d\tUse a delay that is not the default.\n");
|
|
printf("\t--appearance <file>, -a\tSelect an appearance file.\n");
|
|
printf("\t-h\t\t\tDisplay help screen.\n");
|
|
}
|
|
|
|
void
|
|
pressEvent(XButtonEvent * xev)
|
|
{
|
|
(void)xev;
|
|
ForceUpdate = 1;
|
|
return;
|
|
}
|
|
|
|
void
|
|
readFileSystems()
|
|
{
|
|
/* Wipe mp[] */
|
|
int i;
|
|
for (i = 0; i < 100; i++)
|
|
if (mp[i]) {
|
|
free(mp[i]);
|
|
mp[i]=0;
|
|
}
|
|
|
|
/* Look for the goods between #if defined(__OpenBSD__) -- tschroed */
|
|
#if defined(__OpenBSD__) || defined(__FreeBSD__)
|
|
#define MAXMOUNT 32
|
|
struct statfs sfs[MAXMOUNT];
|
|
int fscount;
|
|
|
|
/* OpenBSD has no /etc/mtab, we use getfsstat instead -- tschroed */
|
|
if ((fscount = getfsstat(sfs, sizeof(sfs), 0)) == -1) {
|
|
perror("getfsstat");
|
|
exit(1);
|
|
}
|
|
for (numberfs = 0; numberfs < fscount; numberfs++)
|
|
/* We won't watch RO mounts. */
|
|
if (sfs[numberfs].f_flags & MNT_RDONLY) {
|
|
numberfs--;
|
|
fscount--;
|
|
}
|
|
else
|
|
mp[numberfs] = strdup(sfs[numberfs].f_mntonname);
|
|
#else /* __OpenBSD__ || __FreeBSD__ */
|
|
|
|
FILE *fp;
|
|
char mountPoint[255], dummy[255], fstype[255], options[512];
|
|
|
|
#if defined(SunOS)
|
|
/* Solaris uses /etc/mnttab */
|
|
fp = fopen("/etc/mnttab", "r");
|
|
if (!fp) {
|
|
fprintf(stderr, "%s:Can't open /etc/mnttab for reading\n", myName);
|
|
exit(1);
|
|
}
|
|
#else
|
|
fp = fopen("/etc/mtab", "r");
|
|
if (!fp) {
|
|
fprintf(stderr, "%s:Can't open /etc/mtab for reading\n", myName);
|
|
exit(1);
|
|
}
|
|
#endif
|
|
|
|
numberfs = 0;
|
|
while (!feof(fp) && numberfs < 100) {
|
|
#if defined(SunOS)
|
|
/* only five entries per row in /etc/mnttab */
|
|
if (fscanf(fp, "%s %s %s %s %s\n", dummy, mountPoint, fstype, options, dummy) < 5)
|
|
fprintf(stderr, "%s:/etc/mnttab not in expected format\n", myName);
|
|
#else
|
|
if (fscanf(fp, "%s %s %s %s %s %s\n", dummy, mountPoint, fstype, options, dummy, dummy) < 6)
|
|
fprintf(stderr, "%s:/etc/mtab not in expected format\n", myName);
|
|
#endif
|
|
|
|
if (
|
|
#if defined IRIX64 || defined(SunOS)
|
|
strcmp(fstype, "hwgfs") && strcmp(fstype, "autofs") && strcmp(fstype, "proc") && strcmp(fstype, "fd") && !strstr(options, "ignore")
|
|
#elif defined linux
|
|
strcmp(fstype, "proc") && strcmp(fstype, "tmpfs") && strcmp(fstype, "devfs") && strcmp(fstype, "ramfs") && strcmp(fstype, "sysfs") && strcmp(fstype, "devpts") && strcmp(fstype, "usbfs")
|
|
#else
|
|
1
|
|
#endif
|
|
) {
|
|
if (mp[numberfs]) free(mp[numberfs]);
|
|
mp[numberfs++] = strdup(mountPoint);
|
|
}
|
|
}
|
|
#endif /* __OpenBSD__ || __FreeBSD__ */
|
|
|
|
fclose (fp);
|
|
excludeFileSystems();
|
|
}
|
|
|
|
void
|
|
excludeFileSystems()
|
|
{
|
|
char confFileName[255];
|
|
char workString[255];
|
|
char *home;
|
|
int i, j, exnumberfs = 0;
|
|
int excluded, finalnumberfs = 0;
|
|
char *mount_points[100];
|
|
FILE *confFile;
|
|
int include = -1;
|
|
int included = 0;
|
|
|
|
home = getenv("HOME");
|
|
if (home == NULL) {
|
|
fprintf(stderr, "error: HOME must be defined\n");
|
|
exit(1);
|
|
}
|
|
if (strlen(home) > 244) {
|
|
fprintf(stderr,
|
|
"error: HOME may not be longer than 244 characters\n");
|
|
exit(1);
|
|
}
|
|
strncpy(confFileName, home, 245);
|
|
strcat(confFileName, "/.wmfsmrc");
|
|
confFile = fopen(confFileName, "r");
|
|
if (confFile) {
|
|
while (!feof(confFile)) {
|
|
if (fgets(workString, 255, confFile)) {
|
|
if(strstr(workString, "\n")){
|
|
/* This is probably dangerous, but easy */
|
|
*(strstr(workString, "\n")) = '\0';
|
|
}
|
|
if (!strcmp(workString, "[include]")) {
|
|
include = 1;
|
|
}
|
|
else if (!strcmp(workString, "[exclude]")) {
|
|
include = 0;
|
|
}
|
|
else {
|
|
mount_points[exnumberfs] = strdup(workString);
|
|
exnumberfs++;
|
|
}
|
|
}
|
|
|
|
}
|
|
if (include < 0) {
|
|
printf("wmfsm: Please specify either to include or exclude filesystems.\n");
|
|
exit(1);
|
|
}
|
|
fclose(confFile);
|
|
}
|
|
else {
|
|
numberfs = numberfs > 9 ? 9 : numberfs;
|
|
goto CLEANUP;
|
|
}
|
|
if (!exnumberfs) {
|
|
numberfs = numberfs > 9 ? 9 : numberfs;
|
|
goto CLEANUP;
|
|
}
|
|
excluded = 0;
|
|
for (i = 0; i < numberfs; i++) {
|
|
for (j = 0, excluded = 0; j < exnumberfs; j++) {
|
|
if (!strcmp(mp[i], mount_points[j]) && !include) {
|
|
excluded = 1;
|
|
} else if (!strcmp(mp[i], mount_points[j]) && include) {
|
|
included = 1;
|
|
}
|
|
}
|
|
if ((!excluded && !include) || (included && include)) {
|
|
if (finalnumberfs == i) {
|
|
finalnumberfs++;
|
|
} else {
|
|
if (mp[finalnumberfs]) free(mp[finalnumberfs]);
|
|
mp[finalnumberfs++] = strdup(mp[i]);
|
|
}
|
|
}
|
|
included = excluded = 0;
|
|
}
|
|
numberfs = finalnumberfs > 9 ? 9 : finalnumberfs;
|
|
|
|
CLEANUP:
|
|
for (j = 0; j < exnumberfs; j++)
|
|
if (mount_points[j]) {
|
|
free(mount_points[j]);
|
|
mount_points[j]=0;
|
|
}
|
|
}
|