💾 Archived View for uscoffings.net › retro-computing › components › PC › interrupts › INT.C captured on 2023-03-20 at 21:31:05.

View Raw

More Information

⬅️ Previous capture (2022-06-04)

-=-=-=-=-=-=-

/*
 * File:	int.c
 *
 * Synopsis:	PC interrupt caller, memory R/W, I/O port R/W
 *
 * System:	MSDOS - Turbo C or Borland C (other compilers need some work)
 *
 * A utility for PC systems programmers to:
 *	- perform and visualise the results of interrupt calls
 *	- use buffers (inc files) to set register pairs
 *	- view the interrupt vector table
 *	- read & write I/O ports
 *
 * NB: This utility must be used with EXTREME CARE. Using bad interrupt and/or
 * I/O calls can destroy data on your memory/disk(s) and might crash your
 * machine.
 *
 * Compatible with int.c ditributed with Ralf Brown's interrupt lists
 * Comments/suggestions welcome on the email address below.
 *
 *
 * Copyright (c) 1992-1994 Angelo Haritsis <ah@doc.ic.ac.uk>
 *
 * Redistribution and use in source and binary forms are permitted provided
 * that the above copyright notice and this paragraph are duplicated in all
 * such forms and that any documentation, advertising materials, and other
 * materials related to such distribution and use acknowledge that the
 * software was developed by Angelo Haritsis.
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
 * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
 *
 * TODO:
 * > -fbuf filename (write mem with vals from a file)
 */

#ifndef lint
static char rcsid[] = "$Header: E:/SRC/MISC\RCS\int.c 1.2 1994/04/11 20:11:36 ah Exp ah $";
#endif

#include <stdio.h>
#include <io.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <ctype.h>
#include <dos.h>

#define PROG "int"
#define VERSION "1.0"
#define NOTREACHED 0


#define PARSE_JUSTARGS	0x01
#define PARSE_NOACTION	0x02

/*
 * Types
 */
typedef int (*fun_ptr) (char **, int *);

typedef struct s_opt {
	char *opts;		/* option name */
	int size;		/* bytes for the ptr element (1, 2, 4);
				 * 0 means execute the function following
				 * parsing (argv, &opt_now)
				 */
	void *ptr;		/* pointer to the data element to be loaded or a function if size = 0 */
} OPTION;

typedef struct s_dbuff {
	unsigned *ptr_to_seg;	/* the area where the segment is held */
	unsigned *ptr_to_off;	/* the area where the offset is held */
	void far *addr; 	/* the address (when explicitly given) */
	char logo[20];
#define DBUFF_HEX 0x01
#define DBUFF_ASC 0x02
#define DBUFF_BIN 0x04
	int mode;		/* display mode */
	long bytes;		/* # of bytes to show */
} DBUFFER;

typedef struct s_doserr {
	int errno;
	int class;
	int action;
	int locus;
} DOS_ERROR;

/*
 * Globals
 */
char switchchar[] = "-+/";

char usage[] =
"usage: " PROG " [-h] [-q] [int=# | int# | -int #] [[-]reg[=]val] ...\n"
"       [-buf sreg:reg=\"str\"]\n"
"       [-d[a|b|hx] sreg:reg[:#bytes]] [-d[a|hx] faraddr[:#bytes]]\n"
"       [-divt int#-int#]]\n"
"       [-in[w] port#] [-out[w] port#,val]\n"
"       str: printf fmt (%#s allowed to setup buffers)\n";

char help[] =
" eg.\n"
"     o Interrupts (+ buffer setup)\n"
"       " PROG " 0x10 -ax 0x12 same as: int int=0x10 ax=0x12 (set VGA graphics mode)\n"
"       " PROG " 0x10 ah=0x10 al=1 bh=9 (set VGA border color; use to adjust display)\n"
"       " PROG " 0x21 -ah 9 -buf ds:dx=\"hello world\\n$\" -q (INT 21,9 'hello world')\n"
"       " PROG " an_int -buf es:bx=\"%512s\" (creates a scratch buffer of 512 bytes)\n"
"       " PROG " 0x33 ax=9 bx=0 cx=0 -fbuf es:bx=cursor.dat (mouse cursor from file)\n"
"     o Memory (-d: display: called after interrupt)\n"
"       " PROG " -q -dhx 0x400049:18 (BIOS video area data)\n"
"       " PROG " -q es=0x40 -bx 0x49 -dhx es:bx:18 (same as above)\n"
"       " PROG " -q -db 0xF8000000:512 >file (binary dump of AMIBIOS serial data)\n"
"     o IVT\n"
"       " PROG " -q -divt (display vectors from interrupt vector table = IVT)\n"
"       " PROG " -q -divt 0x10-0x1F (display IVT slots 0x10 to 0x1F)\n"
"     o I/O Ports\n"
"       " PROG " -out 0x70,0 -in 0x71 (read seconds from CMOS)\n"
"       " PROG " -out 0x70,0 -out 0x71,0 (zero-out seconds in CMOS)\n";

char *int_error[] = {
	NULL,
	"Parse",
};

char str_flags[] = "0N..ODITSZ0A0P1C";  /* 80x86 flag register */

union REGS reg;
struct SREGS sreg;
unsigned int reg_bp, reg_sp, flags;
int int_call;			/* the interrupt to be called */
int dos_errno;
int quiet_mode = 0;		/* quiet means just execute int - no reg display * /* does not apply on display
				 * opts */
int buff_now = 0;
char *buff[10]; 	    /* array of read buffer pointers */

int dbuff_now = 0;
DBUFFER dbuff[10];		/* the buffers to show at end of intr */


/* --- Prototypes --- */
OPTION *parse_an_opt(char *argv[], int *opt_now, int mode);
void doit(void);
void reg_display(void);
int set_int_val(char **argv, int *opt_now);
int ivt_display(char **argv, int *opt_now);
int ioport_out(char **argv, int *opt_now);
int ioport_in(char **argv, int *opt_now);
int read_escape(char **p);
int set_buff(char **argv, int *opt_now);
int set_fbuff(char **argv, int *opt_now);
int set_dbuff(char **argv, int *opt_now);
int dbuff_parse(void **ptr, char *tok);
int set_dbuff(char **argv, int *opt_now);
int show_help(char **argv, int *opt_now);
int set_quiet(char **argv, int *opt_now);
void error_exit(int err, char *s);

/* --- */

/*
 * Structure with all the `action' to be done for an option
 * NB: Care with matching prefixes (longer must come first)
 */
OPTION Opt[] = {

	/* NB: put the longer strings first ! */

	{"$DEFAULT", 0, (void *) set_int_val},
	{"DIVT", 0, (void *) ivt_display},      /* display int vector table */
	{"INT", 2, (void *) &int_call},

	{"OUTW", 0, (void *) ioport_out},       /* I/O port write, read */
	{"OUT", 0, (void *) ioport_out},
	{"INW", 0, (void *) ioport_in},
	{"IN", 0, (void *) ioport_in},

	{"DBUF", 0, (void *) set_dbuff},
	{"FBUF", 0, (void *) set_fbuff},
	{"BUF", 0, (void *) set_buff},
	{"DHX", 0, (void *) set_dbuff},         /* display mem contents (hex) */

	{"FL", 2, (void *) &(reg.x.flags)},     /* set flags (won't affect anything) */

	{"AX", 2, (void *) &(reg.x.ax)},        /* set general registers */
	{"BX", 2, (void *) &(reg.x.bx)},
	{"CX", 2, (void *) &(reg.x.cx)},
	{"DX", 2, (void *) &(reg.x.dx)},
	{"AH", 1, (void *) &(reg.h.ah)},
	{"BH", 1, (void *) &(reg.h.bh)},
	{"CH", 1, (void *) &(reg.h.ch)},
	{"DH", 1, (void *) &(reg.h.dh)},
	{"AL", 1, (void *) &(reg.h.al)},
	{"BL", 1, (void *) &(reg.h.bl)},
	{"CL", 1, (void *) &(reg.h.cl)},
	{"DL", 1, (void *) &(reg.h.dl)},

	{"SI", 2, (void *) &(reg.x.si)},        /* set index, stack registers */
	{"DI", 2, (void *) &(reg.x.di)},
	{"BP", 0, 0},
	{"SP", 0, 0},

	{"CS", 2, (void *) &(sreg.cs)},         /* set segment registers */
	{"DS", 2, (void *) &(sreg.ds)},
	{"ES", 2, (void *) &(sreg.es)},
	{"SS", 2, (void *) &(sreg.ss)},

	{"DA", 0, (void *) set_dbuff},          /* display mem contents (ascii) */
	{"D", 0, (void *) set_dbuff},           /* display mem contents (ascii+hex) */
	{"Q", 0, (void *) set_quiet},           /* quiet (no disply of reg contents) */
	{"H", 0, (void *) show_help},           /* help */
};


int cdecl
main(int argc, char *argv[])
{
	int opt_now;

	if (argc == 1)
		error_exit(0, NULL);

	/* parse the arguments and do proper action */
	for (opt_now = 1; opt_now < argc; opt_now++)
		if (parse_an_opt(argv, &opt_now, 0) == NULL)
			error_exit(1, NULL);
	doit();
	return (0);
}


/*
 * Parses an argument and calls proper function or assigns numbers
 * accordingly. Reentrant.
 */
OPTION *
parse_an_opt(char *argv[], int *opt_now, int mode)
{
	int i, arg_len, get_next_arg;
	char *opts, *optarg, *p;
	long val;

	opts = argv[*opt_now];
	if (strchr(switchchar, opts[0]))	/* option starts with a switch char, skip it */
		opts++, argv[*opt_now]++;
	for (i = 0; i < (sizeof(Opt) / sizeof(OPTION)); i++) {
		arg_len = strlen(Opt[i].opts);
		get_next_arg = opts[arg_len] == 0;
		if (strncmpi(opts, Opt[i].opts, arg_len) == 0) {
			if (mode & PARSE_NOACTION)	/* do not perform action */
				return (&Opt[i]);	/* just return ptr to Opt slot */
			switch (Opt[i].size) {

			case 0: /* call the function */
				if (!(mode & PARSE_JUSTARGS) && Opt[i].ptr != (void *) 0)
					if ((*((fun_ptr) Opt[i].ptr)) (argv, opt_now) == 0)
						return (NULL);	/* error */
				return (&Opt[i]);

			case 1: /* ptr is a byte, short, int */
			case 2:
			case 4:
				if (get_next_arg)
					optarg = argv[++(*opt_now)];	/* get next option */
				else
					optarg = opts + arg_len + 1;	/* skip a separator (eg =) */
				val = strtol(optarg, &p, 0);
				if (p == optarg)
					return (NULL);
				switch (Opt[i].size) {
				case 1:
					*((char *) Opt[i].ptr) = (char) val;
					break;
				case 2:
					*((short *) Opt[i].ptr) = (short) val;
					break;
				case 4:
					*((long *) Opt[i].ptr) = (long) val;
					break;
				}
				return (&Opt[i]);

			default:
				assert(NOTREACHED);
			}
		}
	}
	if (mode & PARSE_JUSTARGS)
		return (&Opt[0]);	/* default */
	else {
		i = (*((fun_ptr) Opt[0].ptr)) (argv, opt_now);	/* default */
		return (i == 0 ? NULL : &Opt[0]);
	}
}

/*
 * Call the interrupt if asked and display the result buffers
 */
void
doit(void)
{
	int i;
	long j;
	unsigned char far *ptr;
	unsigned char b;

	dos_errno = 0;
	if (int_call != -1) {
		reg_bp = _BP;
		reg_sp = _SP;
		flags = _FLAGS;
		reg_display();
	}
	if (int_call > 0 && int_call < 256) {
		quiet_mode || printf("\nINT: 0x%02X\n", int_call);
		int86x(int_call, &reg, &reg, &sreg);	/**/
		if (reg.x.cflag != 0)	/* error occured */
			dos_errno = _doserrno;
		reg_bp = _BP;
		reg_sp = _SP;
		quiet_mode || printf("\n");
		flags = reg.x.flags;
		reg_display();
		if (!quiet_mode && (int_call == 0x21 || int_call == 0x24))	/* dos call */
			printf("DOSERR: %04X (%u)\n", dos_errno, dos_errno);
	}
	/* display dbuffers */

	for (i = 0; i < dbuff_now; i++) {
		ptr = (unsigned char far *) MK_FP(*(dbuff[i].ptr_to_seg), *(dbuff[i].ptr_to_off));

		if (dbuff[i].mode & DBUFF_BIN)		/* binary */
			setmode(1, O_BINARY);
		else
			printf("\n*<%s> {\n", dbuff[i].logo);
		for (j = 0; j < dbuff[i].bytes; j++, ptr++) {
			b = *ptr & 0x00FF;		/* byte to display */
			if (dbuff[i].mode & DBUFF_BIN) {	/* binary */
				putchar(b);
				continue;	/* nothing else */
			}
			if (dbuff[i].mode & DBUFF_HEX)
				printf("%02X", b);
			if (dbuff[i].mode == DBUFF_ASC)
				putchar(iscntrl(b) ? '.' : b);
			else if (dbuff[i].mode & DBUFF_ASC)
				printf("(%c)", iscntrl(b) ? '.' : b);
			if (dbuff[i].mode != DBUFF_ASC)
				printf(" ");
		}
		fflush(stdout);
		if (dbuff[i].mode & DBUFF_BIN)	       /* binary */
			setmode(1, O_TEXT);
		else
			printf("}\n");
	}
	/* free the read buffers allocated */
	for (i = 0; i < buff_now; i++)
		free(buff[i]);
}

void
reg_display(void)
{
	char s[32];
	int i, bit_on;

	if (quiet_mode)
		return;
	printf(
	       "AX=%04X   BX=%04X   CX=%04X   DX=%04X\n"
	       "SI=%04X   DI=%04X   BP=%04X   SP=%04X\n"
	       "CS=%04X   DS=%04X   ES=%04X   SS=%04X",
	       reg.x.ax, reg.x.bx, reg.x.cx, reg.x.dx,
	       reg.x.si, reg.x.di, reg_bp, reg_sp,
	       sreg.cs, sreg.ds, sreg.es, sreg.ss);
	strncpy(s, str_flags, 32);	/* use a scratch copy */
	/* and now the flags */
	flags = reg.x.flags;
	for (i = 0; i < 16; i++) {
		bit_on = (flags & ((unsigned) 0x8000 >> i)) != 0;
		if (s[i] == '.')
			s[i] = bit_on ? '1' : '0';
		else
			s[i] = bit_on ? s[i] : tolower(s[i]);
	}
	printf("   CPU Flags: %16s\n", s);
}

/*
 * 'default' argument function - see if it is an interrupt
 */
int
set_int_val(char **argv, int *opt_now)
{
	long val;
	char *p;

	val = strtol(argv[*opt_now], &p, 0);
	if (val <= 0 || val > 255 || p - argv[*opt_now] != strlen(argv[*opt_now]))
		return (0);	/* problems */
	int_call = (int) val;
	return (1);
}

/*
 * Display a slot of the Interrupt Vector Table
 */
int
ivt_display(char **argv, int *opt_now)
{
	char sfrom[20], sto[20];
	int from, to, i;
	void far *p;

	if ((i = sscanf(argv[*opt_now + 1], "%[0-9xX]-%s", sfrom, sto)) == 2) { /* is a range given ? */
		(*opt_now)++;	/* consume next arg */
		from = (int) strtol(sfrom, (char **) &sfrom, 0);
		to = (int) strtol(sto, (char **) &sto, 0);
	} else {
		from = 0;
		to = 255;
	}
	/* do it now */
	printf("Interrupt Vector Table (0x%02X to 0x%02X)\n", from, to);
	for (i = from; i <= to; i++) {
		disable();	/* just in case ... */
		p = (void far *) *((long far *) (4L * i));
		enable();
		printf(" * 0x%02X (%3u):  %Fp\n", i, i, p);
	}
	printf("\n");
	return (1);
}

int
ioport_out(char **argv, int *opt_now)
{
	char *optarg, sport[10], sval[10];
	int word_op, port, val;

	optarg = argv[*opt_now];
	word_op = (toupper(optarg[3]) == 'W') ? 1 : 0;
	if (isdigit(optarg[3 + word_op]))	/* arg follows with no delimiter */
		optarg += 3 + word_op;
	else
		optarg = argv[++(*opt_now)];
	if (sscanf(optarg, "%[^ ,;]%*[ ,;]%s", sport, sval) != 2)
		return (0);
	port = (int) strtol(sport, (char **) &sport, 0);
	val = (int) strtol(sval, (char **) &sval, 0);
	if (word_op)
		outport(port, (unsigned) val);
	else
		outportb(port, val);
	int_call = -1;
	return (1);
}

int
ioport_in(char **argv, int *opt_now)
{
	char *optarg, sport[10];
	int word_op, port, val;

	optarg = argv[*opt_now];
	word_op = (toupper(optarg[2]) == 'W') ? 1 : 0;
	if (isdigit(optarg[2 + word_op]))	/* arg follows with no delimiter */
		optarg += 2 + word_op;
	else
		optarg = argv[++(*opt_now)];
	if (sscanf(optarg, "%s", sport) != 1)
		return (0);
	port = (int) strtol(sport, (char **) &sport, 0);
	if (word_op) {
		val = inport(port);
		quiet_mode || printf("INW 0x%04X (%5u):  0x%04X (%5u)\n", port, port, val, val);
	} else {
		val = inportb(port);
		quiet_mode || printf("IN 0x%04X (%5u):  0x%02X (%3u)\n", port, port, val, val);
	}
	quiet_mode || printf("\n");
	int_call = -1;
	return (1);
}

#define ESCAPES 10
static int esc_to_code[ESCAPES][2] = {
	{'n', '\n'},
	{'t', '\t'},
	{'v', '\v'},
	{'b', 'b'},
	{'r', '\r'},
	{'f', '\f'},
	{'a', '\a'},
	{'\\', '\\'},
	{'\?', '?'},
	{'\'', '\''},
};

/*
 * returns with *p pointing to char after the one(s) consumed
 */
int
read_escape(char **p)
{
	int i;

	if (isdigit(**p))	/* octal */
		return ((int) strtol(*p, p, 8));
	else if (**p == 'x')    /* hex */
		return ((int) strtol(*p + 1, p, 16));
	for (i = 0; i < ESCAPES; i++)
		if (**p == esc_to_code[i][0]) {
			(*p)++; /* consume it */
			return (esc_to_code[i][1]);
		}
	/* otherwise, return the character un-translated */
	(*p)++; 		/* consume it */
	return (**p);
}


/*
 * load seg register values to point ot the created buffer
 */
void
load_regpair(char *s_seg, char *s_off, void far *buff)
{
	int len;
	char stmp[50], *argv_fake[3];

	/* load the regs */
	argv_fake[0] = stmp;
	argv_fake[1] = "";
	len = 0;
	sprintf(stmp, "-%s 0x%X", s_seg, FP_SEG(buff));
	parse_an_opt(argv_fake, (int *) &len, PARSE_JUSTARGS);	/* make it think it's an option */
	sprintf(stmp, "-%s 0x%X", s_off, FP_OFF(buff));
	parse_an_opt(argv_fake, (int *) &len, PARSE_JUSTARGS);	/* and again for offs register */
}

/*
 * set registers accordingly
 */
int
set_buff(char **argv, int *opt_now)
{
	char *optarg, *p, *dp;
	char s_seg[10], s_off[10];
	char stmp[50];
	static char dummy[] = "";       /* for case of %s in fmt str */
	unsigned int len;

	optarg = argv[++(*opt_now)];	/* s_off pair */
	sscanf(optarg, "%[^:]:%s", &s_seg, &s_off);
	if (s_off[2] == '=')
		s_off[2] = 0;
	optarg = argv[++(*opt_now)];	/* printf like string */
	/* how big a buffer ? */
	len = strlen(optarg);
	/* add the %# lengths (extra buffer space) */
	for (p = strchr(optarg, '%'); p != NULL; p = strchr(p + 1, '%'))
		len += atoi(p + 1);

	if ((buff[buff_now] = (char *) malloc(len)) == NULL)
		return (0);
	/* create escape chars again (since cmd processing makes \ into \\) */
	p = optarg, dp = stmp;
	while (*p)
		if (*p == '\\') {
			p++;	/* consume \ */
			*dp++ = read_escape(&p);
		} else
			*dp++ = *p++;

	/* load the buffer; 5 % fields are enough (XXX ..f func problem if \0 appears before end of fmt) */
	sprintf(buff[buff_now], stmp, dummy, dummy, dummy, dummy, dummy);

	load_regpair(s_seg, s_off, (void far *) buff[buff_now]);
	buff_now++;
	return (1);
}

/*
 * set register pair to point to buffer with data from a file
 */
int
set_fbuff(char **argv, int *opt_now)
{
	char *optarg, *fname;
	char s_seg[10], s_off[80];
	long len;
	FILE *f;

	optarg = argv[++(*opt_now)];	/* s_off pair */
	sscanf(optarg, "%[^:]:%s", &s_seg, &s_off);
	if (s_off[2] == '=') {
		s_off[2] = 0;
		fname = &s_off[3];
	} else
		return (0);

	if ((f = fopen(fname, "rb")) == NULL)
		return (0);
	len = filelength(fileno(f));
	if (len > 65500L || (buff[buff_now] = (char *) malloc((unsigned int)len)) == NULL)
		return (0);
	if (fread(buff[buff_now], (int) len, 1, f) != 1)
		return (0);
	load_regpair(s_seg, s_off, (void far *) buff[buff_now]);
	buff_now++;
	fclose(f);
	return (1);
}

int
dbuff_parse(void **ptr, char *tok)
{
	char stmp[50], *argv_fake[3];
	OPTION *p_opt;
	int len = 0;

	argv_fake[0] = stmp;
	argv_fake[1] = "";
	sprintf(stmp, "-%s 0", tok);
	p_opt = parse_an_opt(argv_fake, &len, PARSE_JUSTARGS | PARSE_NOACTION);
	if (p_opt == NULL || p_opt == &Opt[0])
		return (0);
	*ptr = (unsigned *) p_opt->ptr;
	return (1);
}

/*
 * add to the list of the buffers to be displayed at end
 */
int
set_dbuff(char **argv, int *opt_now)
{
	char mode_char;
	char *optarg, *p;
	char tok[3][15];	/* max 3 tokens of 15 chars each */
	int i;
	long addr;
	unsigned long num = 1L; /* number of bytes to display */
	DBUFFER *dpb;

	dpb = &dbuff[dbuff_now];
	dpb->mode = DBUFF_HEX | DBUFF_ASC;
	mode_char = toupper(argv[*opt_now][1]);
	dpb->mode &= (mode_char == 'A') ? ~DBUFF_HEX : ~0;
	dpb->mode &= (mode_char == 'H') ? ~DBUFF_ASC : ~0;
	if (mode_char == 'B') {         /* binary mode */
		dpb->mode &= ~(DBUFF_HEX | DBUFF_ASC);
		dpb->mode |= DBUFF_BIN;
	}
	optarg = argv[++(*opt_now)];	/* reg pair */
	strncpy(dpb->logo, optarg, 20);
	/* collect tokens */
	for (i = 0, p = strtok(optarg, ":="); p; p = strtok(NULL, ":="))
		strcpy(tok[i++], p);
	if (i > 3)
		return (0);
	/* process them */
	addr = strtoul(tok[0], &p, 0);
	if ((p - tok[0]) > 0) { /* first is addr */
		if (i > 1) {	/* there's a 2nd token */
			num = strtoul(tok[1], &p, 0);
			if ((p - tok[1]) == 0 || num == 0)
				return (0);	/* wrong argument */
		}
		dpb->addr = (void far *) addr;
		dpb->ptr_to_off = (unsigned *) &(dpb->addr);
		dpb->ptr_to_seg = ((unsigned *) &(dpb->addr)) + 1;
	} else {		/* should be Reg:Reg[:#] format */
		if (dbuff_parse((void **) &(dpb->ptr_to_seg), tok[0]) == 0)
			return (0);
		if (dbuff_parse((void **) &(dpb->ptr_to_off), tok[1]) == 0)
			return (0);
		if (i > 2) {	/* num argument */
			num = strtoul(tok[2], &p, 0);
			if ((p - tok[2]) == 0 || num == 0)
				return (0);	/* wrong argument */
		}
	}
	dpb->bytes = num;
	dbuff_now++;		/* have inserted an element */
	return (1);
}

int
set_quiet(char **argv, int *opt_now)
{
	argv = argv, opt_now = opt_now; /* eliminate warning */
	return (quiet_mode = 1);
}

#define fmsg stdout		/* DOS stderr cannot be redir'ed >:-{ */

int
show_help(char **argv, int *opt_now)
{
	argv = argv, opt_now = opt_now; /* eliminate warning */
	fprintf(fmsg,
		PROG ": Execute and investigate interrupts/system data (ver " VERSION ")\n"
		"Copyright (c) 1992-1994 A. Haritsis <ah@doc.ic.ac.uk>. Distribute freely.\n");
	error_exit(0, help);
	return (1);
}

void
error_exit(int err, char *s)
{
	if (err > 0)
		fprintf(fmsg, PROG ": %s error\n", int_error[err]);
	fprintf(fmsg, "%s", usage);
	if (s != NULL)
		fprintf(fmsg, "%s", s);
	exit(err);
}