123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by Florian Klaempfl
- This file implements the go32v2 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 Graph;
- interface
- { the code of the unit fits in 64kb in the medium memory model, but exceeds 64kb
- in the large and huge memory models, so enable huge code in these models. }
- {$if defined(FPC_MM_LARGE) or defined(FPC_MM_HUGE)}
- {$hugecode on}
- {$endif}
- {$define asmgraph}
- {$i graphh.inc}
- {$i vesah.inc}
- CONST
- m640x200x16 = VGALo;
- m640x400x16 = VGAMed;
- m640x480x16 = VGAHi;
- { VESA Specific video modes. }
- m320x200x32k = $10D;
- m320x200x64k = $10E;
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m320x200x16m = $10F;
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m640x400x256 = $100;
- m640x480x256 = $101;
- m640x480x32k = $110;
- m640x480x64k = $111;
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m640x480x16m = $112;
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m800x600x16 = $102;
- m800x600x256 = $103;
- m800x600x32k = $113;
- m800x600x64k = $114;
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m800x600x16m = $115;
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m1024x768x16 = $104;
- m1024x768x256 = $105;
- m1024x768x32k = $116;
- m1024x768x64k = $117;
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m1024x768x16m = $118;
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m1280x1024x16 = $106;
- m1280x1024x256 = $107;
- m1280x1024x32k = $119;
- m1280x1024x64k = $11A;
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- m1280x1024x16m = $11B;
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- { Helpful variable to get save/restore support in IDE PM }
- const
- DontClearGraphMemory : boolean = false;
- implementation
- uses
- dos,ports;
- const
- InternalDriverName = 'DOSGX';
- {$i graph.inc}
- const
- VideoOfs : word = 0; { Segment to draw to }
- FirstPlane = $0102; (* 02 = Index to Color plane Select, *)
- (* 01 = Enable color plane 1 *)
- { ; ===== VGA Register Values ===== }
- SCREEN_WIDTH = 80 ; { MODE-X 320 SCREEN WIDTH }
- { CHANGE THE VALUE IF OTHER MODES }
- { OTHER THEN 320 ARE USED. }
- ATTRIB_Ctrl = $03C0 ; { VGA Attribute Controller }
- GC_Index = $03CE ; { VGA Graphics Controller }
- SC_Index = $03C4 ; { VGA Sequencer Controller }
- SC_Data = $03C5 ; { VGA Sequencer Data Port }
- CRTC_Index = $03D4 ; { VGA CRT Controller }
- CRTC_Data = $03D5 ; { VGA CRT Controller Data }
- MISC_OUTPUT = $03C2 ; { VGA Misc Register }
- INPUT_1 = $03DA ; { Input Status #1 Register }
- DAC_WRITE_ADDR = $03C8 ; { VGA DAC Write Addr Register }
- DAC_READ_ADDR = $03C7 ; { VGA DAC Read Addr Register }
- PEL_DATA_REG = $03C9 ; { VGA DAC/PEL data Register R/W }
- PIXEL_PAN_REG = $033 ; { Attrib Index: Pixel Pan Reg }
- MAP_MASK = $002 ; { S= $Index: Write Map Mask reg }
- READ_MAP = $004 ; { GC Index: Read Map Register }
- START_DISP_HI = $00C ; { CRTC Index: Display Start Hi }
- START_DISP_LO = $00D ; { CRTC Index: Display Start Lo }
- MAP_MASK_PLANE1 = $00102 ; { Map Register + Plane 1 }
- MAP_MASK_PLANE2 = $01102 ; { Map Register + Plane 1 }
- ALL_PLANES_ON = $00F02 ; { Map Register + All Bit Planes }
- CHAIN4_OFF = $00604 ; { Chain 4 mode Off }
- ASYNC_RESET = $00100 ; { (A)synchronous Reset }
- SEQU_RESTART = $00300 ; { Sequencer Restart }
- LATCHES_ON = $00008 ; { Bit Mask + Data from Latches }
- LATCHES_OFF = $0FF08 ; { Bit Mask + Data from CPU }
- VERT_RETRACE = $08 ; { INPUT_1: Vertical Retrace Bit }
- PLANE_BITS = $03 ; { Bits 0-1 of Xpos = Plane # }
- ALL_PLANES = $0F ; { All Bit Planes Selected }
- CHAR_BITS = $0F ; { Bits 0-3 of Character Data }
- GET_CHAR_PTR = $01130 ; { VGA BIOS Func: Get Char Set }
- ROM_8x8_Lo = $03 ; { ROM 8x8 Char Set Lo Pointer }
- ROM_8x8_Hi = $04 ; { ROM 8x8 Char Set Hi Pointer }
- { Constants Specific for these routines }
- NUM_MODES = $8 ; { # of Mode X Variations }
- { in 16 color modes, the actual colors used are not 0..15, but: }
- ToRealCols16: Array[0..15] of word =
- (0,1,2,3,4,5,20,7,56,57,58,59,60,61,62,63);
- var
- ScrWidth : word absolute $40:$4a;
- inWindows: boolean;
- {$ifndef tp}
- Procedure seg_bytemove(sseg : word;source : word;dseg : word;dest : word;count : word); assembler;
- asm
- push ds
- cld
- mov es, dseg
- mov si, source
- mov di, dest
- mov cx, count
- mov ds,sseg
- rep movsb
- pop ds
- end;
- {$endif tp}
- Procedure CallInt10(val_ax : word); assembler;
- asm
- mov ax,val_ax
- push ds
- push bp
- int 10h
- pop bp
- pop ds
- end;
- Procedure InitInt10hMode(mode : byte);
- begin
- if DontClearGraphMemory then
- CallInt10(mode or $80)
- else
- CallInt10(mode);
- end;
- {************************************************************************}
- {* 720x348x2 Hercules mode routines *}
- {************************************************************************}
- var
- DummyHGCBkColor: Word;
- procedure InitHGC720;
- const
- RegValues: array [0..11] of byte =
- ($35, $2D, $2E, $07, $5B, $02, $57, $57, $02, $03, $00, $00);
- var
- I: Integer;
- begin
- Port[$3BF] := 3; { graphic and page 2 possible }
- Port[$3B8] := 2; { display page 0, graphic mode, display off }
- for I := 0 to 11 do
- PortW[$3B4] := I or (RegValues[I] shl 8);
- Port[$3B8] := 10; { display page 0, graphic mode, display on }
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegB000
- mov es, ax
- mov es, es:[SegB000]
- {$else FPC_MM_HUGE}
- mov es, [SegB000]
- {$endif FPC_MM_HUGE}
- mov cx, 32768
- xor di, di
- xor ax, ax
- cld
- rep stosw
- end ['ax','cx','di'];
- VideoOfs := 0;
- DummyHGCBkColor := 0;
- end;
- { compatible with TP7's HERC.BGI }
- procedure SetBkColorHGC720(ColorNum: ColorType);
- begin
- if ColorNum > 15 then
- exit;
- DummyHGCBkColor := ColorNum;
- end;
- { compatible with TP7's HERC.BGI }
- function GetBkColorHGC720: ColorType;
- begin
- GetBkColorHGC720 := DummyHGCBkColor;
- end;
- procedure SetHGCRGBPalette(ColorNum, RedValue, GreenValue,
- BlueValue : smallint);
- begin
- end;
- procedure GetHGCRGBPalette(ColorNum: smallint; Var
- RedValue, GreenValue, BlueValue : smallint);
- begin
- end;
- procedure PutPixelHGC720(X, Y: SmallInt; Pixel: ColorType);
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 2) * 90 + (X shr 3) + VideoOfs;
- case Y and 3 of
- 1: Inc(Offset, $2000);
- 2: Inc(Offset, $4000);
- 3: Inc(Offset, $6000);
- end;
- Shift := 7 - (X and 7);
- Mask := 1 shl Shift;
- B := Mem[SegB000:Offset];
- B := B and (not Mask) or (Pixel shl Shift);
- Mem[SegB000:Offset] := B;
- end;
- function GetPixelHGC720(X, Y: SmallInt): ColorType;
- var
- Offset: Word;
- B, Shift: Byte;
- begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 2) * 90 + (X shr 3) + VideoOfs;
- case Y and 3 of
- 1: Inc(Offset, $2000);
- 2: Inc(Offset, $4000);
- 3: Inc(Offset, $6000);
- end;
- Shift := 7 - (X and 7);
- B := Mem[SegB000:Offset];
- GetPixelHGC720 := (B shr Shift) and 1;
- end;
- procedure DirectPutPixelHGC720(X, Y: SmallInt);
- { x,y -> must be in global coordinates. No clipping. }
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- Offset := (Y shr 2) * 90 + (X shr 3) + VideoOfs;
- case Y and 3 of
- 1: Inc(Offset, $2000);
- 2: Inc(Offset, $4000);
- 3: Inc(Offset, $6000);
- end;
- Shift := 7 - (X and 7);
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:Offset] := Mem[SegB000:Offset] xor (CurrentColor shl Shift);
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:Offset] := Mem[SegB000:Offset] or (CurrentColor shl Shift);
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB000:Offset] := Mem[SegB000:Offset] and (not (1 shl Shift));
- end;
- NotPut:
- begin
- Mask := 1 shl Shift;
- B := Mem[SegB000:Offset];
- B := B and (not Mask) or ((CurrentColor xor $01) shl Shift);
- Mem[SegB000:Offset] := B;
- end
- else
- begin
- Mask := 1 shl Shift;
- B := Mem[SegB000:Offset];
- B := B and (not Mask) or (CurrentColor shl Shift);
- Mem[SegB000:Offset] := B;
- end;
- end;
- end;
- procedure HLineHGC720(X, X2, Y: SmallInt);
- var
- Color: Word;
- YOffset, LOffset, ROffset, CurrentOffset, MiddleAreaLength: Word;
- B, ForeMask, LForeMask, LBackMask, RForeMask, RBackMask: Byte;
- xtmp: SmallInt;
- begin
- { must we swap the values? }
- if x > x2 then
- begin
- xtmp := x2;
- x2 := x;
- x:= xtmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- X2 := X2 + StartXViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- begin
- if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- YOffset := (Y shr 2) * 90 + VideoOfs;
- case Y and 3 of
- 1: Inc(YOffset, $2000);
- 2: Inc(YOffset, $4000);
- 3: Inc(YOffset, $6000);
- end;
- LOffset := YOffset + (X shr 3);
- ROffset := YOffset + (X2 shr 3);
- if CurrentWriteMode = NotPut then
- Color := CurrentColor xor $01
- else
- Color := CurrentColor;
- if Color = 1 then
- ForeMask := $FF
- else
- ForeMask := $00;
- LBackMask := Byte($FF00 shr (X and $07));
- LForeMask := (not LBackMask) and ForeMask;
- RBackMask := Byte(not ($FF shl (7 - (X2 and $07))));
- RForeMask := (not RBackMask) and ForeMask;
- if LOffset = ROffset then
- begin
- LBackMask := LBackMask or RBackMask;
- LForeMask := LForeMask and RForeMask;
- end;
- CurrentOffset := LOffset;
- { check if the first byte is only partially full
- (otherwise, it's completely full and is handled as a part of the middle area) }
- if LBackMask <> 0 then
- begin
- { draw the first byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] xor LForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] or LForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] and LBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB000:CurrentOffset];
- B := B and LBackMask or LForeMask;
- Mem[SegB000:CurrentOffset] := B;
- end;
- end;
- Inc(CurrentOffset);
- end;
- if CurrentOffset > ROffset then
- exit;
- MiddleAreaLength := ROffset + 1 - CurrentOffset;
- if RBackMask <> 0 then
- Dec(MiddleAreaLength);
- { draw the middle area }
- if MiddleAreaLength > 0 then
- begin
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] xor $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB000:CurrentOffset] := $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB000:CurrentOffset] := 0;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- else
- begin
- { note: NotPut is also handled here }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB000:CurrentOffset] := ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- end;
- end;
- { draw the final right byte, if less than 100% full }
- if RBackMask <> 0 then
- begin
- { draw the last byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] xor RForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] or RForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB000:CurrentOffset] := Mem[SegB000:CurrentOffset] and RBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB000:CurrentOffset];
- B := B and RBackMask or RForeMask;
- Mem[SegB000:CurrentOffset] := B;
- end;
- end;
- end;
- end;
- procedure SetVisualHGC720(page: word);
- { two page supPort... }
- begin
- if page > HardwarePages then exit;
- case page of
- 0 : Port[$3B8] := 10; { display page 0, graphic mode, display on }
- 1 : Port[$3B8] := 10+128; { display page 1, graphic mode, display on }
- end;
- end;
- procedure SetActiveHGC720(page: word);
- { two page supPort... }
- begin
- case page of
- 0 : VideoOfs := 0;
- 1 : VideoOfs := 32768;
- else
- VideoOfs := 0;
- end;
- end;
- {************************************************************************}
- {* 320x200x4 CGA mode routines *}
- {************************************************************************}
- var
- CurrentCGABorder: Word;
- procedure SetCGAPalette(CGAPaletteID: Byte); assembler;
- asm
- mov ax,CGAPaletteID
- mov bl, al
- mov bh, 1
- mov ah, 0Bh
- push ds
- push bp
- int 10h
- pop bp
- pop ds
- end;
- procedure SetCGABorder(CGABorder: Byte); assembler;
- asm
- mov ax,CGABorder
- mov bl, al
- mov bh, 0
- mov ah, 0Bh
- push ds
- push bp
- int 10h
- pop bp
- pop ds
- end;
- procedure SetBkColorCGA320(ColorNum: ColorType);
- begin
- if ColorNum > 15 then
- exit;
- CurrentCGABorder := (CurrentCGABorder and 16) or ColorNum;
- SetCGABorder(CurrentCGABorder);
- end;
- function GetBkColorCGA320: ColorType;
- begin
- GetBkColorCGA320 := CurrentCGABorder and 15;
- end;
- procedure InitCGA320C0;
- begin
- InitInt10hMode($04);
- VideoOfs := 0;
- SetCGAPalette(0);
- SetCGABorder(16);
- CurrentCGABorder := 16;
- end;
- procedure InitCGA320C1;
- begin
- InitInt10hMode($04);
- VideoOfs := 0;
- SetCGAPalette(1);
- SetCGABorder(16);
- CurrentCGABorder := 16;
- end;
- procedure InitCGA320C2;
- begin
- InitInt10hMode($04);
- VideoOfs := 0;
- SetCGAPalette(2);
- SetCGABorder(0);
- CurrentCGABorder := 0;
- end;
- procedure InitCGA320C3;
- begin
- InitInt10hMode($04);
- VideoOfs := 0;
- SetCGAPalette(3);
- SetCGABorder(0);
- CurrentCGABorder := 0;
- end;
- procedure PutPixelCGA320(X, Y: SmallInt; Pixel: ColorType);
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 1) * 80 + (X shr 2);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 6 - ((X and 3) shl 1);
- Mask := $03 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or (Pixel shl Shift);
- Mem[SegB800:Offset] := B;
- end;
- function GetPixelCGA320(X, Y: SmallInt): ColorType;
- var
- Offset: Word;
- B, Shift: Byte;
- begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 1) * 80 + (X shr 2);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 6 - ((X and 3) shl 1);
- B := Mem[SegB800:Offset];
- GetPixelCGA320 := (B shr Shift) and $03;
- end;
- procedure DirectPutPixelCGA320(X, Y: SmallInt);
- { x,y -> must be in global coordinates. No clipping. }
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- Offset := (Y shr 1) * 80 + (X shr 2);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 6 - ((X and 3) shl 1);
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:Offset] := Mem[SegB800:Offset] xor (CurrentColor shl Shift);
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:Offset] := Mem[SegB800:Offset] or (CurrentColor shl Shift);
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 3 then
- exit;
- Mask := $03 shl Shift;
- Mem[SegB800:Offset] := Mem[SegB800:Offset] and ((CurrentColor shl Shift) or (not Mask));
- end;
- NotPut:
- begin
- Mask := $03 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or ((CurrentColor xor $03) shl Shift);
- Mem[SegB800:Offset] := B;
- end
- else
- begin
- Mask := $03 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or (CurrentColor shl Shift);
- Mem[SegB800:Offset] := B;
- end;
- end;
- end;
- procedure HLineCGA320(X, X2, Y: SmallInt);
- var
- Color: Word;
- YOffset, LOffset, ROffset, CurrentOffset, MiddleAreaLength: Word;
- B, ForeMask, LForeMask, LBackMask, RForeMask, RBackMask: Byte;
- xtmp: SmallInt;
- begin
- { must we swap the values? }
- if x > x2 then
- begin
- xtmp := x2;
- x2 := x;
- x:= xtmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- X2 := X2 + StartXViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- begin
- if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- YOffset := (Y shr 1) * 80;
- if (Y and 1) <> 0 then
- Inc(YOffset, 8192);
- LOffset := YOffset + (X shr 2);
- ROffset := YOffset + (X2 shr 2);
- if CurrentWriteMode = NotPut then
- Color := CurrentColor xor $03
- else
- Color := CurrentColor;
- case Color of
- 0: ForeMask := $00;
- 1: ForeMask := $55;
- 2: ForeMask := $AA;
- 3: ForeMask := $FF;
- end;
- LBackMask := Byte($FF00 shr ((X and $03) shl 1));
- LForeMask := (not LBackMask) and ForeMask;
- RBackMask := Byte(not ($FF shl (6 - ((X2 and $03) shl 1))));
- RForeMask := (not RBackMask) and ForeMask;
- if LOffset = ROffset then
- begin
- LBackMask := LBackMask or RBackMask;
- LForeMask := LForeMask and RForeMask;
- end;
- CurrentOffset := LOffset;
- { check if the first byte is only partially full
- (otherwise, it's completely full and is handled as a part of the middle area) }
- if LBackMask <> 0 then
- begin
- { draw the first byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor LForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] or LForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 3 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] and (LBackMask or LForeMask);
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB800:CurrentOffset];
- B := B and LBackMask or LForeMask;
- Mem[SegB800:CurrentOffset] := B;
- end;
- end;
- Inc(CurrentOffset);
- end;
- if CurrentOffset > ROffset then
- exit;
- MiddleAreaLength := ROffset + 1 - CurrentOffset;
- if RBackMask <> 0 then
- Dec(MiddleAreaLength);
- { draw the middle area }
- if MiddleAreaLength > 0 then
- begin
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] or ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 3 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] and ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- else
- begin
- { note: NotPut is also handled here }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- end;
- end;
- { draw the final right byte, if less than 100% full }
- if RBackMask <> 0 then
- begin
- { draw the last byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor RForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] or RForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 3 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] and (RBackMask or RForeMask);
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB800:CurrentOffset];
- B := B and RBackMask or RForeMask;
- Mem[SegB800:CurrentOffset] := B;
- end;
- end;
- end;
- end;
- {************************************************************************}
- {* 640x200x2 CGA mode routines *}
- {************************************************************************}
- procedure InitCGA640;
- begin
- InitInt10hMode($06);
- VideoOfs := 0;
- CurrentCGABorder := 0; {yes, TP7 CGA.BGI behaves *exactly* like that}
- end;
- {yes, TP7 CGA.BGI behaves *exactly* like that}
- procedure SetBkColorCGA640(ColorNum: ColorType);
- begin
- if ColorNum > 15 then
- exit;
- CurrentCGABorder := ColorNum;
- if ColorNum = 0 then
- exit;
- SetCGABorder(CurrentCGABorder);
- end;
- function GetBkColorCGA640: ColorType;
- begin
- GetBkColorCGA640 := CurrentCGABorder and 15;
- end;
- procedure PutPixelCGA640(X, Y: SmallInt; Pixel: ColorType);
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 1) * 80 + (X shr 3);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 7 - (X and 7);
- Mask := 1 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or (Pixel shl Shift);
- Mem[SegB800:Offset] := B;
- end;
- function GetPixelCGA640(X, Y: SmallInt): ColorType;
- var
- Offset: Word;
- B, Shift: Byte;
- begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := (Y shr 1) * 80 + (X shr 3);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 7 - (X and 7);
- B := Mem[SegB800:Offset];
- GetPixelCGA640 := (B shr Shift) and 1;
- end;
- procedure DirectPutPixelCGA640(X, Y: SmallInt);
- { x,y -> must be in global coordinates. No clipping. }
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- Offset := (Y shr 1) * 80 + (X shr 3);
- if (Y and 1) <> 0 then
- Inc(Offset, 8192);
- Shift := 7 - (X and 7);
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:Offset] := Mem[SegB800:Offset] xor (CurrentColor shl Shift);
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:Offset] := Mem[SegB800:Offset] or (CurrentColor shl Shift);
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB800:Offset] := Mem[SegB800:Offset] and (not (1 shl Shift));
- end;
- NotPut:
- begin
- Mask := 1 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or ((CurrentColor xor $01) shl Shift);
- Mem[SegB800:Offset] := B;
- end
- else
- begin
- Mask := 1 shl Shift;
- B := Mem[SegB800:Offset];
- B := B and (not Mask) or (CurrentColor shl Shift);
- Mem[SegB800:Offset] := B;
- end;
- end;
- end;
- procedure HLineCGA640(X, X2, Y: SmallInt);
- var
- Color: Word;
- YOffset, LOffset, ROffset, CurrentOffset, MiddleAreaLength: Word;
- B, ForeMask, LForeMask, LBackMask, RForeMask, RBackMask: Byte;
- xtmp: SmallInt;
- begin
- { must we swap the values? }
- if x > x2 then
- begin
- xtmp := x2;
- x2 := x;
- x:= xtmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- X2 := X2 + StartXViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- begin
- if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- YOffset := (Y shr 1) * 80;
- if (Y and 1) <> 0 then
- Inc(YOffset, 8192);
- LOffset := YOffset + (X shr 3);
- ROffset := YOffset + (X2 shr 3);
- if CurrentWriteMode = NotPut then
- Color := CurrentColor xor $01
- else
- Color := CurrentColor;
- if Color = 1 then
- ForeMask := $FF
- else
- ForeMask := $00;
- LBackMask := Byte($FF00 shr (X and $07));
- LForeMask := (not LBackMask) and ForeMask;
- RBackMask := Byte(not ($FF shl (7 - (X2 and $07))));
- RForeMask := (not RBackMask) and ForeMask;
- if LOffset = ROffset then
- begin
- LBackMask := LBackMask or RBackMask;
- LForeMask := LForeMask and RForeMask;
- end;
- CurrentOffset := LOffset;
- { check if the first byte is only partially full
- (otherwise, it's completely full and is handled as a part of the middle area) }
- if LBackMask <> 0 then
- begin
- { draw the first byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor LForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] or LForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] and LBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB800:CurrentOffset];
- B := B and LBackMask or LForeMask;
- Mem[SegB800:CurrentOffset] := B;
- end;
- end;
- Inc(CurrentOffset);
- end;
- if CurrentOffset > ROffset then
- exit;
- MiddleAreaLength := ROffset + 1 - CurrentOffset;
- if RBackMask <> 0 then
- Dec(MiddleAreaLength);
- { draw the middle area }
- if MiddleAreaLength > 0 then
- begin
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := 0;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- else
- begin
- { note: NotPut is also handled here }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegB800:CurrentOffset] := ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- end;
- end;
- { draw the final right byte, if less than 100% full }
- if RBackMask <> 0 then
- begin
- { draw the last byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] xor RForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] or RForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegB800:CurrentOffset] := Mem[SegB800:CurrentOffset] and RBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegB800:CurrentOffset];
- B := B and RBackMask or RForeMask;
- Mem[SegB800:CurrentOffset] := B;
- end;
- end;
- end;
- end;
- {************************************************************************}
- {* 640x480x2 MCGA mode routines *}
- {************************************************************************}
- procedure InitMCGA640;
- begin
- InitInt10hMode($11);
- VideoOfs := 0;
- CurrentCGABorder := 0; {yes, TP7 CGA.BGI behaves *exactly* like that}
- end;
- procedure SetBkColorMCGA640(ColorNum: ColorType);
- begin
- if ColorNum > 15 then
- exit;
- CurrentCGABorder := (CurrentCGABorder and 16) or ColorNum;
- SetCGABorder(CurrentCGABorder);
- end;
- function GetBkColorMCGA640: ColorType;
- begin
- GetBkColorMCGA640 := CurrentCGABorder and 15;
- end;
- procedure PutPixelMCGA640(X, Y: SmallInt; Pixel: ColorType);
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := Y * 80 + (X shr 3);
- Shift := 7 - (X and 7);
- Mask := 1 shl Shift;
- B := Mem[SegA000:Offset];
- B := B and (not Mask) or (Pixel shl Shift);
- Mem[SegA000:Offset] := B;
- end;
- function GetPixelMCGA640(X, Y: SmallInt): ColorType;
- var
- Offset: Word;
- B, Shift: Byte;
- begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Offset := Y * 80 + (X shr 3);
- Shift := 7 - (X and 7);
- B := Mem[SegA000:Offset];
- GetPixelMCGA640 := (B shr Shift) and 1;
- end;
- procedure DirectPutPixelMCGA640(X, Y: SmallInt);
- { x,y -> must be in global coordinates. No clipping. }
- var
- Offset: Word;
- B, Mask, Shift: Byte;
- begin
- Offset := Y * 80 + (X shr 3);
- Shift := 7 - (X and 7);
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:Offset] := Mem[SegA000:Offset] xor (CurrentColor shl Shift);
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:Offset] := Mem[SegA000:Offset] or (CurrentColor shl Shift);
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegA000:Offset] := Mem[SegA000:Offset] and (not (1 shl Shift));
- end;
- NotPut:
- begin
- Mask := 1 shl Shift;
- B := Mem[SegA000:Offset];
- B := B and (not Mask) or ((CurrentColor xor $01) shl Shift);
- Mem[SegA000:Offset] := B;
- end
- else
- begin
- Mask := 1 shl Shift;
- B := Mem[SegA000:Offset];
- B := B and (not Mask) or (CurrentColor shl Shift);
- Mem[SegA000:Offset] := B;
- end;
- end;
- end;
- procedure HLineMCGA640(X, X2, Y: SmallInt);
- var
- Color: Word;
- YOffset, LOffset, ROffset, CurrentOffset, MiddleAreaLength: Word;
- B, ForeMask, LForeMask, LBackMask, RForeMask, RBackMask: Byte;
- xtmp: SmallInt;
- begin
- { must we swap the values? }
- if x > x2 then
- begin
- xtmp := x2;
- x2 := x;
- x:= xtmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- X2 := X2 + StartXViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- begin
- if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- YOffset := Y * 80;
- LOffset := YOffset + (X shr 3);
- ROffset := YOffset + (X2 shr 3);
- if CurrentWriteMode = NotPut then
- Color := CurrentColor xor $01
- else
- Color := CurrentColor;
- if Color = 1 then
- ForeMask := $FF
- else
- ForeMask := $00;
- LBackMask := Byte($FF00 shr (X and $07));
- LForeMask := (not LBackMask) and ForeMask;
- RBackMask := Byte(not ($FF shl (7 - (X2 and $07))));
- RForeMask := (not RBackMask) and ForeMask;
- if LOffset = ROffset then
- begin
- LBackMask := LBackMask or RBackMask;
- LForeMask := LForeMask and RForeMask;
- end;
- CurrentOffset := LOffset;
- { check if the first byte is only partially full
- (otherwise, it's completely full and is handled as a part of the middle area) }
- if LBackMask <> 0 then
- begin
- { draw the first byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] xor LForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] or LForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] and LBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegA000:CurrentOffset];
- B := B and LBackMask or LForeMask;
- Mem[SegA000:CurrentOffset] := B;
- end;
- end;
- Inc(CurrentOffset);
- end;
- if CurrentOffset > ROffset then
- exit;
- MiddleAreaLength := ROffset + 1 - CurrentOffset;
- if RBackMask <> 0 then
- Dec(MiddleAreaLength);
- { draw the middle area }
- if MiddleAreaLength > 0 then
- begin
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] xor $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- while MiddleAreaLength > 0 do
- begin
- Mem[SegA000:CurrentOffset] := $FF;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegA000:CurrentOffset] := 0;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- else
- begin
- { note: NotPut is also handled here }
- while MiddleAreaLength > 0 do
- begin
- Mem[SegA000:CurrentOffset] := ForeMask;
- Inc(CurrentOffset);
- Dec(MiddleAreaLength);
- end;
- end;
- end;
- end;
- if RBackMask <> 0 then
- begin
- { draw the last byte }
- case CurrentWriteMode of
- XORPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] xor RForeMask;
- end;
- OrPut:
- begin
- { optimization }
- if CurrentColor = 0 then
- exit;
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] or RForeMask;
- end;
- AndPut:
- begin
- { optimization }
- if CurrentColor = 1 then
- exit;
- { therefore, CurrentColor must be 0 }
- Mem[SegA000:CurrentOffset] := Mem[SegA000:CurrentOffset] and RBackMask;
- end;
- else
- begin
- { note: NotPut is also handled here }
- B := Mem[SegA000:CurrentOffset];
- B := B and RBackMask or RForeMask;
- Mem[SegA000:CurrentOffset] := B;
- end;
- end;
- end;
- end;
- {************************************************************************}
- {* 4-bit planar VGA mode routines *}
- {************************************************************************}
- Procedure Init640x200x16;
- begin
- InitInt10hMode($e);
- VideoOfs := 0;
- end;
- Procedure Init640x350x16;
- begin
- InitInt10hMode($10);
- VideoOfs := 0;
- end;
- Procedure Init640x480x16;
- begin
- InitInt10hMode($12);
- VideoOfs := 0;
- end;
- {$ifndef asmgraph}
- Procedure PutPixel16(X,Y : smallint; Pixel: ColorType);
- var offset: word;
- dummy: byte;
- Begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- offset := y * 80 + (x shr 3) + VideoOfs;
- PortW[$3ce] := $0f01; { Index 01 : Enable ops on all 4 planes }
- PortW[$3ce] := (Pixel and $ff) shl 8; { Index 00 : Enable correct plane and write color }
- PortW[$3ce] := ($8000 shr (x and $7)) or 8; { Select correct bits to modify }
- dummy := Mem[SegA000: offset]; { Latch the data into host space. }
- Mem[Sega000: offset] := dummy; { Write the data into video memory }
- PortW[$3ce] := $ff08; { Enable all bit planes. }
- PortW[$3ce] := $0001; { Index 01 : Disable ops on all four planes. }
- end;
- {$else asmgraph}
- Procedure PutPixel16(X,Y : smallint; Pixel: ColorType); assembler;
- asm
- mov si, [X]
- mov bx, [Y]
- cmp byte ptr [ClipPixels], 0
- je @@ClipDone
- test si, si
- js @@Done
- test bx, bx
- js @@Done
- cmp si, [ViewWidth]
- jg @@Done
- cmp bx, [ViewHeight]
- jg @@Done
- @@ClipDone:
- add si, [StartXViewPort]
- add bx, [StartYViewPort]
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- { enable the set / reset function and load the color }
- mov dx, 3ceh
- mov ax, 0f01h
- out dx, ax
- { setup set/reset register }
- mov ah, byte ptr [Pixel]
- xor al, al
- out dx, ax
- { setup the bit mask register }
- mov al, 8
- { load the bitmask register }
- mov cx, si
- and cl, 07h
- mov ah, 80h
- shr ah, cl
- out dx, ax
- { get the x index and divide by 8 for 16-color }
- mov cl, 3
- shr si, cl
- { determine the address }
- inc cx { CL=4 }
- shl bx, cl
- mov di, bx
- shl di, 1
- shl di, 1
- add di, bx
- add di, si
- add di, [VideoOfs]
- { send the data through the display memory through set/reset }
- mov bl,es:[di]
- stosb
- { reset for formal vga operation }
- mov ax,0ff08h
- out dx,ax
- { restore enable set/reset register }
- mov ax,0001h
- out dx,ax
- @@Done:
- end;
- {$endif asmgraph}
- {$ifndef asmgraph}
- Function GetPixel16(X,Y: smallint):ColorType;
- Var dummy, offset: Word;
- shift: byte;
- Begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- offset := Y * 80 + (x shr 3) + VideoOfs;
- PortW[$3ce] := $0004;
- shift := 7 - (X and 7);
- dummy := (Mem[Sega000:offset] shr shift) and 1;
- Port[$3cf] := 1;
- dummy := dummy or (((Mem[Sega000:offset] shr shift) and 1) shl 1);
- Port[$3cf] := 2;
- dummy := dummy or (((Mem[Sega000:offset] shr shift) and 1) shl 2);
- Port[$3cf] := 3;
- dummy := dummy or (((Mem[Sega000:offset] shr shift) and 1) shl 3);
- GetPixel16 := dummy;
- end;
- {$else asmgraph}
- Function GetPixel16(X,Y: smallint):ColorType;assembler;
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- mov dx,03ceh
- mov ax,0304h
- out dx,ax
- inc dx
- mov di, [X] { Get X address }
- add di, [StartXViewPort]
- mov ax, di
- mov cl, 3
- shr di, cl
- mov bx, [Y]
- add bx, [StartYViewPort]
- inc cx { CL=4 }
- shl bx, cl { BX=16*(Y+StartYViewPort)*16 }
- mov si, bx { SI=16*(Y+StartYViewPort)*16 }
- shl si, 1 { SI=32*(Y+StartYViewPort)*32 }
- shl si, 1 { SI=64*(Y+StartYViewPort)*64 }
- add si, bx { SI=(64+16)*(Y+StartYViewPort)=80*(Y+StartYViewPort) }
- add si, di { SI=correct offset into video segment }
- add si, [VideoOfs] { Point to correct page offset... }
- xchg ax, cx { 1 byte shorter than 'mov cx, ax' }
- and cl,7
- mov bh, 080h
- shr bh, cl
- { read plane 3 }
- mov ah,es:[si] { read display memory }
- and ah,bh { save bit in AH }
- { read plane 2 }
- mov al,2 { Select plane to read }
- out dx,al
- mov bl,es:[si]
- and bl,bh
- rol ah,1
- or ah,bl { save bit in AH }
- { read plane 1 }
- dec ax { Select plane to read }
- out dx,al
- mov bl,es:[si]
- and bl,bh
- rol ah,1
- or ah,bl { save bit in AH }
- { read plane 0 }
- dec ax { Select plane to read }
- out dx,al
- seges lodsb
- and al,bh
- rol ah,1
- or al,ah { add previous bits from AH into AL }
- inc cx
- rol al,cl { 16-bit pixel in AX }
- { 1 byte shorter than 'xor ah, ah'; will always set ah to 0, because sign(al)=0 }
- cbw
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- { 1 byte shorter than 'xor dx, dx'; will always set dx to 0, because sign(ah)=0 }
- cwd
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- end;
- {$endif asmgraph}
- Procedure GetScanLine16(x1, x2, y: smallint; var data);
- var dummy: word;
- Offset, count, count2, amount, index: word;
- plane: byte;
- Begin
- inc(x1,StartXViewPort);
- inc(x2,StartXViewPort);
- {$ifdef logging}
- LogLn('GetScanLine16 start, length to get: '+strf(x2-x1+1)+' at y = '+strf(y));
- {$Endif logging}
- offset := (Y + StartYViewPort) * 80 + (x1 shr 3) + VideoOfs;
- {$ifdef logging}
- LogLn('Offset: '+HexStr(offset,4)+' - ' + strf(offset));
- {$Endif logging}
- { first get enough pixels so offset is 16bit aligned }
- amount := 0;
- index := 0;
- If ((x1 and 15) <> 0) Or
- ((x2-x1+1) < 16) Then
- Begin
- If ((x2-x1+1) >= 16+16-(x1 and 15)) Then
- amount := 16-(x1 and 15)
- Else amount := x2-x1+1;
- {$ifdef logging}
- LogLn('amount to align to 16bits or to get all: ' + strf(amount));
- {$Endif logging}
- For count := 0 to amount-1 do
- WordArray(Data)[Count] := getpixel16(x1-StartXViewPort+Count,y);
- index := amount;
- Inc(Offset,(amount+7) shr 3);
- {$ifdef logging}
- LogLn('offset now: '+HexStr(offset,4)+' - ' + strf(offset));
- LogLn('index now: '+strf(index));
- {$Endif logging}
- End;
- amount := x2-x1+1 - amount;
- {$ifdef logging}
- LogLn('amount left: ' + strf(amount));
- {$Endif logging}
- If amount = 0 Then Exit;
- { first get everything from plane 3 (4th plane) }
- PortW[$3ce] := $0304;
- Count := 0;
- For Count := 1 to (amount shr 4) Do
- Begin
- dummy := MemW[SegA000:offset+(Count-1)*2];
- dummy :=
- ((dummy and $ff) shl 8) or
- ((dummy and $ff00) shr 8);
- For Count2 := 15 downto 0 Do
- Begin
- WordArray(Data)[index+Count2] := Dummy and 1;
- Dummy := Dummy shr 1;
- End;
- Inc(Index, 16);
- End;
- { Now get the data from the 3 other planes }
- plane := 3;
- Repeat
- Dec(Index,Count*16);
- Dec(plane);
- Port[$3cf] := plane;
- Count := 0;
- For Count := 1 to (amount shr 4) Do
- Begin
- dummy := MemW[SegA000:offset+(Count-1)*2];
- dummy :=
- ((dummy and $ff) shl 8) or
- ((dummy and $ff00) shr 8);
- For Count2 := 15 downto 0 Do
- Begin
- WordArray(Data)[index+Count2] :=
- (WordArray(Data)[index+Count2] shl 1) or (Dummy and 1);
- Dummy := Dummy shr 1;
- End;
- Inc(Index, 16);
- End;
- Until plane = 0;
- amount := amount and 15;
- Dec(index);
- {$ifdef Logging}
- LogLn('Last array index written to: '+strf(index));
- LogLn('amount left: '+strf(amount)+' starting at x = '+strf(index+1));
- {$Endif logging}
- dec(x1,startXViewPort);
- For Count := 1 to amount Do
- WordArray(Data)[index+Count] := getpixel16(x1+index+Count,y);
- {$ifdef logging}
- inc(x1,startXViewPort);
- LogLn('First 16 bytes gotten with getscanline16: ');
- If x2-x1+1 >= 16 Then
- Count2 := 16
- Else Count2 := x2-x1+1;
- For Count := 0 to Count2-1 Do
- Log(strf(WordArray(Data)[Count])+' ');
- LogLn('');
- If x2-x1+1 >= 16 Then
- Begin
- LogLn('Last 16 bytes gotten with getscanline16: ');
- For Count := 15 downto 0 Do
- Log(strf(WordArray(Data)[x2-x1-Count])+' ');
- End;
- LogLn('');
- GetScanLineDefault(x1-StartXViewPort,x2-StartXViewPort,y,Data);
- LogLn('First 16 bytes gotten with getscanlinedef: ');
- If x2-x1+1 >= 16 Then
- Count2 := 16
- Else Count2 := x2-x1+1;
- For Count := 0 to Count2-1 Do
- Log(strf(WordArray(Data)[Count])+' ');
- LogLn('');
- If x2-x1+1 >= 16 Then
- Begin
- LogLn('Last 16 bytes gotten with getscanlinedef: ');
- For Count := 15 downto 0 Do
- Log(strf(WordArray(Data)[x2-x1-Count])+' ');
- End;
- LogLn('');
- LogLn('GetScanLine16 end');
- {$Endif logging}
- End;
- {$ifndef asmgraph}
- Procedure DirectPutPixel16(X,Y : smallint);
- { x,y -> must be in global coordinates. No clipping. }
- var
- color: word;
- offset: word;
- dummy: byte;
- begin
- If CurrentWriteMode <> NotPut Then
- Color := CurrentColor
- else Color := not CurrentColor;
- case CurrentWriteMode of
- XORPut:
- PortW[$3ce]:=((3 shl 3) shl 8) or 3;
- ANDPut:
- PortW[$3ce]:=((1 shl 3) shl 8) or 3;
- ORPut:
- PortW[$3ce]:=((2 shl 3) shl 8) or 3;
- {not needed, this is the default state (e.g. PutPixel16 requires it)}
- {NormalPut, NotPut:
- PortW[$3ce]:=$0003
- else
- PortW[$3ce]:=$0003}
- end;
- offset := Y * 80 + (X shr 3) + VideoOfs;
- PortW[$3ce] := $f01;
- PortW[$3ce] := Color shl 8;
- PortW[$3ce] := ($8000 shr (X and 7)) or 8;
- dummy := Mem[SegA000: offset];
- Mem[Sega000: offset] := dummy;
- PortW[$3ce] := $ff08;
- PortW[$3ce] := $0001;
- if (CurrentWriteMode = XORPut) or
- (CurrentWriteMode = ANDPut) or
- (CurrentWriteMode = ORPut) then
- PortW[$3ce] := $0003;
- end;
- {$else asmgraph}
- Procedure DirectPutPixel16(X,Y : smallint); assembler;
- const
- DataRotateRegTbl: array [NormalPut..NotPut] of Byte=($00,$18,$10,$08,$00);
- { x,y -> must be in global coordinates. No clipping. }
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- mov dx, 3ceh
- xor ch, ch { Color mask = 0 }
- mov bx, [CurrentWriteMode]
- test bl, 4 { NotPut? }
- jz @@NoNotPut
- { NotPut }
- mov ch, 15 { Color mask for NotPut }
- @@NoNotPut:
- mov ah, byte ptr [DataRotateRegTbl + bx]
- test ah, ah
- jz @@NormalPut
- mov al, 3
- out dx, ax
- @@NormalPut:
- { enable the set / reset function and load the color }
- mov ax, 0f01h
- out dx, ax
- { setup set/reset register }
- mov ah, byte ptr [CurrentColor]
- xor ah, ch { Maybe apply the NotPut mask }
- xor al, al
- out dx, ax
- { setup the bit mask register }
- mov al, 8
- { load the bitmask register }
- mov si, [X]
- mov cx, si
- and cl, 07h
- mov ah, 80h
- shr ah, cl
- out dx, ax
- { get the x index and divide by 8 for 16-color }
- mov cl, 3
- shr si, cl
- { determine the address }
- mov bx, [Y]
- inc cx { CL=4 }
- shl bx, cl
- mov di, bx
- shl di, 1
- shl di, 1
- add di, bx
- add di, si
- add di, [VideoOfs] { add correct page }
- { send the data through the display memory through set/reset }
- mov al,es:[di]
- stosb
- { reset for formal vga operation }
- mov ax,0ff08h
- out dx,ax
- { restore enable set/reset register }
- mov ax,0001h
- out dx,ax
- test bl, 3 { NormalPut or NotPut? }
- jz @@Done { If yes, skip }
- mov ax,0003h
- out dx,ax
- @@Done:
- end;
- {$endif asmgraph}
- procedure HLine16(x,x2,y: smallint);
- var
- xtmp: smallint;
- ScrOfs,HLength : word;
- LMask,RMask : byte;
- Begin
- { must we swap the values? }
- if x > x2 then
- Begin
- xtmp := x2;
- x2 := x;
- x:= xtmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- X2 := X2 + StartXViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- Begin
- if LineClipped(x,y,x2,y,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- ScrOfs:=y*ScrWidth+x div 8 + VideoOfs;
- HLength:=x2 div 8-x div 8;
- LMask:=$ff shr (x and 7);
- {$push}
- {$r-}
- {$q-}
- RMask:=$ff shl (7-(x2 and 7));
- {$pop}
- if HLength=0 then
- LMask:=LMask and RMask;
- If CurrentWriteMode <> NotPut Then
- PortW[$3ce]:= CurrentColor shl 8
- else PortW[$3ce]:= (not CurrentColor) shl 8;
- PortW[$3ce]:=$0f01;
- case CurrentWriteMode of
- XORPut:
- PortW[$3ce]:=((3 shl 3) shl 8) or 3;
- ANDPut:
- PortW[$3ce]:=((1 shl 3) shl 8) or 3;
- ORPut:
- PortW[$3ce]:=((2 shl 3) shl 8) or 3;
- NormalPut, NotPut:
- PortW[$3ce]:=$0003
- else
- PortW[$3ce]:=$0003
- end;
- PortW[$3ce]:=(LMask shl 8) or 8;
- {$push}
- {$r-}
- {$q-}
- Mem[SegA000:ScrOfs]:=Mem[SegA000:ScrOfs]+1;
- {$pop}
- {Port[$3ce]:=8;}{not needed, the register is already selected}
- if HLength>0 then
- begin
- dec(HLength);
- inc(ScrOfs);
- if HLength>0 then
- begin
- Port[$3cf]:=$ff;
- {$ifndef tp}
- seg_bytemove(SegA000,ScrOfs,SegA000,ScrOfs,HLength);
- {$else}
- move(Ptr(SegA000,ScrOfs)^, Ptr(SegA000,ScrOfs)^, HLength);
- {$endif}
- ScrOfs:=ScrOfs+HLength;
- end;
- Port[$3cf]:=RMask;
- {$push}
- {$r-}
- {$q-}
- Mem[Sega000:ScrOfs]:=Mem[SegA000:ScrOfs]+1;
- {$pop}
- end;
- { clean up }
- {Port[$3cf]:=0;}{not needed, the register is reset by the next operation:}
- PortW[$3ce]:=$ff08;
- PortW[$3ce]:=$0001;
- PortW[$3ce]:=$0003;
- end;
- procedure VLine16(x,y,y2: smallint);
- var
- ytmp,i: smallint;
- ScrOfs: word;
- BitMask : byte;
- Begin
- { must we swap the values? }
- if y > y2 then
- Begin
- ytmp := y2;
- y2 := y;
- y:= ytmp;
- end;
- { First convert to global coordinates }
- X := X + StartXViewPort;
- Y2 := Y2 + StartYViewPort;
- Y := Y + StartYViewPort;
- if ClipPixels then
- Begin
- if LineClipped(x,y,x,y2,StartXViewPort,StartYViewPort,
- StartXViewPort+ViewWidth, StartYViewPort+ViewHeight) then
- exit;
- end;
- ScrOfs:=y*ScrWidth+x div 8 + VideoOfs;
- BitMask:=$80 shr (x and 7);
- If CurrentWriteMode <> NotPut Then
- PortW[$3ce]:= (CurrentColor shl 8)
- else PortW[$3ce]:= (not CurrentColor) shl 8;
- PortW[$3ce]:=$0f01;
- PortW[$3ce]:=(BitMask shl 8) or 8;
- case CurrentWriteMode of
- XORPut:
- PortW[$3ce]:=((3 shl 3) shl 8) or 3;
- ANDPut:
- PortW[$3ce]:=((1 shl 3) shl 8) or 3;
- ORPut:
- PortW[$3ce]:=((2 shl 3) shl 8) or 3;
- NormalPut, NotPut:
- PortW[$3ce]:=$0003
- else
- PortW[$3ce]:=$0003
- end;
- for i:=y to y2 do
- begin
- {$push}
- {$r-}
- {$q-}
- Mem[SegA000:ScrOfs]:=Mem[Sega000:ScrOfs]+1;
- {$pop}
- ScrOfs:=ScrOfs+ScrWidth;
- end;
- { clean up }
- {Port[$3cf]:=0;}{not needed, the register is reset by the next operation}
- PortW[$3ce]:=$ff08;
- PortW[$3ce]:=$0001;
- PortW[$3ce]:=$0003;
- End;
- procedure SetVisual200_350(page: word);
- begin
- if page > HardwarePages then exit;
- asm
- mov al, byte ptr [page] { only lower byte is supported. }
- mov ah,05h
- push ds
- push bp
- int 10h
- pop bp
- pop ds
- end ['DX','CX','BX','AX','SI','DI'];
- end;
- procedure SetActive200(page: word);
- { four page support... }
- begin
- case page of
- 0 : VideoOfs := 0;
- 1 : VideoOfs := 16384;
- 2 : VideoOfs := 32768;
- 3 : VideoOfs := 49152;
- else
- VideoOfs := 0;
- end;
- end;
- procedure SetActive350(page: word);
- { one page supPort... }
- begin
- case page of
- 0 : VideoOfs := 0;
- 1 : VideoOfs := 32768;
- else
- VideoOfs := 0;
- end;
- end;
- {************************************************************************}
- {* 320x200x256c Routines *}
- {************************************************************************}
- Procedure Init320;
- begin
- InitInt10hMode($13);
- end;
- {$ifndef asmgraph}
- Procedure PutPixel320(X,Y : smallint; Pixel: ColorType);
- { x,y -> must be in local coordinates. Clipping if required. }
- Begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- Mem[SegA000:Y*320+X] := Pixel;
- end;
- {$else asmgraph}
- Procedure PutPixel320(X,Y : smallint; Pixel: ColorType); assembler;
- asm
- mov ax, [Y]
- mov di, [X]
- cmp byte ptr [ClipPixels], 0
- je @@ClipDone
- test ax, ax
- js @@Done
- test di, di
- js @@Done
- cmp ax, [ViewHeight]
- jg @@Done
- cmp di, [ViewWidth]
- jg @@Done
- @@ClipDone:
- {$ifdef FPC_MM_HUGE}
- mov bx, SEG SegA000
- mov es, bx
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- add ax, [StartYViewPort]
- add di, [StartXViewPort]
- xchg ah, al { The value of Y must be in AH }
- add di, ax
- shr ax, 1
- shr ax, 1
- add di, ax
- mov al, byte ptr [Pixel]
- stosb
- @@Done:
- end;
- {$endif asmgraph}
- {$ifndef asmgraph}
- Function GetPixel320(X,Y: smallint):ColorType;
- Begin
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- GetPixel320 := Mem[SegA000:Y*320+X];
- end;
- {$else asmgraph}
- Function GetPixel320(X,Y: smallint):ColorType; assembler;
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- mov ax, [Y]
- add ax, [StartYViewPort]
- mov si, [X]
- add si, [StartXViewPort]
- xchg ah, al { The value of Y must be in AH }
- add si, ax
- shr ax, 1
- shr ax, 1
- add si, ax
- seges lodsb
- xor ah, ah
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- { 1 byte shorter than 'xor dx, dx'; will always set dx to 0, because sign(ah)=0 }
- cwd
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- end;
- {$endif asmgraph}
- {$ifndef asmgraph}
- Procedure DirectPutPixel320(X,Y : smallint);
- { x,y -> must be in global coordinates. No clipping. }
- var offset: word;
- dummy: Byte;
- begin
- dummy := CurrentColor;
- offset := y * 320 + x;
- case CurrentWriteMode of
- XorPut: dummy := dummy xor Mem[Sega000:offset];
- OrPut: dummy := dummy or Mem[Sega000:offset];
- AndPut: dummy := dummy and Mem[SegA000:offset];
- NotPut: dummy := Not dummy;
- end;
- Mem[SegA000:offset] := dummy;
- end;
- {$else asmgraph}
- Procedure DirectPutPixel320(X,Y : smallint); assembler;
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- mov ax, [Y]
- mov di, [X]
- xchg ah, al { The value of Y must be in AH }
- add di, ax
- shr ax, 1
- shr ax, 1
- add di, ax
- mov al, byte ptr [CurrentColor]
- { check write mode }
- mov bl, byte ptr [CurrentWriteMode]
- cmp bl, NormalPut
- jne @@1
- stosb
- jmp @Done
- @@1:
- cmp bl, XorPut
- jne @@2
- xor es:[di], al
- jmp @Done
- @@2:
- cmp bl, OrPut
- jne @@3
- or es:[di], al
- jmp @Done
- @@3:
- cmp bl, AndPut
- jne @NotPutMode
- and es:[di], al
- jmp @Done
- @NotPutMode:
- not al
- stosb
- @Done:
- end;
- {$endif asmgraph}
- procedure SetVisual320(page: word);
- { no page supPort... }
- begin
- end;
- procedure SetActive320(page: word);
- { no page supPort... }
- begin
- end;
- {************************************************************************}
- {* Mode-X related routines *}
- {************************************************************************}
- const CrtAddress: word = 0;
- {$ifndef asmgraph}
- procedure InitModeX;
- begin
- {see if we are using color-/monochrome display}
- if (Port[$3CC] and 1) <> 0 then
- CrtAddress := $3D4 { color }
- else
- CrtAddress := $3B4; { monochrome }
- InitInt10hMode($13);
- Port[$3C4] := $04; {select memory-mode-register at sequencer port }
- { bit 3 := 0: don't chain the 4 planes }
- { bit 2 := 1: no odd/even mechanism }
- Port[$3C5] := (Port[$3C5] and $F7) or $04;
- Port[$3C4] := $02; {s.a.: address sequencer reg. 2 (=map-mask),... }
- Port[$3C5] := $0F; {...and allow access to all 4 bit maps }
- { starting with segment A000h, set 8000h logical words = 4*8000h
- physical words (because of 4 bitplanes) to 0 }
- asm
- {$ifdef FPC_MM_HUGE}
- MOV AX,SEG SegA000
- MOV ES,AX
- MOV ES,ES:[SegA000]
- {$else FPC_MM_HUGE}
- MOV ES, [SegA000]
- {$endif FPC_MM_HUGE}
- XOR DI,DI
- XOR AX,AX
- MOV CX,8000h
- CLD
- REP STOSW
- end ['AX','CX','DI'];
- {address the underline-location-register at the CRT-controller
- port, read out the according data register: }
- Port[CRTAddress] := $14;
- {bit 6:=0: no double word addressing scheme in video RAM }
- Port[CRTAddress+1] := Port[CRTAddress+1] and $BF;
- Port[CRTAddress] := $17; {select mode control register }
- {bit 6 := 1: memory access scheme=linear bit array }
- Port[CRTAddress+1] := Port[CRTAddress+1] or $40;
- end;
- {$else asmgraph}
- procedure InitModeX; assembler;
- asm
- {see if we are using color-/monochrome display}
- MOV DX,3CCh {use output register: }
- IN AL,DX
- TEST AL,1 {is it a color display? }
- MOV DX,3D4h
- JNZ @L1 {yes }
- MOV DX,3B4h {no }
- @L1: {DX = 3B4h / 3D4h = CRTAddress-register for monochrome/color}
- MOV CRTAddress,DX
- MOV AX, 0013h
- CMP BYTE PTR [DontClearGraphMemory],0
- JE @L2
- OR AL, 080h
- @L2:
- push ds
- push bp
- INT 10h
- pop bp
- pop ds
- MOV DX,03C4h {select memory-mode-register at sequencer Port }
- MOV AL,04
- OUT DX,AL
- INC DX {read in data via the according data register }
- IN AL,DX
- AND AL,0F7h {bit 3 := 0: don't chain the 4 planes}
- OR AL,04 {bit 2 := 1: no odd/even mechanism }
- OUT DX,AL {activate new settings }
- DEC DX {s.a.: address sequencer reg. 2 (=map-mask),... }
- MOV AL,02
- OUT DX,AL
- INC DX
- MOV AL,0Fh {...and allow access to all 4 bit maps }
- OUT DX,AL
- {starting with segment A000h, set 8000h logical }
- {words = 4*8000h physical words (because of 4 }
- {bitplanes) to 0 }
- {$ifdef FPC_MM_HUGE}
- MOV AX,SEG SegA000
- MOV ES,AX
- MOV ES,ES:[SegA000]
- {$else FPC_MM_HUGE}
- MOV ES, [SegA000]
- {$endif FPC_MM_HUGE}
- XOR DI,DI
- XOR AX,AX
- MOV CX,8000h
- CLD
- REP STOSW
- MOV DX,CRTAddress {address the underline-location-register at }
- MOV AL,14h {the CRT-controller Port, read out the according }
- OUT DX,AL {data register: }
- INC DX
- IN AL,DX
- AND AL,0BFh {bit 6:=0: no double word addressing scheme in}
- OUT DX,AL {video RAM }
- DEC DX
- MOV AL,17h {select mode control register }
- OUT DX,AL
- INC DX
- IN AL,DX
- OR AL,40h {bit 6 := 1: memory access scheme=linear bit array }
- OUT DX,AL
- end;
- {$endif asmgraph}
- {$undef asmgraph}
- {$ifndef asmgraph}
- function GetPixelX(X,Y: smallint): ColorType;
- var offset: word;
- begin
- X := X + StartXViewPort;
- Y := Y + StartYViewPort;
- offset := y * 80 + x shr 2 + VideoOfs;
- PortW[$3ce] := ((x and 3) shl 8) + 4;
- GetPixelX := Mem[SegA000:offset];
- end;
- {$else asmgraph}
- function GetPixelX(X,Y: smallint): ColorType; assembler;
- asm
- {$ifdef FPC_MM_HUGE}
- mov ax, SEG SegA000
- mov es, ax
- mov es, es:[SegA000]
- {$else FPC_MM_HUGE}
- mov es, [SegA000]
- {$endif FPC_MM_HUGE}
- mov di,[Y] ; (* DI = Y coordinate *)
- add di,[StartYViewPort]
- (* Multiply by 80 start *)
- mov cl, 4
- shl di, cl
- mov bx, di
- shl di, 1
- shl di, 1
- add di, bx ; (* Multiply Value by 80 *)
- (* End multiply by 80 *)
- mov cx, [X]
- add cx, [StartXViewPort]
- mov ax, cx
- {DI = Y * LINESIZE, BX = X, coordinates admissible}
- shr ax, 1 ; (* Faster on 286/86 machines *)
- shr ax, 1
- add di, ax ; {DI = Y * LINESIZE + (X SHR 2) }
- add di, [VideoOfs] ; (* Pointing at start of Active page *)
- (* Select plane to use *)
- mov dx, 03c4h
- mov ax, FirstPlane ; (* Map Mask & Plane Select Register *)
- and cl, 03h ; (* Get Plane Bits *)
- shl ah, cl ; (* Get Plane Select Value *)
- out dx, ax
- (* End selection of plane *)
- mov al, ES:[DI]
- xor ah, ah
- {$ifdef FPC_GRAPH_SUPPORTS_TRUECOLOR}
- { 1 byte shorter than 'xor dx, dx'; will always set dx to 0, because sign(ah)=0 }
- cwd
- {$endif FPC_GRAPH_SUPPORTS_TRUECOLOR}
- end;
- {$endif asmgraph}
- procedure SetVisualX(page: word);
- { 4 page supPort... }
- Procedure SetVisibleStart(AOffset: word); Assembler;
- (* Select where the left corner of the screen will be *)
- { By Matt Pritchard }
- asm
- { Wait if we are currently in a Vertical Retrace }
- MOV DX, INPUT_1 { Input Status #1 Register }
- @DP_WAIT0:
- IN AL, DX { Get VGA status }
- AND AL, VERT_RETRACE { In Display mode yet? }
- JNZ @DP_WAIT0 { If Not, wait for it }
- { Set the Start Display Address to the new page }
- MOV DX, CRTC_Index { We Change the VGA Sequencer }
- MOV AL, START_DISP_LO { Display Start Low Register }
- MOV AH, BYTE PTR [AOffset] { Low 8 Bits of Start Addr }
- OUT DX, AX { Set Display Addr Low }
- MOV AL, START_DISP_HI { Display Start High Register }
- MOV AH, BYTE PTR [AOffset+1] { High 8 Bits of Start Addr }
- OUT DX, AX { Set Display Addr High }
- { Wait for a Vertical Retrace to smooth out things }
- MOV DX, INPUT_1 { Input Status #1 Register }
- @DP_WAIT1:
- IN AL, DX { Get VGA status }
- AND AL, VERT_RETRACE { Vertical Retrace Start? }
- JZ @DP_WAIT1 { If Not, wait for it }
- { Now Set Display Starting Address }
- end;
- {$ifdef fpc}
- {$undef asmgraph}
- {$endif fpc}
- begin
- Case page of
- 0: SetVisibleStart(0);
- 1: SetVisibleStart(16000);
- 2: SetVisibleStart(32000);
- 3: SetVisibleStart(48000);
- else
- SetVisibleStart(0);
- end;
- end;
- procedure SetActiveX(page: word);
- { 4 page supPort... }
- begin
- case page of
- 0: VideoOfs := 0;
- 1: VideoOfs := 16000;
- 2: VideoOfs := 32000;
- 3: VideoOfs := 48000;
- else
- VideoOfs:=0;
- end;
- end;
- Procedure PutPixelX(X,Y: smallint; color:ColorType);
- {$ifndef asmgraph}
- var offset: word;
- {$endif asmgraph}
- begin
- { verify clipping and then convert to absolute coordinates...}
- if ClipPixels then
- begin
- if (X < 0) or (X > ViewWidth) then
- exit;
- if (Y < 0) or (Y > ViewHeight) then
- exit;
- end;
- X:= X + StartXViewPort;
- Y:= Y + StartYViewPort;
- {$ifndef asmgraph}
- offset := y * 80 + x shr 2 + VideoOfs;
- PortW[$3c4] := (hi(word(FirstPlane)) shl 8) shl (x and 3)+ lo(word(FirstPlane));
- Mem[SegA000:offset] := color;
- {$else asmgraph}
- asm
- push ax
- push bx
- push cx
- push dx
- push es
- push di
- mov di,[Y] ; (* DI = Y coordinate *)
- (* Multiply by 80 start *)
- mov bx, di
- shl di, 6 ; (* Faster on 286/386/486 machines *)
- shl bx, 4
- add di, bx ; (* Multiply Value by 80 *)
- (* End multiply by 80 *)
- mov cx, [X]
- mov ax, cx
- {DI = Y * LINESIZE, BX = X, coordinates admissible}
- shr ax, 2
- add di, ax ; {DI = Y * LINESIZE + (X SHR 2) }
- add di, [VideoOfs] ; (* Pointing at start of Active page *)
- (* Select plane to use *)
- mov dx, 03c4h
- mov ax, FirstPlane ; (* Map Mask & Plane Select Register *)
- and cl, 03h ; (* Get Plane Bits *)
- shl ah, cl ; (* Get Plane Select Value *)
- out dx, ax
- (* End selection of plane *)
- mov es,[SegA000]
- mov ax,[Color] ; { only lower byte is used. }
- cmp [CurrentWriteMode],XORPut { check write mode }
- jne @MOVMode
- mov ah,es:[di] { read the byte... }
- xor al,ah { xor it and return value into AL }
- @MovMode:
- mov es:[di], al
- pop di
- pop es
- pop dx
- pop cx
- pop bx
- pop ax
- end;
- {$endif asmgraph}
- end;
- Procedure DirectPutPixelX(X,Y: smallint);
- { x,y -> must be in global coordinates. No clipping. }
- {$ifndef asmgraph}
- Var offset: Word;
- dummy: Byte;
- begin
- offset := y * 80 + x shr 2 + VideoOfs;
- case CurrentWriteMode of
- XorPut:
- begin
- PortW[$3ce] := ((x and 3) shl 8) + 4;
- dummy := CurrentColor xor Mem[Sega000: offset];
- end;
- OrPut:
- begin
- PortW[$3ce] := ((x and 3) shl 8) + 4;
- dummy := CurrentColor or Mem[Sega000: offset];
- end;
- AndPut:
- begin
- PortW[$3ce] := ((x and 3) shl 8) + 4;
- dummy := CurrentColor and Mem[Sega000: offset];
- end;
- NotPut: dummy := Not CurrentColor;
- else dummy := CurrentColor;
- end;
- PortW[$3c4] := (hi(word(FirstPlane)) shl 8) shl (x and 3)+ lo(word(FirstPlane));
- Mem[Sega000: offset] := Dummy;
- end;
- {$else asmgraph}
- { note: still needs or/and/notput support !!!!! (JM) }
- Assembler;
- asm
- push ax
- push bx
- push cx
- push dx
- push es
- push di
- {$IFDEF REGCALL}
- mov cl, al
- mov di, dx
- {$ELSE REGCALL}
- mov cx, [X]
- mov ax, cx
- mov di, [Y] ; (* DI = Y coordinate *)
- {$ENDIF REGCALL}
- (* Multiply by 80 start *)
- mov bx, di
- shl di, 6 ; (* Faster on 286/386/486 machines *)
- shl bx, 4
- add di, bx ; (* Multiply Value by 80 *)
- (* End multiply by 80 *)
- {DI = Y * LINESIZE, BX = X, coordinates admissible}
- shr ax, 2
- add di, ax ; {DI = Y * LINESIZE + (X SHR 2) }
- add di, [VideoOfs] ; (* Pointing at start of Active page *)
- (* Select plane to use *)
- mov dx, 03c4h
- mov ax, FirstPlane ; (* Map Mask & Plane Select Register *)
- and cl, 03h ; (* Get Plane Bits *)
- shl ah, cl ; (* Get Plane Select Value *)
- out dx, ax
- (* End selection of plane *)
- mov es,[SegA000]
- mov ax,[CurrentColor] ; { only lower byte is used. }
- cmp [CurrentWriteMode],XORPut { check write mode }
- jne @MOVMode
- mov ah,es:[di] { read the byte... }
- xor al,ah { xor it and return value into AL }
- @MovMode:
- mov es:[di], al
- pop di
- pop es
- pop dx
- pop cx
- pop bx
- pop ax
- end;
- {$endif asmgraph}
- {************************************************************************}
- {* General routines *}
- {************************************************************************}
- var
- SavePtr : pointer; { pointer to video state }
- { CrtSavePtr: pointer;} { pointer to video state when CrtMode gets called }
- StateSize: word; { size in 64 byte blocks for video state }
- VideoMode: byte; { old video mode before graph mode }
- SaveSupPorted : Boolean; { Save/Restore video state supPorted? }
- Procedure SaveStateVGA;
- var
- regs: Registers;
- begin
- SaveSupPorted := FALSE;
- SavePtr := nil;
- { Get the video mode }
- regs.ah:=$0f;
- intr($10,regs);
- VideoMode:=regs.al;
- { saving/restoring video state screws up Windows (JM) }
- if inWindows then
- exit;
- { Prepare to save video state...}
- regs.ax:=$1C00; { get buffer size to save state }
- regs.cx:=%00000111; { Save DAC / Data areas / Hardware states }
- intr($10,regs);
- StateSize:=regs.bx;
- SaveSupPorted:=(regs.al=$1c);
- if SaveSupPorted then
- begin
- GetMem(SavePtr, 64*StateSize); { values returned in 64-byte blocks }
- if not assigned(SavePtr) then
- RunError(203);
- { call the real mode interrupt ... }
- regs.ax := $1C01; { save the state buffer }
- regs.cx := $07; { Save DAC / Data areas / Hardware states }
- regs.es := Seg(SavePtr^);
- regs.bx := Ofs(SavePtr^);
- Intr($10,regs);
- { restore state, according to Ralph Brown Interrupt list }
- { some BIOS corrupt the hardware after a save... }
- regs.ax := $1C02; { restore the state buffer }
- regs.cx := $07; { rest DAC / Data areas / Hardware states }
- regs.es := Seg(SavePtr^);
- regs.bx := Ofs(SavePtr^);
- Intr($10,regs);
- end;
- end;
- procedure RestoreStateVGA;
- var
- regs:Registers;
- SavePtrCopy: Pointer;
- begin
- { go back to the old video mode...}
- regs.ax:=VideoMode;
- intr($10,regs);
- { then restore all state information }
- if assigned(SavePtr) and SaveSupPorted then
- begin
- regs.ax := $1C02; { restore the state buffer }
- regs.cx := $07; { rest DAC / Data areas / Hardware states }
- regs.es := Seg(SavePtr^);
- regs.bx := Ofs(SavePtr^);
- Intr($10,regs);
- SavePtrCopy := SavePtr;
- SavePtr := nil;
- FreeMem(SavePtrCopy, 64*StateSize);
- end;
- end;
- Procedure SetVGARGBAllPalette(const Palette:PaletteType);
- var
- c: byte;
- begin
- { wait for vertical retrace start/end}
- while (port[$3da] and $8) <> 0 do;
- while (port[$3da] and $8) = 0 do;
- If MaxColor = 16 Then
- begin
- for c := 0 to 15 do
- begin
- { translate the color number for 16 color mode }
- portb[$3c8] := toRealCols16[c];
- portb[$3c9] := palette.colors[c].red shr 2;
- portb[$3c9] := palette.colors[c].green shr 2;
- portb[$3c9] := palette.colors[c].blue shr 2;
- end
- end
- else
- begin
- portb[$3c8] := 0;
- for c := 0 to 255 do
- begin
- { no need to set port[$3c8] every time if you set the entries }
- { for successive colornumbers (JM) }
- portb[$3c9] := palette.colors[c].red shr 2;
- portb[$3c9] := palette.colors[c].green shr 2;
- portb[$3c9] := palette.colors[c].blue shr 2;
- end
- end;
- End;
- { VGA is never a direct color mode, so no need to check ... }
- Procedure SetVGARGBPalette(ColorNum, RedValue, GreenValue,
- BlueValue : smallint);
- begin
- { translate the color number for 16 color mode }
- If MaxColor = 16 Then
- ColorNum := ToRealCols16[ColorNum];
- asm
- { on some hardware - there is a snow like effect }
- { when changing the palette register directly }
- { so we wait for a vertical retrace start period. }
- push ax
- push dx
- mov dx, $03da
- @1:
- in al, dx { Get input status register }
- test al, $08 { check if in vertical retrace }
- jnz @1 { yes, complete it }
- { we have to wait for the next }
- { retrace to assure ourselves }
- { that we have time to complete }
- { the DAC operation within }
- { the vertical retrace period }
- @2:
- in al, dx
- test al, $08
- jz @2 { repeat until vertical retrace start }
- mov dx, $03c8 { Set color register address to use }
- mov ax, [ColorNum]
- out dx, al
- inc dx { Point to DAC registers }
- mov ax, [RedValue] { Get RedValue }
- shr ax, 1
- shr ax, 1
- out dx, al
- mov ax, [GreenValue]{ Get RedValue }
- shr ax, 1
- shr ax, 1
- out dx, al
- mov ax, [BlueValue] { Get RedValue }
- shr ax, 1
- shr ax, 1
- out dx, al
- pop dx
- pop ax
- end
- End;
- { VGA is never a direct color mode, so no need to check ... }
- Procedure GetVGARGBPalette(ColorNum: smallint; Var
- RedValue, GreenValue, BlueValue : smallint);
- begin
- If MaxColor = 16 Then
- ColorNum := ToRealCols16[ColorNum];
- Port[$03C7] := ColorNum;
- { we must convert to lsb values... because the vga uses the 6 msb bits }
- { which is not compatible with anything. }
- RedValue := smallint(Port[$3C9]) shl 2;
- GreenValue := smallint(Port[$3C9]) shl 2;
- BlueValue := smallint(Port[$3C9]) shl 2;
- end;
- {************************************************************************}
- {* VESA related routines *}
- {************************************************************************}
- {$I vesa.inc}
- {************************************************************************}
- {* General routines *}
- {************************************************************************}
- procedure CloseGraph;
- Begin
- If not isgraphmode then
- begin
- _graphresult := grnoinitgraph;
- exit
- end;
- if not assigned(RestoreVideoState) then
- RunError(216);
- RestoreVideoState;
- isgraphmode := false;
- end;
- (*
- procedure LoadFont8x8;
- var
- r : registers;
- x,y,c : longint;
- data : array[0..127,0..7] of byte;
- begin
- r.ah:=$11;
- r.al:=$30;
- r.bh:=1;
- RealIntr($10,r);
- dosmemget(r.es,r.bp,data,sizeof(data));
- for c:=0 to 127 do
- for y:=0 to 7 do
- for x:=0 to 7 do
- if (data[c,y] and ($80 shr x))<>0 then
- DefaultFontData[chr(c),y,x]:=1
- else
- DefaultFontData[chr(c),y,x]:=0;
- { second part }
- r.ah:=$11;
- r.al:=$30;
- r.bh:=0;
- RealIntr($10,r);
- dosmemget(r.es,r.bp,data,sizeof(data));
- for c:=0 to 127 do
- for y:=0 to 7 do
- for x:=0 to 7 do
- if (data[c,y] and ($80 shr x))<>0 then
- DefaultFontData[chr(c+128),y,x]:=1
- else
- DefaultFontData[chr(c+128),y,x]:=0;
- 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! }
- function Test6845(CRTCPort: Word): Boolean;
- const
- TestRegister = $0F;
- var
- OldValue, TestValue, ReadValue: Byte;
- begin
- { save the old value }
- Port[CRTCPort] := TestRegister;
- OldValue := Port[CRTCPort + 1];
- TestValue := OldValue xor $56;
- { try writing a new value to the CRTC register }
- Port[CRTCPort] := TestRegister;
- Port[CRTCPort + 1] := TestValue;
- { check if the value has been written }
- Port[CRTCPort] := TestRegister;
- ReadValue := Port[CRTCPort + 1];
- if ReadValue = TestValue then
- begin
- Test6845 := True;
- { restore old value }
- Port[CRTCPort] := TestRegister;
- Port[CRTCPort + 1] := OldValue;
- end
- else
- Test6845 := False;
- end;
- procedure FillCommonCGA320(var mode: TModeInfo);
- begin
- mode.HardwarePages := 0;
- mode.MaxColor := 4;
- mode.PaletteSize := 16;
- mode.DirectColor := FALSE;
- mode.MaxX := 319;
- mode.MaxY := 199;
- mode.DirectPutPixel:=@DirectPutPixelCGA320;
- mode.PutPixel:=@PutPixelCGA320;
- mode.GetPixel:=@GetPixelCGA320;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.HLine := @HLineCGA320;
- mode.SetBkColor := @SetBkColorCGA320;
- mode.GetBkColor := @GetBkColorCGA320;
- mode.XAspect := 8333;
- mode.YAspect := 10000;
- end;
- procedure FillCommonCGA640(var mode: TModeInfo);
- begin
- mode.HardwarePages := 0;
- mode.MaxColor := 2;
- mode.PaletteSize := 16;
- mode.DirectColor := FALSE;
- mode.MaxX := 639;
- mode.MaxY := 199;
- mode.DirectPutPixel:=@DirectPutPixelCGA640;
- mode.PutPixel:=@PutPixelCGA640;
- mode.GetPixel:=@GetPixelCGA640;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.HLine := @HLineCGA640;
- mode.SetBkColor := @SetBkColorCGA640;
- mode.GetBkColor := @GetBkColorCGA640;
- mode.XAspect := 4167;
- mode.YAspect := 10000;
- end;
- procedure FillCommonEGAVGA16(var mode: TModeInfo);
- begin
- mode.MaxColor := 16;
- mode.DirectColor := FALSE;
- mode.PaletteSize := mode.MaxColor;
- mode.DirectPutPixel:=@DirectPutPixel16;
- mode.PutPixel:=@PutPixel16;
- mode.GetPixel:=@GetPixel16;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.HLine := @HLine16;
- mode.VLine := @VLine16;
- mode.GetScanLine := @GetScanLine16;
- end;
- procedure FillCommonVESA16(var mode: TModeInfo);
- begin
- mode.MaxColor := 16;
- { the ModeInfo is automatically set if the mode is supPorted }
- { by the call to SearchVESAMode. }
- mode.HardwarePages := VESAModeInfo.NumberOfPages;
- mode.DirectColor := FALSE;
- mode.PaletteSize := mode.MaxColor;
- mode.DirectPutPixel:=@DirectPutPixVESA16;
- mode.SetRGBPalette := @SetVESARGBPalette;
- mode.GetRGBPalette := @GetVESARGBPalette;
- {$ifdef fpc}
- mode.SetAllPalette := @SetVESARGBAllPalette;
- {$endif fpc}
- mode.PutPixel:=@PutPixVESA16;
- mode.GetPixel:=@GetPixVESA16;
- mode.SetVisualPage := @SetVisualVESA;
- mode.SetActivePage := @SetActiveVESA;
- mode.HLine := @HLineVESA16;
- end;
- procedure FillCommonVESA256(var mode: TModeInfo);
- begin
- mode.MaxColor := 256;
- { the ModeInfo is automatically set if the mode is supPorted }
- { by the call to SearchVESAMode. }
- mode.HardwarePages := VESAModeInfo.NumberOfPages;
- mode.PaletteSize := mode.MaxColor;
- mode.DirectColor := FALSE;
- mode.DirectPutPixel:=@DirectPutPixVESA256;
- mode.PutPixel:=@PutPixVESA256;
- mode.GetPixel:=@GetPixVESA256;
- mode.SetRGBPalette := @SetVESARGBPalette;
- mode.GetRGBPalette := @GetVESARGBPalette;
- {$ifdef fpc}
- mode.SetAllPalette := @SetVESARGBAllPalette;
- {$endif fpc}
- mode.SetVisualPage := @SetVisualVESA;
- mode.SetActivePage := @SetActiveVESA;
- mode.hline := @HLineVESA256;
- mode.vline := @VLineVESA256;
- mode.GetScanLine := @GetScanLineVESA256;
- mode.PatternLine := @PatternLineVESA256;
- end;
- procedure FillCommonVESA32kOr64k(var mode: TModeInfo);
- begin
- { the ModeInfo is automatically set if the mode is supPorted }
- { by the call to SearchVESAMode. }
- mode.HardwarePages := VESAModeInfo.NumberOfPages;
- mode.DirectColor := TRUE;
- mode.DirectPutPixel:=@DirectPutPixVESA32kOr64k;
- mode.PutPixel:=@PutPixVESA32kOr64k;
- mode.GetPixel:=@GetPixVESA32kOr64k;
- mode.SetRGBPalette := @SetVESARGBPalette;
- mode.GetRGBPalette := @GetVESARGBPalette;
- mode.SetVisualPage := @SetVisualVESA;
- mode.SetActivePage := @SetActiveVESA;
- mode.HLine := @HLineVESA32kOr64k;
- end;
- procedure FillCommonVESA32k(var mode: TModeInfo);
- begin
- FillCommonVESA32kOr64k(mode);
- mode.MaxColor := 32768;
- mode.PaletteSize := mode.MaxColor;
- end;
- procedure FillCommonVESA64k(var mode: TModeInfo);
- begin
- FillCommonVESA32kOr64k(mode);
- mode.MaxColor := 65536;
- mode.PaletteSize := mode.MaxColor;
- end;
- procedure FillCommonVESA320x200(var mode: TModeInfo);
- begin
- mode.DriverNumber := VESA;
- mode.ModeName:='320 x 200 VESA';
- mode.MaxX := 319;
- mode.MaxY := 199;
- mode.XAspect := 8333;
- mode.YAspect := 10000;
- end;
- procedure FillCommonVESA640x480(var mode: TModeInfo);
- begin
- mode.DriverNumber := VESA;
- mode.ModeName:='640 x 480 VESA';
- mode.MaxX := 639;
- mode.MaxY := 479;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- end;
- procedure FillCommonVESA800x600(var mode: TModeInfo);
- begin
- mode.DriverNumber := VESA;
- mode.ModeName:='800 x 600 VESA';
- mode.MaxX := 799;
- mode.MaxY := 599;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- end;
- procedure FillCommonVESA1024x768(var mode: TModeInfo);
- begin
- mode.DriverNumber := VESA;
- mode.ModeName:='1024 x 768 VESA';
- mode.MaxX := 1023;
- mode.MaxY := 767;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- end;
- procedure FillCommonVESA1280x1024(var mode: TModeInfo);
- begin
- mode.DriverNumber := VESA;
- mode.ModeName:='1280 x 1024 VESA';
- mode.MaxX := 1279;
- mode.MaxY := 1023;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- end;
- var
- HGCDetected : Boolean = FALSE;
- CGADetected : Boolean = FALSE; { TRUE means real CGA, *not* EGA or VGA }
- EGAColorDetected : Boolean = FALSE; { TRUE means true EGA with a color monitor }
- EGAMonoDetected : Boolean = FALSE; { TRUE means true EGA with a monochrome (MDA) monitor }
- MCGADetected : Boolean = FALSE;
- VGADetected : Boolean = FALSE;
- mode: TModeInfo;
- regs: Registers;
- begin
- QueryAdapterInfo := ModeList;
- { If the mode listing already exists... }
- { simply return it, without changing }
- { anything... }
- if assigned(ModeList) then
- exit;
- { check if VGA/MCGA adapter supported... }
- regs.ax:=$1a00;
- intr($10,regs); { get display combination code...}
- if regs.al=$1a then
- begin
- while regs.bx <> 0 do
- begin
- case regs.bl of
- 1: { monochrome adapter (MDA or HGC) }
- begin
- { check if Hercules adapter supported ... }
- HGCDetected:=Test6845($3B4);
- end;
- 2: CGADetected:=TRUE;
- 4: EGAColorDetected:=TRUE;
- 5: EGAMonoDetected:=TRUE;
- {6: PGA, this is rare stuff, how do we handle it? }
- 7, 8: VGADetected:=TRUE;
- 10, 11, 12: MCGADetected:=TRUE;
- end;
- { check both primary and secondary display adapter }
- regs.bx:=regs.bx shr 8;
- end;
- end;
- if VGADetected then
- begin
- { now check if this is the ATI EGA }
- regs.ax:=$1c00; { get state size for save... }
- { ... all important data }
- regs.cx:=$07;
- intr($10,regs);
- VGADetected:=regs.al=$1c;
- end;
- if not VGADetected and not MCGADetected and
- not EGAColorDetected and not EGAMonoDetected and
- not CGADetected and not HGCDetected then
- begin
- { check if EGA adapter supported... }
- regs.ah:=$12;
- regs.bx:=$FF10;
- intr($10,regs); { get EGA information }
- if regs.bh<>$FF then
- case regs.cl of
- 0..3, { primary: MDA/HGC, secondary: EGA color }
- 6..9: { primary: EGA color, secondary: MDA/HGC (optional) }
- begin
- EGAColorDetected:=TRUE;
- { check if Hercules adapter supported ... }
- HGCDetected:=Test6845($3B4);
- end;
- 4..5, { primary: CGA, secondary: EGA mono }
- 10..11: { primary: EGA mono, secondary: CGA (optional) }
- begin
- EGAMonoDetected:=TRUE;
- { check if CGA adapter supported ... }
- CGADetected := Test6845($3D4);
- end;
- end;
- end;
- { older than EGA? }
- if not VGADetected and not MCGADetected and
- not EGAColorDetected and not EGAMonoDetected and
- not CGADetected and not HGCDetected then
- begin
- { check if Hercules adapter supported ... }
- HGCDetected := Test6845($3B4);
- { check if CGA adapter supported ... }
- CGADetected := Test6845($3D4);
- end;
- {$ifdef logging}
- LogLn('HGC detected: '+strf(Longint(HGCDetected)));
- LogLn('CGA detected: '+strf(Longint(CGADetected)));
- LogLn('EGA color detected: '+strf(Longint(EGAColorDetected)));
- LogLn('EGA mono detected: '+strf(Longint(EGAMonoDetected)));
- LogLn('MCGA detected: '+strf(Longint(MCGADetected)));
- LogLn('VGA detected: '+strf(Longint(VGADetected)));
- {$endif logging}
- if HGCDetected then
- begin
- { HACK:
- until we create Save/RestoreStateHGC, we use Save/RestoreStateVGA
- with the inWindows flag enabled (so we only save the mode number
- and nothing else) }
- if not VGADetected then
- inWindows := true;
- SaveVideoState := @SaveStateVGA;
- RestoreVideoState := @RestoreStateVGA;
- InitMode(mode);
- mode.DriverNumber := HercMono;
- mode.HardwarePages := 1;
- mode.ModeNumber := HercMonoHi;
- mode.ModeName:='720 x 348 HERCULES';
- mode.MaxColor := 2;
- mode.PaletteSize := 16;
- mode.DirectColor := FALSE;
- mode.MaxX := 719;
- mode.MaxY := 347;
- mode.DirectPutPixel:=@DirectPutPixelHGC720;
- mode.PutPixel:=@PutPixelHGC720;
- mode.GetPixel:=@GetPixelHGC720;
- mode.SetRGBPalette := @SetHGCRGBPalette;
- mode.GetRGBPalette := @GetHGCRGBPalette;
- mode.SetVisualPage := @SetVisualHGC720;
- mode.SetActivePage := @SetActiveHGC720;
- mode.InitMode := @InitHGC720;
- mode.HLine := @HLineHGC720;
- mode.SetBkColor := @SetBkColorHGC720;
- mode.GetBkColor := @GetBkColorHGC720;
- mode.XAspect := 7500;
- mode.YAspect := 10000;
- AddMode(mode);
- end;
- if CGADetected or EGAColorDetected or MCGADetected or VGADetected then
- begin
- { HACK:
- until we create Save/RestoreStateCGA, we use Save/RestoreStateVGA
- with the inWindows flag enabled (so we only save the mode number
- and nothing else) }
- if not VGADetected then
- inWindows := true;
- SaveVideoState := @SaveStateVGA;
- RestoreVideoState := @RestoreStateVGA;
- { now add all standard CGA modes... }
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := CGA;
- mode.ModeNumber := CGAC0;
- mode.ModeName:='320 x 200 CGA C0';
- mode.InitMode := @InitCGA320C0;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := CGA;
- mode.ModeNumber := CGAC1;
- mode.ModeName:='320 x 200 CGA C1';
- mode.InitMode := @InitCGA320C1;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := CGA;
- mode.ModeNumber := CGAC2;
- mode.ModeName:='320 x 200 CGA C2';
- mode.InitMode := @InitCGA320C2;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := CGA;
- mode.ModeNumber := CGAC3;
- mode.ModeName:='320 x 200 CGA C3';
- mode.InitMode := @InitCGA320C3;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA640(mode);
- mode.DriverNumber := CGA;
- mode.ModeNumber := CGAHi;
- mode.ModeName:='640 x 200 CGA';
- mode.InitMode := @InitCGA640;
- AddMode(mode);
- end;
- if EGAColorDetected or VGADetected then
- begin
- { HACK:
- until we create Save/RestoreStateEGA, we use Save/RestoreStateVGA
- with the inWindows flag enabled (so we only save the mode number
- and nothing else) }
- if not VGADetected then
- inWindows := true;
- SaveVideoState := @SaveStateVGA;
- RestoreVideoState := @RestoreStateVGA;
- InitMode(mode);
- FillCommonEGAVGA16(mode);
- mode.ModeNumber:=EGALo;
- mode.DriverNumber := EGA;
- mode.ModeName:='640 x 200 EGA';
- mode.MaxX := 639;
- mode.MaxY := 199;
- mode.HardwarePages := 3;
- mode.SetVisualPage := @SetVisual200_350;
- mode.SetActivePage := @SetActive200;
- mode.InitMode := @Init640x200x16;
- mode.XAspect := 4500;
- mode.YAspect := 10000;
- AddMode(mode);
- InitMode(mode);
- FillCommonEGAVGA16(mode);
- mode.ModeNumber:=EGAHi;
- mode.DriverNumber := EGA;
- mode.ModeName:='640 x 350 EGA';
- mode.MaxX := 639;
- mode.MaxY := 349;
- mode.HardwarePages := 1;
- mode.SetVisualPage := @SetVisual200_350;
- mode.SetActivePage := @SetActive350;
- mode.InitMode := @Init640x350x16;
- mode.XAspect := 7750;
- mode.YAspect := 10000;
- AddMode(mode);
- end;
- if MCGADetected or VGADetected then
- begin
- { HACK:
- until we create Save/RestoreStateEGA, we use Save/RestoreStateVGA
- with the inWindows flag enabled (so we only save the mode number
- and nothing else) }
- if not VGADetected then
- inWindows := true;
- SaveVideoState := @SaveStateVGA;
- {$ifdef logging}
- LogLn('Setting VGA SaveVideoState to '+strf(longint(SaveVideoState)));
- {$endif logging}
- RestoreVideoState := @RestoreStateVGA;
- {$ifdef logging}
- LogLn('Setting VGA RestoreVideoState to '+strf(longint(RestoreVideoState)));
- {$endif logging}
- { now add all standard MCGA modes... }
- { yes, most of these are the same as the CGA modes; this is TP7
- compatible }
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := MCGA;
- mode.ModeNumber := MCGAC0;
- mode.ModeName:='320 x 200 CGA C0'; { yes, it says 'CGA' even for the MCGA driver; this is TP7 compatible }
- mode.InitMode := @InitCGA320C0;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := MCGA;
- mode.ModeNumber := MCGAC1;
- mode.ModeName:='320 x 200 CGA C1'; { yes, it says 'CGA' even for the MCGA driver; this is TP7 compatible }
- mode.InitMode := @InitCGA320C1;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := MCGA;
- mode.ModeNumber := MCGAC2;
- mode.ModeName:='320 x 200 CGA C2'; { yes, it says 'CGA' even for the MCGA driver; this is TP7 compatible }
- mode.InitMode := @InitCGA320C2;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA320(mode);
- mode.DriverNumber := MCGA;
- mode.ModeNumber := MCGAC3;
- mode.ModeName:='320 x 200 CGA C3'; { yes, it says 'CGA' even for the MCGA driver; this is TP7 compatible }
- mode.InitMode := @InitCGA320C3;
- AddMode(mode);
- InitMode(mode);
- FillCommonCGA640(mode);
- mode.DriverNumber := MCGA;
- mode.ModeNumber := MCGAMed;
- mode.ModeName:='640 x 200 CGA'; { yes, it says 'CGA' even for the MCGA driver; this is TP7 compatible }
- mode.InitMode := @InitCGA640;
- AddMode(mode);
- InitMode(mode);
- mode.DriverNumber := MCGA;
- mode.HardwarePages := 0;
- mode.ModeNumber := MCGAHi;
- mode.ModeName:='640 x 480 MCGA';
- mode.MaxColor := 2;
- mode.PaletteSize := 16;
- mode.DirectColor := FALSE;
- mode.MaxX := 639;
- mode.MaxY := 479;
- mode.DirectPutPixel:=@DirectPutPixelMCGA640;
- mode.PutPixel:=@PutPixelMCGA640;
- mode.GetPixel:=@GetPixelMCGA640;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.InitMode := @InitMCGA640;
- mode.HLine := @HLineMCGA640;
- mode.SetBkColor := @SetBkColorMCGA640;
- mode.GetBkColor := @GetBkColorMCGA640;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- AddMode(mode);
- InitMode(mode);
- { now add all standard VGA modes... }
- mode.DriverNumber:= LowRes;
- mode.HardwarePages:= 0;
- mode.ModeNumber:=0;
- mode.ModeName:='320 x 200 VGA';
- mode.MaxColor := 256;
- mode.PaletteSize := mode.MaxColor;
- mode.DirectColor := FALSE;
- mode.MaxX := 319;
- mode.MaxY := 199;
- mode.DirectPutPixel:=@DirectPutPixel320;
- mode.PutPixel:=@PutPixel320;
- mode.GetPixel:=@GetPixel320;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.InitMode := @Init320;
- mode.XAspect := 8333;
- mode.YAspect := 10000;
- AddMode(mode);
- end;
- if VGADetected then
- begin
- SaveVideoState := @SaveStateVGA;
- {$ifdef logging}
- LogLn('Setting VGA SaveVideoState to '+strf(longint(SaveVideoState)));
- {$endif logging}
- RestoreVideoState := @RestoreStateVGA;
- {$ifdef logging}
- LogLn('Setting VGA RestoreVideoState to '+strf(longint(RestoreVideoState)));
- {$endif logging}
- { now add all standard VGA modes... }
- InitMode(mode);
- mode.DriverNumber:= LowRes;
- mode.ModeNumber:=1;
- mode.HardwarePages := 3; { 0..3 }
- mode.ModeName:='320 x 200 ModeX';
- mode.MaxColor := 256;
- mode.DirectColor := FALSE;
- mode.PaletteSize := mode.MaxColor;
- mode.MaxX := 319;
- mode.MaxY := 199;
- mode.DirectPutPixel:=@DirectPutPixelX;
- mode.PutPixel:=@PutPixelX;
- mode.GetPixel:=@GetPixelX;
- mode.SetRGBPalette := @SetVGARGBPalette;
- mode.GetRGBPalette := @GetVGARGBPalette;
- mode.SetAllPalette := @SetVGARGBAllPalette;
- mode.SetVisualPage := @SetVisualX;
- mode.SetActivePage := @SetActiveX;
- mode.InitMode := @InitModeX;
- mode.XAspect := 8333;
- mode.YAspect := 10000;
- AddMode(mode);
- InitMode(mode);
- FillCommonEGAVGA16(mode);
- mode.ModeNumber:=VGALo;
- mode.DriverNumber := VGA;
- mode.ModeName:='640 x 200 EGA'; { yes, it says 'EGA' even for the VGA driver; this is TP7 compatible }
- mode.MaxX := 639;
- mode.MaxY := 199;
- mode.HardwarePages := 3;
- mode.SetVisualPage := @SetVisual200_350;
- mode.SetActivePage := @SetActive200;
- mode.InitMode := @Init640x200x16;
- mode.XAspect := 4500;
- mode.YAspect := 10000;
- AddMode(mode);
- InitMode(mode);
- FillCommonEGAVGA16(mode);
- mode.ModeNumber:=VGAMed;
- mode.DriverNumber := VGA;
- mode.ModeName:='640 x 350 EGA'; { yes, it says 'EGA' even for the VGA driver; this is TP7 compatible }
- mode.MaxX := 639;
- mode.MaxY := 349;
- mode.HardwarePages := 1;
- mode.SetVisualPage := @SetVisual200_350;
- mode.SetActivePage := @SetActive350;
- mode.InitMode := @Init640x350x16;
- mode.XAspect := 7750;
- mode.YAspect := 10000;
- AddMode(mode);
- InitMode(mode);
- FillCommonEGAVGA16(mode);
- mode.ModeNumber:=VGAHi;
- mode.DriverNumber := VGA;
- mode.ModeName:='640 x 480 VGA';
- mode.MaxX := 639;
- mode.MaxY := 479;
- mode.HardwarePages := 0;
- mode.InitMode := @Init640x480x16;
- mode.XAspect := 10000;
- mode.YAspect := 10000;
- AddMode(mode);
- end;
- { check if VESA adapter supPorted... }
- {$ifndef noSupPortVESA}
- hasVesa := getVesaInfo(VESAInfo);
- { VBE Version v1.00 is unstable, therefore }
- { only VBE v1.1 and later are supported. }
- if (hasVESA=TRUE) and (VESAInfo.Version <= $0100) then
- hasVESA := False;
- {$else noSupPortVESA}
- hasVESA := false;
- {$endif noSupPortVESA}
- if hasVesa then
- begin
- { We have to set and restore the entire VESA state }
- { otherwise, if we use the VGA BIOS only function }
- { there might be a crash under DPMI, such as in the}
- { ATI Mach64 }
- SaveVideoState := @SaveStateVESA;
- {$ifdef logging}
- LogLn('Setting SaveVideoState to '+strf(longint(SaveVideoState)));
- {$endif logging}
- RestoreVideoState := @RestoreStateVESA;
- {$ifdef logging}
- LogLn('Setting RestoreVideoState to '+strf(longint(RestoreVideoState)));
- {$endif logging}
- { now check all supported modes...}
- if SearchVESAModes(m320x200x32k) then
- begin
- InitMode(mode);
- FillCommonVESA32k(mode);
- FillCommonVESA320x200(mode);
- mode.ModeNumber:=m320x200x32k;
- mode.InitMode := @Init320x200x32k;
- AddMode(mode);
- end;
- if SearchVESAModes(m320x200x64k) then
- begin
- InitMode(mode);
- FillCommonVESA64k(mode);
- FillCommonVESA320x200(mode);
- mode.ModeNumber:=m320x200x64k;
- mode.InitMode := @Init320x200x64k;
- AddMode(mode);
- end;
- if SearchVESAModes(m640x400x256) then
- begin
- InitMode(mode);
- FillCommonVESA256(mode);
- mode.ModeNumber:=m640x400x256;
- mode.DriverNumber := VESA;
- mode.ModeName:='640 x 400 VESA';
- mode.MaxX := 639;
- mode.MaxY := 399;
- mode.InitMode := @Init640x400x256;
- mode.XAspect := 8333;
- mode.YAspect := 10000;
- AddMode(mode);
- end;
- if SearchVESAModes(m640x480x256) then
- begin
- InitMode(mode);
- FillCommonVESA256(mode);
- FillCommonVESA640x480(mode);
- mode.ModeNumber:=m640x480x256;
- mode.InitMode := @Init640x480x256;
- AddMode(mode);
- end;
- if SearchVESAModes(m640x480x32k) then
- begin
- InitMode(mode);
- FillCommonVESA32k(mode);
- FillCommonVESA640x480(mode);
- mode.ModeNumber:=m640x480x32k;
- mode.InitMode := @Init640x480x32k;
- AddMode(mode);
- end;
- if SearchVESAModes(m640x480x64k) then
- begin
- InitMode(mode);
- FillCommonVESA64k(mode);
- FillCommonVESA640x480(mode);
- mode.ModeNumber:=m640x480x64k;
- mode.InitMode := @Init640x480x64k;
- AddMode(mode);
- end;
- if SearchVESAModes(m800x600x16) then
- begin
- InitMode(mode);
- FillCommonVESA16(mode);
- FillCommonVESA800x600(mode);
- mode.ModeNumber:=m800x600x16;
- mode.InitMode := @Init800x600x16;
- AddMode(mode);
- end;
- if SearchVESAModes(m800x600x256) then
- begin
- InitMode(mode);
- FillCommonVESA256(mode);
- FillCommonVESA800x600(mode);
- mode.ModeNumber:=m800x600x256;
- mode.InitMode := @Init800x600x256;
- AddMode(mode);
- end;
- if SearchVESAModes(m800x600x32k) then
- begin
- InitMode(mode);
- FillCommonVESA32k(mode);
- FillCommonVESA800x600(mode);
- mode.ModeNumber:=m800x600x32k;
- mode.InitMode := @Init800x600x32k;
- AddMode(mode);
- end;
- if SearchVESAModes(m800x600x64k) then
- begin
- InitMode(mode);
- FillCommonVESA64k(mode);
- FillCommonVESA800x600(mode);
- mode.ModeNumber:=m800x600x64k;
- mode.InitMode := @Init800x600x64k;
- AddMode(mode);
- end;
- if SearchVESAModes(m1024x768x16) then
- begin
- InitMode(mode);
- FillCommonVESA16(mode);
- FillCommonVESA1024x768(mode);
- mode.ModeNumber:=m1024x768x16;
- mode.InitMode := @Init1024x768x16;
- AddMode(mode);
- end;
- if SearchVESAModes(m1024x768x256) then
- begin
- InitMode(mode);
- FillCommonVESA256(mode);
- FillCommonVESA1024x768(mode);
- mode.ModeNumber:=m1024x768x256;
- mode.InitMode := @Init1024x768x256;
- AddMode(mode);
- end;
- if SearchVESAModes(m1024x768x32k) then
- begin
- InitMode(mode);
- FillCommonVESA32k(mode);
- FillCommonVESA1024x768(mode);
- mode.ModeNumber:=m1024x768x32k;
- mode.InitMode := @Init1024x768x32k;
- AddMode(mode);
- end;
- if SearchVESAModes(m1024x768x64k) then
- begin
- InitMode(mode);
- FillCommonVESA64k(mode);
- FillCommonVESA1024x768(mode);
- mode.ModeNumber:=m1024x768x64k;
- mode.InitMode := @Init1024x768x64k;
- AddMode(mode);
- end;
- if SearchVESAModes(m1280x1024x16) then
- begin
- InitMode(mode);
- FillCommonVESA16(mode);
- FillCommonVESA1280x1024(mode);
- mode.ModeNumber:=m1280x1024x16;
- mode.InitMode := @Init1280x1024x16;
- AddMode(mode);
- end;
- if SearchVESAModes(m1280x1024x256) then
- begin
- InitMode(mode);
- FillCommonVESA256(mode);
- FillCommonVESA1280x1024(mode);
- mode.ModeNumber:=m1280x1024x256;
- mode.InitMode := @Init1280x1024x256;
- AddMode(mode);
- end;
- if SearchVESAModes(m1280x1024x32k) then
- begin
- InitMode(mode);
- FillCommonVESA32k(mode);
- FillCommonVESA1280x1024(mode);
- mode.ModeNumber:=m1280x1024x32k;
- mode.InitMode := @Init1280x1024x32k;
- AddMode(mode);
- end;
- if SearchVESAModes(m1280x1024x64k) then
- begin
- InitMode(mode);
- FillCommonVESA64k(mode);
- FillCommonVESA1280x1024(mode);
- mode.ModeNumber:=m1280x1024x64k;
- mode.InitMode := @Init1280x1024x64k;
- AddMode(mode);
- end;
- end;
- end;
- var
- go32exitsave: codepointer;
- procedure freeSaveStateBuffer;
- begin
- if savePtr <> nil then
- begin
- FreeMem(SavePtr, 64*StateSize);
- SavePtr := nil;
- end;
- exitproc := go32exitsave;
- end;
- var
- regs: Registers;
- begin
- { must be done *before* initialize graph is called, because the save }
- { buffer can be used in the normal exit_proc (which is hooked in }
- { initializegraph and as such executed first) (JM) }
- go32exitsave := exitproc;
- exitproc := @freeSaveStateBuffer;
- { windows screws up the display if the savestate/restore state }
- { stuff is used (or uses an abnormal amount of cpu time after }
- { such a problem has exited), so detect its presense and do not }
- { use those functions if it's running. I'm really tired of }
- { working around Windows bugs :( (JM) }
- regs.ax:=$160a;
- intr($2f,regs);
- inWindows:=regs.ax=0;
- InitializeGraph;
- end.
|