home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 7 / 07.iso / c / c072 / 1.ddi / PRG9_2D.C < prev    next >
Encoding:
C/C++ Source or Header  |  1987-09-19  |  5.0 KB  |  195 lines

  1. /*Program 9_2d -- Display a clock on the screen w/ Keyboard Control
  2.     by Stephen R. Davis, 1987
  3.  
  4.   This program should be compiled under the TINY memory
  5.   mode and may then be converted into .COM file using the command:
  6.  
  7.   EXE2BIN PRG9_2D.EXE PRG9_2D.COM
  8.  
  9.   It can also be executed as an .EXE file.  Do not execute from IDE!
  10.  
  11.   This version adds the capability that the delta time can be
  12.   hidden by entering '~' and cleared by entering '`'.  This is
  13.   only intended as an example of a full TSR type program.
  14.   (This version makes allowances for its own stack so that
  15.    SSTACK should not be necessary.)
  16. */
  17.  
  18. #include <stdio.h>
  19. #include <dos.h>
  20. #include <stdlib.h>
  21. #include <process.h>
  22.  
  23. #ifndef __TINY__
  24.      #error Should use TINY compilation model
  25. #endif
  26.  
  27. /*first the prototyping definitions*/
  28.  
  29. void interrupt clock (void);
  30. void display (int, int, int);
  31. void out (char *, int, int);
  32. void interrupt screenout(unsigned, unsigned, unsigned, unsigned,
  33.                          unsigned, unsigned, unsigned, unsigned,
  34.                          unsigned);
  35. void init (void);
  36.  
  37. /*define our data structures*/
  38.  
  39. union {
  40.        long ltime;
  41.        int stime [2];
  42.       } p;
  43. struct REGS regs;
  44. void interrupt (*oldclock)(void);
  45. void interrupt (*oldscreen)(void);
  46. int prevtime, time, minute, hour, dflag, temp;
  47. char buffer [6];
  48.  
  49. int far *screen;
  50. char digits [] = {"0123456789"};
  51.  
  52. /*allocate space for our own stack area*/
  53.  
  54. int savess, savesp, sflag;
  55. char stack [0x1000];
  56.  
  57. /*Clock - grab the interrupt and provide the function*/
  58. void interrupt clock (void)
  59. {
  60.      /*put in our own stack*/
  61.      if (!sflag) {
  62.           savesp = _SP;
  63.           savess = _SS;
  64.           _CX = (int)&stack[sizeof(stack)];
  65.           _SS = _DS;
  66.           _SP = _CX;
  67.      }
  68.      sflag++;
  69.  
  70.      /*get the curren time (long format) using the BIOS call
  71.        1a.  Divide this down into number of minutes (short
  72.        format).*/
  73.      regs.h.ah = 0;
  74.      int86 (0x1a, ®s, ®s);
  75.      p.stime [0] = regs.x.dx;
  76.      p.stime [1] = regs.x.cx;
  77.      time = (int)(p.ltime / (long)1092);
  78.  
  79.      /*now display the current time in 24 hour format*/
  80.      display (time, 0, 75);
  81.  
  82.      /*then display the delta*/
  83.      if (prevtime == -1)
  84.           prevtime = time;
  85.      if ((time -= prevtime) < 0)
  86.           time += (24 * 60);
  87.      if (dflag)
  88.           display (time, 1, 75);
  89.  
  90.      /*pass control on to the next interrupt routine*/
  91.      (*oldclock)();
  92.  
  93.      /*now restore caller's stack*/
  94.      if (!--sflag) {
  95.           _SS = savess;
  96.           _SP = savesp;
  97.      }
  98. }
  99.  
  100. /*Display - put a time on the screen in the position indicated.
  101.             Remember we can't do any DOS calls here.*/
  102. void display (number, y, x)
  103.      int number, x, y;
  104. {
  105.      hour = number / 60;
  106.      minute = _DX;
  107.  
  108.      /*stuff this into an ascii buffer for output*/
  109.      buffer [0] = digits [hour / 10];
  110.      buffer [1] = digits [hour % 10];
  111.      buffer [2] = ':';
  112.      buffer [3] = digits [minute / 10];
  113.      buffer [4] = digits [minute % 10];
  114.      buffer [5] = '\0';
  115.  
  116.      out (buffer, y, x);
  117. }
  118.  
  119. /*Out - out a string onto the screen w/o using system call*/
  120. void out (buffer, y, x)
  121.      char *buffer;
  122.      int y, x;
  123. {
  124.      int far *scrptr;
  125.  
  126.      scrptr = screen + (y * 80 + x);
  127.      while (*buffer)
  128.           *scrptr++ = 0x1700 + *buffer++;
  129. }
  130.  
  131. /*Screenout - examine characters being output using BIOS
  132.               int 0x10.  When a '`' is encountered, clear
  133.               the duration clock; if a '~' is seen, toggle
  134.               the delta display on and off.*/
  135. void interrupt screenout (bp, di, si, ds, es,
  136.                           dx, cx, bx, ax)
  137.      unsigned ax, bx, cx, dx, si, di, bp, ds, es;
  138. {
  139.      /*check for our characters:*/
  140.      if ((temp = (ax & 0xff00)) == 0x0900
  141.                         || temp == 0x0e00) {
  142.           if ((temp = (ax & 0x00ff)) == '`')
  143.                prevtime = -1;
  144.           if (temp == '~')
  145.                dflag = !dflag;
  146.      }
  147.  
  148.      /*pass control onto BIOS routine to get character*/
  149.      _AX = ax;
  150.      _BX = bx;
  151.      _CX = cx;
  152.      _DX = dx;
  153.      (*oldscreen)();
  154.      ax = _AX;
  155.      bx = _BX;
  156.      cx = _CX;
  157.      dx = _DX;
  158. }
  159.  
  160. /*Main - install the above routine.*/
  161. main ()
  162. {
  163.      init ();                          /*set screen pointer*/
  164.      oldclock  = getvect (0x1c);
  165.      oldscreen = getvect (0x10);
  166.      setvect (0x1c, clock);
  167.      setvect (0x10, screenout);
  168.  
  169.      keep (0, 0x1000);
  170. }
  171.  
  172. /*Init - set the screen address and clear the screen*/
  173. void init ()
  174. {
  175.     #define mono (int far *)0xb0000000 /*for mono displays...*/
  176.     #define cga  (int far *)0xb8000000 /*...for ega and cga*/
  177.     int mode;
  178.  
  179.     prevtime = -1; dflag = 1;
  180.  
  181.     regs.h.ah = 0x0f;
  182.     int86 (0x10, ®s, ®s);
  183.     mode = regs.h.al;
  184.     if (regs.h.ah != 80)               /*fixed to 80 columns*/
  185.          abort ();
  186.  
  187.     if (mode == 7)
  188.          screen = mono;
  189.     else
  190.          if (mode == 3 || mode == 2)
  191.               screen = cga;
  192.          else
  193.               abort ();
  194. }
  195.