home *** CD-ROM | disk | FTP | other *** search
- /*Program 9_2a -- Display a clock on the screen
- by Stephen R. Davis, 1987
-
- As an example of an interrupt routine, inserted onto the timer
- tick interrupt. On each clock tick, display an ASCII clock
- in the upper right hand corner of the clock. Display the
- current time and the delta time since installed, both in
- 24 hour format.
-
- If DEBUG is set, install routine in a harmless vector and
- invoke it from main(). This is to allow breakpoints to be
- set in the interrupt routine easily. If DEBUG is
- either 0 or not defined, install normally (define via Options
- menu).
-
- (As always, this program may not be compatible with other
- TSR type programs.)
- */
-
- #include <stdio.h>
- #include <dos.h>
- #include <stdlib.h>
- #include <process.h>
-
- /*first the prototyping definitions*/
-
- void interrupt clock (void);
- void display (int, int, int);
- void out (char *, int, int);
- void init (void);
- int restore (void);
-
- /*define our data structures*/
-
- union {
- long ltime;
- int stime [2];
- } p;
- struct REGS regs;
- int far *screen;
- int prevtime, time, minute, hour;
- char buffer [10];
- void interrupt (*fn)();
-
- /*define the DEBUG relative information*/
-
- #if DEBUG
- #define vect 0x48
- struct REGS reg2;
- #else
- #define vect 0x1c
- #endif
-
- /*Clock - grab the interrupt and provide the function*/
- void interrupt clock (void)
- {
- /*get the time and convert it to minutes since midnite*/
- regs.h.ah = 0;
- int86 (0x1a, ®s, ®s);
- p.stime [0] = regs.x.dx;
- p.stime [1] = regs.x.cx;
- time = (int)(p.ltime / (long)1092);
-
- /*now display the current time in 24 hour format*/
- display (time, 0, 75);
-
- /*then display the delta*/
- if (prevtime == -1)
- prevtime = time;
- if ((time -= prevtime) < 0)
- time += (24 * 60);
- display (time, 1, 75);
- }
-
- /*Display - put a time on the screen in the position indicated.
- Remember we can't do any DOS calls here.*/
- void display (number, y, x)
- int number, x, y;
- {
- hour = number / 60;
- minute = _DX;
- sprintf (buffer, "%2.2u:%2.2u", hour, minute);
- buffer [5] = '\0';
- out (buffer, y, x);
- }
-
- /*Out - out a string onto the screen w/o using system call*/
- void out (buffer, y, x)
- char *buffer;
- int y, x;
- {
- int far *scrptr;
-
- #if DEBUG
- /*print out the data to make sure it is correct*/
- printf ("%s", buffer);
- #endif
- scrptr = screen + (y * 80 + x);
- while (*buffer)
- *scrptr++ = 0x1700 + *buffer++;
- }
-
- /*Main - exercise the above routine. When DEBUG is set to 1,
- 'vect' is directed to an otherwise unused vector (48).
- This allows us to easily debug the interrupt routine
- using our favorite debugger. Once debugged, DEBUG is
- set to 0 and the vector is redirected to the TimerTick
- interrupt (1C).*/
- main ()
- {
- int i, j, k;
-
- init ();
-
- fn = getvect (vect);
- setvect (vect, clock);
- ctrlbrk (restore);
-
- /*loop around for at least 2 minutes so that you can see
- the clock increment to prove to yourself that it is
- working*/
- for (i = 0; i < 150; i++) {
- for (j = 0; j < 30000; j++)
- k = j * 5;
- printf (" this is just dummy output --"
- " watch the clock\n");
- #if DEBUG
- /*invoke the interrupt so we can debug it*/
- int86 (vect, ®2, ®2);
- #endif
- };
-
- /*restore the interrupt to its former value*/
- setvect (vect, fn);
- }
-
- /*Init - set the screen address and clear the screen*/
- void init ()
- {
- #define mono (int far *)0xb0000000 /*for mono displays...*/
- #define cga (int far *)0xb8000000 /*...for ega and cga*/
- int mode;
-
- prevtime = -1;
-
- regs.h.ah = 0x0f;
- int86 (0x10, ®s, ®s);
- mode = regs.h.al;
- if (regs.h.ah != 80) /*fixed to 80 columns*/
- abort ();
-
- if (mode == 7)
- screen = mono;
- else
- if (mode == 3 || mode == 2)
- screen = cga;
- else
- abort ();
- }
-
- /*Restore - restore the old interrupt address in the event of a
- Control Break*/
- int restore (void)
- {
- /*restore the interrupt to its former value*/
- setvect (vect, fn);
-
- /*tell the operator what he has done*/
- printf ("\nBreak!\n");
-
- /*now exit the program back to DOS*/
- return 0;
- }