mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 22:32:19 +02:00
537 lines
14 KiB
ObjectPascal
537 lines
14 KiB
ObjectPascal
{
|
|
This file is part of the Free Pascal run time library.
|
|
Copyright (c) 1999-2000 by Florian Klaempfl
|
|
|
|
This file implements the linux GGI support for the graph unit
|
|
|
|
See the file COPYING.FPC, included in this distribution,
|
|
for details about the copyright.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
**********************************************************************}
|
|
unit GGIGraph;
|
|
interface
|
|
|
|
{ objfpc is needed for array of const support }
|
|
{$mode objfpc}
|
|
|
|
{$i graphh.inc}
|
|
|
|
Const
|
|
{ Supported modes }
|
|
{(sg) GTEXT deactivated because we need mode #0 as default mode}
|
|
{GTEXT = 0; Compatible with VGAlib v1.2 }
|
|
G320x200x16 = 1;
|
|
G640x200x16 = 2;
|
|
G640x350x16 = 3;
|
|
G640x480x16 = 4;
|
|
G320x200x256 = 5;
|
|
G320x240x256 = 6;
|
|
G320x400x256 = 7;
|
|
G360x480x256 = 8;
|
|
G640x480x2 = 9;
|
|
|
|
G640x480x256 = 10;
|
|
G800x600x256 = 11;
|
|
G1024x768x256 = 12;
|
|
|
|
G1280x1024x256 = 13; { Additional modes. }
|
|
|
|
G320x200x32K = 14;
|
|
G320x200x64K = 15;
|
|
G320x200x16M = 16;
|
|
G640x480x32K = 17;
|
|
G640x480x64K = 18;
|
|
G640x480x16M = 19;
|
|
G800x600x32K = 20;
|
|
G800x600x64K = 21;
|
|
G800x600x16M = 22;
|
|
G1024x768x32K = 23;
|
|
G1024x768x64K = 24;
|
|
G1024x768x16M = 25;
|
|
G1280x1024x32K = 26;
|
|
G1280x1024x64K = 27;
|
|
G1280x1024x16M = 28;
|
|
|
|
G800x600x16 = 29;
|
|
G1024x768x16 = 30;
|
|
G1280x1024x16 = 31;
|
|
|
|
G720x348x2 = 32; { Hercules emulation mode }
|
|
|
|
G320x200x16M32 = 33; { 32-bit per pixel modes. }
|
|
G640x480x16M32 = 34;
|
|
G800x600x16M32 = 35;
|
|
G1024x768x16M32 = 36;
|
|
G1280x1024x16M32 = 37;
|
|
|
|
{ additional resolutions }
|
|
G1152x864x16 = 38;
|
|
G1152x864x256 = 39;
|
|
G1152x864x32K = 40;
|
|
G1152x864x64K = 41;
|
|
G1152x864x16M = 42;
|
|
G1152x864x16M32 = 43;
|
|
|
|
G1600x1200x16 = 44;
|
|
G1600x1200x256 = 45;
|
|
G1600x1200x32K = 46;
|
|
G1600x1200x64K = 47;
|
|
G1600x1200x16M = 48;
|
|
G1600x1200x16M32 = 49;
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
termio;
|
|
|
|
var
|
|
OldIO : TermIos;
|
|
Procedure SetRawMode(b:boolean);
|
|
Var
|
|
Tio : Termios;
|
|
Begin
|
|
if b then
|
|
begin
|
|
TCGetAttr(1,Tio);
|
|
OldIO:=Tio;
|
|
CFMakeRaw(Tio);
|
|
end
|
|
else
|
|
Tio:=OldIO;
|
|
TCSetAttr(1,TCSANOW,Tio);
|
|
End;
|
|
|
|
const
|
|
InternalDriverName = 'LinuxGGI';
|
|
|
|
{$i graph.inc}
|
|
|
|
{ ---------------------------------------------------------------------
|
|
GGI bindings [(c) 1999 Sebastian Guenther]
|
|
---------------------------------------------------------------------}
|
|
{$LINKLIB c}
|
|
{$PACKRECORDS C}
|
|
|
|
const
|
|
GLASTMODE = 49;
|
|
ModeNames: array[0..GLastMode] of PChar =
|
|
('[]', {Let GGI choose a default mode}
|
|
'S320x200[GT_4BIT]',
|
|
'S640x200[GT_4BIT]',
|
|
'S640x350[GT_4BIT]',
|
|
'S640x480[GT_4BIT]',
|
|
'S320x200[GT_8BIT]',
|
|
'S320x240[GT_8BIT]',
|
|
'S320x400[GT_8BIT]',
|
|
'S360x480[GT_8BIT]',
|
|
'S640x480x[GT_1BIT]',
|
|
'S640x480[GT_8BIT]',
|
|
'S800x600[GT_8BIT]',
|
|
'S1024x768[GT_8BIT]',
|
|
'S1280x1024[GT_8BIT]',
|
|
'S320x200[GT_15BIT]',
|
|
'S320x200[GT_16BIT]',
|
|
'S320x200[GT_24BIT]',
|
|
'S640x480[GT_15BIT]',
|
|
'S640x480[GT_16BIT]',
|
|
'S640x480[GT_24BIT]',
|
|
'S800x600[GT_15BIT]',
|
|
'S800x600[GT_16BIT]',
|
|
'S800x600[GT_24BIT]',
|
|
'S1024x768[GT_15BIT]',
|
|
'S1024x768[GT_16BIT]',
|
|
'S1024x768[GT_24BIT]',
|
|
'S1280x1024[GT_15BIT]',
|
|
'S1280x1024[GT_16BIT]',
|
|
'S1280x1024[GT_24BIT]',
|
|
'S800x600[GT_4BIT]',
|
|
'S1024x768[GT_4BIT]',
|
|
'S1280x1024[GT_4BIT]',
|
|
'S720x348x[GT_1BIT]',
|
|
'S320x200[GT_32BIT]',
|
|
'S640x480[GT_32BIT]',
|
|
'S800x600[GT_32BIT]',
|
|
'S1024x768[GT_32BIT]',
|
|
'S1280x1024[GT_32BIT]',
|
|
'S1152x864[GT_4BIT]',
|
|
'S1152x864[gt_8BIT]',
|
|
'S1152x864[GT_15BIT]',
|
|
'S1152x864[GT_16BIT]',
|
|
'S1152x864[GT_24BIT]',
|
|
'S1152x864[GT_32BIT]',
|
|
'S1600x1200[GT_4BIT]',
|
|
'S1600x1200[gt_8BIT]',
|
|
'S1600x1200[GT_15BIT]',
|
|
'S1600x1200[GT_16BIT]',
|
|
'S1600x1200[GT_24BIT]',
|
|
'S1600x1200[GT_32BIT]');
|
|
|
|
type
|
|
TGGIVisual = Pointer;
|
|
TGGIResource = Pointer;
|
|
TGGICoord = record
|
|
x, y: SmallInt;
|
|
end;
|
|
TGGIPixel = LongWord;
|
|
PGGIColor = ^TGGIColor;
|
|
TGGIColor = record
|
|
r, g, b, a: Word;
|
|
end;
|
|
PGGIClut = ^TGGIClut;
|
|
TGGIClut = record
|
|
size: SmallInt;
|
|
data: PGGIColor;
|
|
end;
|
|
TGGIGraphType = LongWord;
|
|
TGGIAttr = LongWord;
|
|
TGGIMode = record // requested by user and changed by driver
|
|
Frames: LongInt; // frames needed
|
|
Visible: TGGICoord; // vis. pixels, may change slightly
|
|
Virt: TGGICoord; // virtual pixels, may change
|
|
Size: TGGICoord; // size of visible in mm
|
|
GraphType: TGGIGraphType; // which mode ?
|
|
dpp: TGGICoord; // dots per pixel
|
|
end;
|
|
|
|
const
|
|
libggi = 'ggi';
|
|
function ggiInit: Longint; cdecl; external libggi;
|
|
procedure ggiExit; cdecl; external libggi;
|
|
function ggiOpen(display: PChar; args: Array of const): TGGIVisual; cdecl; external libggi;
|
|
function ggiClose(vis: TGGIVisual): Longint; cdecl; external libggi;
|
|
function ggiParseMode(s: PChar; var m: TGGIMode): Longint; cdecl; external libggi;
|
|
function ggiSetMode(visual: TGGIVisual; var tm: TGGIMode): Longint; cdecl; external libggi;
|
|
function ggiGetMode(visual: TGGIVisual; var tm: TGGIMode): Longint; cdecl; external libggi;
|
|
function ggiCheckMode(visual: TGGIVisual; var tm: TGGIMode): Longint; cdecl; external libggi;
|
|
|
|
function ggiMapColor(vis: TGGIVisual; Color: TGGIColor): TGGIPixel; cdecl; external libggi;
|
|
|
|
function ggiPutPixel(vis: TGGIVisual; x, y: Longint; pixel: TGGIPixel): Longint; cdecl; external libggi;
|
|
function ggiGetPixel(vis: TGGIVisual; x, y: Longint; var pixel: TGGIPixel): Longint; cdecl; external libggi;
|
|
function ggiDrawBox(vis: TGGIVisual; x, y, w, h: Longint): Longint; cdecl; external libggi;
|
|
function ggiPutBox(vis: TGGIVisual; x, y, w, h: Longint; var buffer): Longint; cdecl; external libggi;
|
|
function ggiGetBox(vis: TGGIVisual; x, y, w, h: Longint; var buffer): Longint; cdecl; external libggi;
|
|
|
|
function ggiGetPalette(vis: TGGIVisual; s, len: Longint; var cmap: TGGIColor): Longint; cdecl; external libggi;
|
|
function ggiSetPalette(vis: TGGIVisual; s, len: Longint; var cmap: TGGIColor): Longint; cdecl; external libggi;
|
|
|
|
|
|
var
|
|
Visual: TGGIVisual;
|
|
CurrentMode, OldMode: TGGIMode;
|
|
|
|
|
|
procedure ggi_savevideostate;
|
|
begin
|
|
ggiGetMode(Visual, OldMode);
|
|
end;
|
|
|
|
procedure ggi_restorevideostate;
|
|
begin
|
|
ggiSetMode(Visual, OldMode);
|
|
end;
|
|
|
|
const
|
|
BgiColors: array[0..15] of TGGIColor = (
|
|
(r: $0000; g: $0000; b: $0000; a: 0),
|
|
(r: $0000; g: $0000; b: $8000; a: 0),
|
|
(r: $0000; g: $8000; b: $0000; a: 0),
|
|
(r: $0000; g: $8000; b: $8000; a: 0),
|
|
(r: $8000; g: $0000; b: $0000; a: 0),
|
|
(r: $8000; g: $0000; b: $8000; a: 0),
|
|
(r: $8000; g: $8000; b: $0000; a: 0),
|
|
(r: $C000; g: $C000; b: $C000; a: 0),
|
|
(r: $8000; g: $8000; b: $8000; a: 0),
|
|
(r: $0000; g: $0000; b: $FFFF; a: 0),
|
|
(r: $0000; g: $FFFF; b: $0000; a: 0),
|
|
(r: $0000; g: $FFFF; b: $FFFF; a: 0),
|
|
(r: $FFFF; g: $0000; b: $0000; a: 0),
|
|
(r: $FFFF; g: $0000; b: $FFFF; a: 0),
|
|
(r: $FFFF; g: $FFFF; b: $0000; a: 0),
|
|
(r: $FFFF; g: $FFFF; b: $FFFF; a: 0));
|
|
|
|
procedure ggi_initmodeproc;
|
|
begin
|
|
ggiParseMode(ModeNames[IntCurrentMode], CurrentMode);
|
|
ggiSetMode(Visual, CurrentMode);
|
|
end;
|
|
|
|
function ClipCoords(var x, y: SmallInt): Boolean;
|
|
{ Adapt to viewport, return TRUE if still in viewport,
|
|
false if outside viewport}
|
|
begin
|
|
x := x + StartXViewPort;
|
|
x := y + StartYViewPort;
|
|
ClipCoords := not ClipPixels;
|
|
if ClipCoords then begin
|
|
ClipCoords := (y < StartXViewPort) or (x > (StartXViewPort + ViewWidth));
|
|
ClipCoords := ClipCoords or
|
|
((y < StartYViewPort) or (y > (StartYViewPort + ViewHeight)));
|
|
ClipCoords := not ClipCoords;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure ggi_directpixelproc(X, Y: smallint);
|
|
var
|
|
Color, CurCol: TGGIPixel;
|
|
begin
|
|
CurCol := ggiMapColor(Visual, BgiColors[CurrentColor]);
|
|
case CurrentWriteMode of
|
|
XORPut: begin
|
|
{ getpixel wants local/relative coordinates }
|
|
ggiGetPixel(Visual, x-StartXViewPort, y-StartYViewPort, Color);
|
|
Color := CurCol xor Color;
|
|
end;
|
|
OrPut: begin
|
|
{ getpixel wants local/relative coordinates }
|
|
ggiGetPixel(Visual, x-StartXViewPort, y-StartYViewPort, Color);
|
|
Color := CurCol or Color;
|
|
end;
|
|
AndPut: begin
|
|
{ getpixel wants local/relative coordinates }
|
|
ggiGetPixel(Visual, x-StartXViewPort, y-StartYViewPort, Color);
|
|
Color := CurCol and Color;
|
|
end;
|
|
NotPut:
|
|
Color := not Color;
|
|
else
|
|
Color := CurCol;
|
|
end;
|
|
ggiPutPixel(Visual, x, y, Color);
|
|
end;
|
|
|
|
procedure ggi_putpixelproc(X,Y: smallint; Color: Word);
|
|
begin
|
|
If Not ClipCoords(X,Y) Then exit;
|
|
ggiputpixel(Visual,x, y, Color);
|
|
end;
|
|
|
|
function ggi_getpixelproc (X,Y: smallint): word;
|
|
|
|
Var i : TGGIPixel;
|
|
|
|
begin
|
|
ClipCoords(X,Y);
|
|
ggigetpixel(Visual,x, y,I);
|
|
ggi_getpixelproc:=i;
|
|
end;
|
|
|
|
procedure ggi_clrviewproc;
|
|
begin
|
|
ggidrawbox(Visual,StartXViewPort,StartYViewPort,ViewWidth,ViewHeight);
|
|
{ reset coordinates }
|
|
CurrentX := 0;
|
|
CurrentY := 0;
|
|
end;
|
|
|
|
{ Bitmap utilities }
|
|
type
|
|
PBitmap = ^TBitmap;
|
|
TBitmap = record
|
|
Width, Height: longint;
|
|
reserved : longint;
|
|
Data: record end;
|
|
end;
|
|
|
|
procedure ggi_putimageproc (X,Y: smallint; var Bitmap; BitBlt: Word);
|
|
begin
|
|
With TBitMap(BitMap) do
|
|
ggiputbox(Visual,x, y, width, height, Data);
|
|
end;
|
|
|
|
procedure ggi_getimageproc (X1,Y1,X2,Y2: smallint; Var Bitmap);
|
|
begin
|
|
with TBitmap(Bitmap) do
|
|
begin
|
|
Width := x2 - x1 + 1;
|
|
Height := y2 - y1 + 1;
|
|
ggigetbox(Visual,x1,y1, x2 - x1 + 1, y2 - y1 + 1, Data);
|
|
end;
|
|
end;
|
|
|
|
function ggi_imagesizeproc (X1,Y1,X2,Y2: smallint): longint;
|
|
begin
|
|
// 32 bits per pixel -- change ASAP !!
|
|
ggi_imagesizeproc := SizeOf(TBitmap) + (x2 - x1 + 1) * (y2 - y1 + 1) * SizeOF(longint);
|
|
end;
|
|
|
|
procedure ggi_hlineproc (x, x2,y : smallint);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_vlineproc (x,y,y2: smallint);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_patternlineproc (x1,x2,y: smallint);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_ellipseproc (X,Y: smallint;XRadius: word;
|
|
YRadius:word; stAngle,EndAngle: word; fp: PatternLineProc);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_lineproc (X1, Y1, X2, Y2 : smallint);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_getscanlineproc (X1, X2, Y : smallint; var data);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_setactivepageproc (page: word);
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_setvisualpageproc (page: word);
|
|
begin
|
|
end;
|
|
|
|
|
|
procedure ggi_savestateproc;
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_restorestateproc;
|
|
begin
|
|
end;
|
|
|
|
procedure ggi_setrgbpaletteproc(ColorNum, RedValue, GreenValue, BlueValue: smallint);
|
|
|
|
Var Col : TGGIcolor;
|
|
|
|
begin
|
|
col.r:=redvalue;
|
|
col.g:=greenvalue;
|
|
col.b:=bluevalue;
|
|
ggisetpalette(Visual,ColorNum,1,col);
|
|
end;
|
|
|
|
procedure ggi_getrgbpaletteproc (ColorNum: smallint;
|
|
var RedValue, GreenValue, BlueValue: smallint);
|
|
|
|
Var Col : TGGIColor;
|
|
|
|
begin
|
|
ggigetpalette(Visual,ColorNum,1,col);
|
|
RedValue:=Col.R;
|
|
GreenValue:=Col.G;
|
|
BlueValue:=Col.B;
|
|
end;
|
|
|
|
{************************************************************************}
|
|
{* General routines *}
|
|
{************************************************************************}
|
|
|
|
procedure CloseGraph;
|
|
begin
|
|
if not IsGraphMode then
|
|
begin
|
|
_graphresult := grnoinitgraph;
|
|
exit
|
|
end;
|
|
RestoreVideoState;
|
|
isgraphmode := false;
|
|
end;
|
|
|
|
function QueryAdapterInfo:PModeInfo;
|
|
{ This routine returns the head pointer to the list }
|
|
{ of supported graphics modes. }
|
|
{ Returns nil if no graphics mode supported. }
|
|
{ This list is READ ONLY! }
|
|
|
|
var
|
|
ModeInfo: TGGIMode;
|
|
|
|
procedure AddGGIMode(i: smallint); // i is the mode number
|
|
var
|
|
mode: TModeInfo;
|
|
begin
|
|
InitMode(Mode);
|
|
with Mode do begin
|
|
ModeNumber := i;
|
|
ModeName := ModeNames[i];
|
|
// Pretend we're VGA always.
|
|
DriverNumber := VGA;
|
|
MaxX := ModeInfo.Visible.X-1;
|
|
MaxY := ModeInfo.Visible.Y-1;
|
|
MaxColor := 1 shl (ModeInfo.graphtype and $ff);
|
|
//MaxColor := 255;
|
|
PaletteSize := MaxColor;
|
|
HardwarePages := 0;
|
|
// necessary hooks ...
|
|
DirectPutPixel := @ggi_DirectPixelProc;
|
|
GetPixel := @ggi_GetPixelProc;
|
|
PutPixel := @ggi_PutPixelProc;
|
|
SetRGBPalette := @ggi_SetRGBPaletteProc;
|
|
GetRGBPalette := @ggi_GetRGBPaletteProc;
|
|
ClearViewPort := @ggi_ClrViewProc;
|
|
PutImage := @ggi_PutImageProc;
|
|
GetImage := @ggi_GetImageProc;
|
|
ImageSize := @ggi_ImageSizeProc;
|
|
{ Add later maybe ?
|
|
SetVisualPage := SetVisualPageProc;
|
|
SetActivePage := SetActivePageProc;
|
|
GetScanLine := @ggi_GetScanLineProc;
|
|
Line := @ggi_LineProc;
|
|
InternalEllipse:= @ggi_EllipseProc;
|
|
PatternLine := @ggi_PatternLineProc;
|
|
HLine := @ggi_HLineProc;
|
|
VLine := @ggi_VLineProc;
|
|
}
|
|
InitMode := @ggi_InitModeProc;
|
|
end;
|
|
AddMode(Mode);
|
|
end;
|
|
|
|
var
|
|
i: longint;
|
|
OldMode: TGGIMode;
|
|
begin
|
|
QueryAdapterInfo := ModeList;
|
|
{ If the mode listing already exists... }
|
|
{ simply return it, without changing }
|
|
{ anything... }
|
|
if Assigned(ModeList) then
|
|
exit;
|
|
SaveVideoState:=@ggi_savevideostate;
|
|
RestoreVideoState:=@ggi_restorevideostate;
|
|
|
|
If ggiInit <> 0 then begin
|
|
_graphresult := grNoInitGraph;
|
|
exit;
|
|
end;
|
|
|
|
Visual := ggiOpen(nil, []); // Use default visual
|
|
|
|
ggiGetMode(Visual, OldMode);
|
|
ggiParseMode('', ModeInfo);
|
|
ggiSetMode(Visual, ModeInfo);
|
|
ggiGetMode(Visual, ModeInfo);
|
|
ggiSetMode(Visual, OldMode);
|
|
AddGGIMode(0);
|
|
|
|
for i := 1 to GLastMode do begin
|
|
// WriteLn('Testing mode: ', ModeNames[i]);
|
|
ggiParseMode(ModeNames[i], ModeInfo);
|
|
If ggiCheckMode(visual, ModeInfo) = 0 then begin
|
|
Writeln('OK for mode ',i,' : ', ModeNames[i]);
|
|
AddGGIMode(i);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
initialization
|
|
InitializeGraph;
|
|
SetRawMode(True);
|
|
finalization
|
|
SetRawMode(False);
|
|
end.
|