home *** CD-ROM | disk | FTP | other *** search
- /*Program 8_3c - 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 the 8086 block
- move instruction this encased in seperate assembler subroutines
- linked in as .OBJ files. The assembler name is Prog9d1.asm and
- the project file name is Prog9d.prj. This must be compiled under
- the Small or Tiny memory models (otherwise, the stack offsets in
- the assembler routine must be changed).
- */
-
- #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 scrollup 0x06
- #define setcursor 0x02
- #define writetele 0x0e
- #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);
-
- /*prototype declarations for external assembler routines*/
- /*(remember that MASM generates uppercase symbols)*/
- void MOVS (unsigned far *, unsigned far *, unsigned);
- /*move from first pointer to second, third arg words*/
- void STOS (unsigned far *, unsigned);
- /*clear at pointer, unsigned words*/
-
- /*be sure compile model is correct*/
- #if sizeof(MOVS)-2
- #error Must compile under Small or Tiny models
- #endif
-
- /*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;
-
- 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;
- MOVS (source, dest, number);
-
- /*now blank the lines abandoned*/
- dest = screen + number;
- number = nlines * screenwidth;
- STOS (dest, number);
-
- 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)
- *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);
- }