home *** CD-ROM | disk | FTP | other *** search
- /*Prg7_4a - High Speed Screen Output
- by Stephen R Davis, 1987
-
- Perform direct screen output by accessing screen memory directly
- via the screen pointer 'screen'. Scroll using standard C
- statements.
- */
-
- #include <stdio.h>
- #include <dos.h>
- #include <stdlib.h>
-
- #define cga (unsigned far *)0xb8000000 /*same for ega*/
- #define mono (unsigned far *)0xb0000000
- #define space 0x20
- #define attrib 0x07
- #define screenheight 25
-
- /*add the screen BIOS functions*/
- #define setcursor 0x02
- #define getmode 0x0f
-
- /*define global variables*/
- unsigned v_pos, h_pos, screenwidth;
- union REGS regs;
- unsigned far *screen; /*screen pointer*/
-
- /*prototype declarations*/
- void init (void);
- void scroll (unsigned);
- void qprintf (char *);
- void pcursor (unsigned, unsigned);
-
- /*Main - test the output routines*/
- int main ()
- {
- int i, j;
-
- init ();
- for (i = 0; i < 20; i++) {
- for (j = 0; j < screenheight; j++) {
- qprintf ("this is BIOS output");
- pcursor(v_pos, 30+j);
- qprintf ("and this\n");
- }
- for (j = 0; j < screenheight; j++)
- printf ("this is normal printf output\n");
- }
- }
-
- /*Init - set the screen address and clear the screen*/
- void init ()
- {
- short mode;
-
- regs.h.ah = getmode;
- int86 (0x10, ®s, ®s);
- mode = regs.h.al;
- screenwidth = regs.h.ah;
-
- if (mode == 7)
- screen = mono;
- else
- if (mode == 3 || mode == 2)
- screen = cga;
- else
- abort ();
-
- scroll (screenheight);
- pcursor (0, 0);
- }
-
- /*Scroll - scroll up N lines using function 6*/
- void scroll (nlines)
- unsigned nlines;
- {
- unsigned far *source, far *dest, number, temp;
-
- if (nlines >= screenheight)
- nlines = screenheight;
-
- h_pos = 0;
- if ((v_pos += nlines) >= screenheight) {
- nlines = (v_pos - screenheight) + 1;
-
- /*scroll the screen up 'nlines' amount*/
- source = screen + (nlines * screenwidth);
- dest = screen;
- number = (screenheight - nlines) * screenwidth;
- for (temp = number; temp; temp--)
- *dest++ = *source++;
-
- /*now blank the lines abandoned*/
- dest = screen + number;
- number = nlines * screenwidth;
- for (; number; number--)
- *dest++ = (attrib << 8) + space;
-
- v_pos = screenheight - 1;
- }
- }
-
- /*Qprintf - output a string using the BIOS screen handler. If
- an attribute is not provided, use the default.*/
-
- #define SCREENLOC screen + ((screenwidth * v_pos) + h_pos)
-
- void qprintf (c)
- char *c;
- {
- unsigned far *sp;
-
- sp = SCREENLOC;
- for (; *c; c++)
- if (*c == '\n') {
- scroll (1);
- sp = SCREENLOC;}
- else
- if (h_pos < screenwidth) {
- h_pos++;
- *sp++ = (attrib << 8) + *c;
- }
- pcursor (v_pos, h_pos);
- }
-
- /*PCursor - place the cursor at the current x and y location.
- To place the cursor, and subsequent output, to any
- arbitrary location, set 'v_pos' and 'h_pos' before
- calling pcursor.*/
- void pcursor (y, x)
- unsigned x, y;
- {
- v_pos = y;
- h_pos = x;
-
- regs.h.ah = setcursor;
- regs.h.bh = 0;
- regs.h.dh = v_pos;
- regs.h.dl = h_pos;
- int86 (0x10, ®s, ®s);
- }