home *** CD-ROM | disk | FTP | other *** search
- (***********************************************************)
- (* *)
- (* TURBO GRAPHIX version 1.06A *)
- (* *)
- (* Graphics module for IBM Color/Graphics Adapter *)
- (* Module version 1.06A *)
- (* *)
- (* Copyright (C) 1985 by *)
- (* BORLAND International *)
- (* *)
- (***********************************************************)
-
- const
- XMaxGlb = 79; { Number of bytes -1 in one screen line }
- XScreenMaxGlb = 639; { Number of pixels -1 in one screen line }
- YMaxGlb = 199; { Number of lines -1 on the screen }
- ScreenSizeGlb = 8191; { Total size in integers of the screen }
- GrafMode = $0006; { BIOS Interrupt 10 AX register }
- { high byte specifies function 0 }
- { low byte selects graphics mode }
- AspectFactor = 0.44; { Aspect ratio for a true circle }
- IVStepGlb = 2; { Initial value of VStepGlb }
-
- HardwareGrafBase : integer = $B800; { Location of the hardware screen }
- FontLoaded : boolean = false; { Flag: has the font been loaded yet? }
- MinForeground : integer = 0; { Lowest allowable foreground color }
- MaxForeground : integer = 15; { Highest allowable foreground color }
- MinBackground : integer = 0; { Lowest allowable background color }
- MaxBackground : integer = 0; { Highest allowable background color }
- SaveStateGlb : integer = 10;
- ForegroundColorGlb : integer = 15;
-
- type
- ScreenType = array[0..ScreenSizeGlb] of integer;
- ScreenPointer = ^ScreenType;
- FontChar = array[0..7] of byte;
- GrfFont = array[0..255] of FontChar;
- WindowStackRecord = record
- W : WindowType;
- Contents : ScreenPointer;
- end;
- Stacks = array[1..MaxWindowsGlb] of WindowStackRecord;
-
- var
- FontHeightGlb : byte;
- LineDistGlb : integer;
- ScreenGlb : ScreenPointer;
- ConOutPtrSave : integer;
- Font : GrfFont;
- Stack : Stacks;
- DisplayType : (IBMPCjr, IBMCGA, IBMEGA, NoDisplay);
- RowBase : array[0..YMaxGlb] of integer;
-
- function BaseAddress(Y : integer) : integer;
- { Calculate address of scanline Y }
- begin
- BaseAddress:=(Y and 1) shl 13 + (Y and -2) shl 5 + (Y and -2) shl 3;
- end; { BaseAddress }
-
- procedure Error(ErrProc, ErrCode : integer); forward; { Code in KERNEL.SYS }
-
- function HardwarePresent : boolean;
- { Test for the presence of a graphics card }
- var
- I, EquipFlag : integer;
- Info, EGASwitch : byte;
- HP : boolean;
- Regs : record case integer of
- 1 : (AX,BX,CX,DX,BP,SI,DI,DS,ES,Flags : integer);
- 2 : (AL,AH,BL,BH,CL,CH,DL,DH : byte);
- end;
-
- begin
- HP := false;
- DisplayType := NoDisplay;
- with Regs do
- begin
- Intr($11, Regs);
- EquipFlag := AX;
- AH := $12;
- BL := $10;
- Intr($10, Regs);
- EGASwitch := CL;
- Info := BH;
- end;
- if Mem[$F000:$FFFE] = $FD then { PCjr }
- begin
- MinForeground := 0; { Actually only 0 and 15 are valid }
- MaxForeground := 15;
- MinBackground := 0;
- MaxBackground := 15;
- DisplayType := IBMPCjr;
- HP := true;
- end
- else if ((EquipFlag and 52) in [0,16,32]) and (Info = 0) then
- begin { EGA present, active, and in color }
- MinForeground := 0;
- MaxForeground := 15;
- MinBackground := 0;
- MaxBackground := 15;
- DisplayType := IBMEGA;
- HP := true;
- end;
- if not HP then
- if ((EquipFlag and 48) in [16,32]) or { CGA }
- (((EquipFlag and 52) = 4) and { EGA but not active }
- (EGASwitch in [4,5,10,11])) then { EGA is mono }
- begin
- MinForeground := 0;
- MaxForeground := 15;
- MinBackground := 0;
- MaxBackground := 0;
- DisplayType := IBMCGA;
- HP := true;
- end;
- HardwarePresent := HP;
- end; { HardwarePresent }
-
- procedure AllocateRAMScreen;
- { Allocates the RAM screen and makes sure that
- ScreenGlb is on a segment (16 byte) boundary }
- var
- Test : ^integer;
- begin
- New(ScreenGlb);
- while Ofs(ScreenGlb^) <> 0 do { Make absolutely certain that ScreenGlb }
- begin { is on a segment (16 byte) boundary! }
- Dispose(ScreenGlb);
- New(Test);
- New(ScreenGlb);
- end;
- end; { AllocateRAMScreen }
-
- procedure LeaveGraphic;
- { Exit from graphics mode and clear the screen }
- var
- Regs : record case integer of
- 1 : (AX,BX,CX,DX,BP,SI,DI,DS,ES,Flags : integer);
- 2 : (AL,AH,BL,BH,CL,CH,DL,DH : byte);
- end;
- begin
- Regs.AX := SaveStateGlb;
- Intr($10, Regs);
- if GrafModeGlb then
- ConOutPtr := ConOutPtrSave;
- GrafModeGlb := false;
- end; { LeaveGraphic }
-
- procedure DC(C: byte);
- { Draw the character C at the position XTextGlb, YTextGlb }
- begin inline(
- $A1/ YTextGlb { MOV AX, YTextGlb }
- /$48 { DEC AX }
- /$8B/$3E/LineDistGlb { MOV DI, LineDistGlb }
- /$F7/$E7 { MUL DI }
- /$D1/$E0 { SHL AX, 1 }
- /$97 { XCHG AX, DI }
-
- /$8A/$9E/ C { MOV BL, C }
- /$B7/$00 { MOV BH, 00 }
- /$D1/$E3 { SHL BX,1 }
- /$D1/$E3 { SHL BX,1 }
- /$D1/$E3 { SHL BX,1 }
- /$81/$C3/Font { ADD BX,Font }
- /$8A/$16/XTextGlb { MOV DL,XTextGlb }
- /$FE/$CA { DEC DL }
- /$B6/$00 { MOV DH,00 }
- /$A1/GrafBase { MOV AX,GrafBase }
- /$8E/$C0 { MOV ES,AX }
- /$8A/$2E/FontHeightGlb { MOV CH,FontHeightGlb }
-
- /$8B/$B5/RowBase { NextRow:MOV SI, [DI+RowBase] }
- /$01/$D6 { ADD SI,DX }
- /$8A/$07 { MOV AL,[BX] }
- /$26 { ES: }
- /$88/$04 { MOV [SI],AL }
- /$43 { INC BX }
- /$47 { INC DI }
- /$47 { INC DI }
- /$FE/$CD { DEC CH }
- /$75/$EE); { JNZ NextRow }
- end; { DC }
-
- procedure DisplayChar(C : byte);
- { Same as DC. Intended for internal use by the graphics system }
- begin
- if C = 8 then
- begin
- if XTextGlb > 1 then
- XTextGlb := XTextGlb - 1;
- end
- else
- if C = 10 then
- begin
- if YTextGlb < 25 then
- YTextGlb := YTextGlb + 1;
- end
- else
- if C = 13 then
- XTextGlb := 1
- else
- begin
- DC(C);
- if XTextGlb < 80 then
- XTextGlb := XTextGlb + 1;
- end;
- end; { DisplayChar }
-
- procedure SetIBMPalette(PaletteNumber, Color : integer);
- { Set up the palette registers on the IBM CGA }
- var
- Regs : record case integer of
- 1 : (AX,BX,CX,DX,BP,SI,DI,DS,ES,Flags : integer);
- 2 : (Al,AH,BL,BH,CL,CH,DL,DH : byte);
- end;
- begin
- with Regs do
- begin
- if PaletteNumber <> 2 then
- begin
- AH := $0B;
- BL := Color;
- BH := PaletteNumber;
- end
- else
- begin
- AX := $1000;
- BL := 1;
- BH := Color;
- end;
- Intr($10, Regs);
- end;
- end; { SetIBMPalette }
-
- procedure SetForegroundColor(Color : integer);
- { Set the foreground color }
- begin
- case DisplayType of
- IBMPCjr : SetIBMPalette(1, 1 - (Color and 1));
- IBMCGA : SetIBMPalette(0, Color);
- IBMEGA : SetIBMPalette(2, Color);
- end;
- ForegroundColorGlb := Color;
- end; { SetForegroundColor }
-
- procedure SetBackgroundColor(Color : integer);
- { Set the background color }
- begin
- case DisplayType of
- IBMPCjr,
- IBMEGA : SetIBMPalette(0, Color);
- end;
- if DisplayType = IBMEGA then
- SetIBMPalette(2, ForegroundColorGlb);
- end; { SetBackgroundColor }
-
- procedure ClearScreen;
- { Clear the displayed screen }
- begin
- FillChar(Mem[GrafBase:0000], ScreenSizeGlb shl 1, 0);
- end; { ClearScreen }
-
- procedure EnterGraphic;
- { Enter graphics mode }
- type
- Reg = record case integer of
- 1 : (AX,BX,CX,DX,BP,SI,DI,DS,ES,Flags : integer);
- 2 : (AL,AH,BL,BH,CL,CH,DL,DH : byte);
- end;
- var
- Regs : Reg;
- FontFile : file of GrfFont;
- I : integer;
- begin
- if not FontLoaded then
- begin
- for I := 0 to YMaxGlb do
- RowBase[I] := BaseAddress(I);
- Assign(FontFile, '8x8.FON');
- {$I-} Reset(FontFile); {$I+}
- if IOresult = 0 then
- begin
- Read(FontFile, Font);
- Close(FontFile);
- lineDistGlb := 8;
- FontHeightGlb := 8;
- end
- else
- FillChar(Font, SizeOf(Font), 0);
- FontLoaded := true;
- end;
- Regs.AX := $0F00;
- Intr($10, Regs);
- if (Regs.AL < 4) or (SaveStateGlb = 10) then
- SaveStateGlb := Regs.AL;
- Regs.AX := GrafMode;
- Intr($10, Regs);
- SetForegroundColor(MaxForeground);
- if not GrafModeGlb then
- ConOutPtrSave := ConOutPtr;
- ConOutPtr := Ofs(DisplayChar);
- GrafModeGlb := true;
- end; { EnterGraphic }
-
- var
- LastAddress : integer;
-
- procedure DP(X, Y : integer);
- { Plot a pixel at (X, Y) }
- begin
- inline(
- $8B/$46/ <Y { MOV AX, Y }
- /$D1/$E0 { SHL AX, 1 }
- /$97 { XCHG AX, DI }
- /$8B/$85/RowBase { MOV AX, [DI+RowBase] }
-
- { AX has RowBase(Y) }
-
- /$8B/$5E/ <X { MOV BX, X }
- /$89/$DA { MOV DX,BX Save X in DX }
- /$B1/$03 { MOV CL,03 }
- /$D3/$EB { SHR BX,CL BX = X div 8 }
- /$01/$C3 { ADD BX,AX }
-
- { AX has RowBase(Y) + X div 8 }
-
- /$88/$D1 { MOV CL,DL }
- /$80/$E1/$07 { AND CL,07 }
- /$B2/$80 { MOV DL,80 }
- /$D2/$EA { SHR DL,CL }
- /$8E/$06/GrafBase { MOV ES,GrafBase }
- /$80/$3E/ColorGlb/$FF { JMP ColorGlb,FF }
- /$75/$05 { JNZ ZeroPix }
-
- /$26 { ES: }
- /$08/$17 { OR [BX],DL }
- /$EB/$05 { JMP 3420 }
- { ZeroPix: }
- /$F6/$D2 { NOT DL }
- /$26 { ES: }
- /$20/$17 { AND [BX],DL }
-
- /$89/$1E/LastAddress); { Done: MOV LastAddress,BX }
- end; { DP }
-
- function PD(X, Y : integer) : boolean;
- { Return true if the color of the pixel at (X, Y) matches ColorGlb }
- begin
- PD := (ColorGlb = 0) xor (Mem[GrafBase:BaseAddress(Y) + X shr 3]
- and (128 shr (X and 7)) <> 0);
- end; { PD }
-
- procedure SetBackground8(Background : BackgroundArray);
- { Fills the active display with the specified bit pattern }
- var
- I : integer;
- begin
- for I := Y1RefGlb to Y2RefGlb do
- FillChar(Mem[GrafBase:BaseAddress(I) + X1RefGlb], X2RefGlb - X1RefGlb + 1,
- Background[I and 7]);
- end; { SetBackground8 }
-
- procedure SetBackground(Byt : byte);
- { Determines the background pattern of the active window }
- var
- Bk : BackgroundArray;
- begin
- FillChar(Bk, 8, Byt);
- SetBackground8(Bk);
- end; { SetBackground }
-
- procedure DrawStraight(X1, X2, Y : integer);
- { Draw a horizontal line from X1,Y to X2,Y }
- var
- I, X : integer;
- DirectModeLoc : boolean;
- begin
- if (not ((X1 < 0) or (X1 > XMaxGlb shl 3 + 7)) and not ((X2 < 0) or
- (X2 > XMaxGlb shl 3 + 7)) and ((Y >= 0) and (Y <= YMaxGlb))) then
- begin
- DirectModeLoc := DirectModeGlb;
- DirectModeGlb := true;
- if X1 > X2 then
- begin
- X := X1;
- X1 := X2;
- X2 := X;
- end;
- if X2 - X1 < 16 then
- for X := X1 to X2 do
- DP(X, Y)
- else
- begin
- X1 := X1 + 8;
- for I := (X1 - 8) to (X1 and -8) do
- DP(I, Y);
- for I := (X2 and -8) to X2 do
- DP(I, Y);
- FillChar(Mem[GrafBase:BaseAddress(Y) + (X1 shr 3)],
- (X2 shr 3) - (X1 shr 3), ColorGlb);
- end;
- DirectModeGlb := DirectModeLoc;
- end;
- end; { DrawStraight }
-
- procedure SaveScreen(FileName : WrkString);
- { Save the current screen on disk using FileName }
- type
- PicFile = file of ScreenType;
- var
- Picture : ScreenPointer;
- PictureFile : PicFile;
- IOErr : boolean;
-
- procedure IOCheck;
- begin
- IOErr := IOresult <> 0;
- if IOErr then
- Error(27, 5);
- end; { IOCheck }
-
- begin
- IOErr := false;
- Picture := Ptr(GrafBase, 0);
- Assign(PictureFile, FileName);
- {$I-} Rewrite(PictureFile); {$I+}
- IOCheck;
- if not IOErr then
- begin
- {$I-} Write(PictureFile, Picture^); {$I+}
- IOCheck;
- end;
- if not IOErr then
- begin
- {$I-} Close(PictureFile); {$I+}
- IOCheck;
- end;
- end; { SaveScreen }
-
- procedure LoadScreen(FileName : WrkString);
- { Load screen from file FileName }
- type
- PicFile = file of ScreenType;
- var
- Picture : ScreenPointer;
- PictureFile : PicFile;
- begin
- Picture := Ptr(GrafBase, 0);
- Assign(PictureFile, FileName);
- {$I-} Reset(PictureFile); {$I+}
- if IOresult <> 0 then
- Error(11, 5)
- else
- begin
- Read(PictureFile, Picture^);
- Close(PictureFile);
- end;
- end; { LoadScreen }
-
- procedure SwapScreen;
- { Exchanges the contents the of the displayed
- screen with the contents of the RAM screen }
- var
- ScanLine,
- LineBase,
- RamScrSeg : integer;
- LineBuffer : array[0..XMaxGlb] of byte;
- begin
- if RamScreenGlb then
- begin
- RamScrSeg := Seg(ScreenGlb^);
- for ScanLine := 0 to YMaxGlb do
- begin
- LineBase := RowBase[Scanline];
- Move(Mem[GrafBase:LineBase], LineBuffer, XMaxGlb + 1);
- Move(Mem[RamScrSeg:LineBase], Mem[GrafBase:LineBase], XMaxGlb + 1);
- Move(LineBuffer, Mem[RamScrSeg:LineBase], XMaxGlb + 1);
- end;
- end;
- end; { SwapScreen }
-
- procedure CopyScreen;
- { Copies the active screen onto the inactive screen }
- var
- ToBase : integer;
- begin
- if RamScreenGlb then
- begin
- if GrafBase = HardwareGrafBase then
- ToBase := Seg(ScreenGlb^)
- else
- ToBase := HardwareGrafBase;
- Move(Mem[GrafBase:0000], Mem[ToBase:0000], ScreenSizeGlb shl 1);
- end;
- end; { CopyScreen }
-
- procedure InvertScreen;
- { Inverts the image on the active screen }
- begin
- inline(
- $1E { PUSH DS }
- /$A1/ GrafBase { MOV AX,[0382] }
- /$8E/$D8 { MOV DS,AX }
- /$B9/ ScreenSizeGlb { MOV CX,4000 }
- /$31/$DB { XOR BX,BX }
-
- /$F7/$17 { Label: NOT WORD PTR [BX] }
- /$43 { INC BX }
- /$43 { INC BX }
- /$E2/$FA { LOOP Label }
- /$1F); { POP DS }
- end; { InvertScreen }