| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155 | {    $Id$    This file is part of the Free Pascal run time library.    and implements some stuff for protected mode programming    Copyright (c) 1993,97 by the Free Pascal development team.    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 go32;{$S-}{no stack check, used by DPMIEXCP !! }{$I os.inc}  interface    const    { contants for the run modes returned by get_run_mode }       rm_unknown = 0;       rm_raw     = 1;     { raw (without HIMEM) }       rm_xms     = 2;     { XMS (for example with HIMEM, without EMM386) }       rm_vcpi    = 3;     { VCPI (for example HIMEM and EMM386) }       rm_dpmi    = 4;     { DPMI (for example DOS box or 386Max) }    { flags }       carryflag     = $001;       parityflag    = $004;       auxcarryflag  = $010;       zeroflag      = $040;       signflag      = $080;       trapflag      = $100;       interruptflag = $200;       directionflag = $400;       overflowflag  = $800;    type       tmeminfo = record          available_memory,          available_pages,          available_lockable_pages,          linear_space,          unlocked_pages,          available_physical_pages,          total_physical_pages,          free_linear_space,          max_pages_in_paging_file,          reserved0,          reserved1,          reserved2 : longint;       end;       tseginfo = record          offset  : pointer;          segment : word;       end;       trealregs = record         case integer of          1: { 32-bit } (EDI, ESI, EBP, Res, EBX, EDX, ECX, EAX: longint;                         Flags, ES, DS, FS, GS, IP, CS, SP, SS: word);          2: { 16-bit } (DI, DI2, SI, SI2, BP, BP2, R1, R2: word;                         BX, BX2, DX, DX2, CX, CX2, AX, AX2: word);          3: { 8-bit }  (stuff: array[1..4] of longint;                         BL, BH, BL2, BH2, DL, DH, DL2, DH2,                         CL, CH, CL2, CH2, AL, AH, AL2, AH2: byte);          4: { Compat } (RealEDI, RealESI, RealEBP, RealRES,                         RealEBX, RealEDX, RealECX, RealEAX: longint;                         RealFlags,                         RealES, RealDS, RealFS, RealGS,                         RealIP, RealCS, RealSP, RealSS: word);       end;      registers = trealregs;    { this works only with real DPMI }    function allocate_ldt_descriptors(count : word) : word;    function free_ldt_descriptor(d : word) : boolean;    function segment_to_descriptor(seg : word) : word;    function get_next_selector_increment_value : word;    function get_segment_base_address(d : word) : longint;    function set_segment_base_address(d : word;s : longint) : boolean;    function set_segment_limit(d : word;s : longint) : boolean;    function create_code_segment_alias_descriptor(seg : word) : word;    function get_linear_addr(phys_addr : longint;size : longint) : longint;    function get_segment_limit(d : word) : longint;    function get_page_size:longint;    function map_device_in_memory_block(handle,offset,pagecount,device:longint):boolean;    function realintr(intnr : word;var regs : trealregs) : boolean;    { is needed for functions which need a real mode buffer }    function global_dos_alloc(bytes : longint) : longint;    function global_dos_free(selector : word) : boolean;    var       { selector for the DOS memory (only usable if in DPMI mode) }       dosmemselector : word;    { this procedure copies data where the source and destination }    { are specified by 48 bit pointers                            }    { Note: the procedure checks only for overlapping if          }    { source selector=destination selector                        }    procedure seg_move(sseg : word;source : longint;dseg : word;dest : longint;count : longint);    { fills a memory area specified by a 48 bit pointer with c }    procedure seg_fillchar(seg : word;ofs : longint;count : longint;c : char);    procedure seg_fillword(seg : word;ofs : longint;count : longint;w : word);    {************************************}    { this works with all PM interfaces: }    {************************************}    function get_meminfo(var meminfo : tmeminfo) : boolean;    function get_pm_interrupt(vector : byte;var intaddr : tseginfo) : boolean;    function set_pm_interrupt(vector : byte;const intaddr : tseginfo) : boolean;    function get_rm_interrupt(vector : byte;var intaddr : tseginfo) : boolean;    function set_rm_interrupt(vector : byte;const intaddr : tseginfo) : boolean;    function get_exception_handler(e : byte;var intaddr : tseginfo) : boolean;    function set_exception_handler(e : byte;const intaddr : tseginfo) : boolean;    function get_pm_exception_handler(e : byte;var intaddr : tseginfo) : boolean;    function set_pm_exception_handler(e : byte;const intaddr : tseginfo) : boolean;    function free_rm_callback(var intaddr : tseginfo) : boolean;    function get_rm_callback(pm_func : pointer;const reg : trealregs;var rmcb : tseginfo) : boolean;    function get_cs : word;    function get_ds : word;    function get_ss : word;    { locking functions }    function allocate_memory_block(size:longint):longint;    function free_memory_block(blockhandle : longint) : boolean;    function request_linear_region(linearaddr, size : longint;                                   var blockhandle : longint) : boolean;    function lock_linear_region(linearaddr, size : longint) : boolean;    function lock_data(var data;size : longint) : boolean;    function lock_code(functionaddr : pointer;size : longint) : boolean;    function unlock_linear_region(linearaddr, size : longint) : boolean;    function unlock_data(var data;size : longint) : boolean;    function unlock_code(functionaddr : pointer;size : longint) : boolean;    { disables and enables interrupts }    procedure disable;    procedure enable;    function inportb(port : word) : byte;    function inportw(port : word) : word;    function inportl(port : word) : longint;    procedure outportb(port : word;data : byte);    procedure outportw(port : word;data : word);    procedure outportl(port : word;data : longint);    function get_run_mode : word;{$ifndef V0_6}    function transfer_buffer : longint;    function tb_size : longint;    procedure copytodos(var addr; len : longint);    procedure copyfromdos(var addr; len : longint);{$endif not VER0_6}    procedure dpmi_dosmemput(seg : word;ofs : word;var data;count : longint);    procedure dpmi_dosmemget(seg : word;ofs : word;var data;count : longint);    procedure dpmi_dosmemmove(sseg,sofs,dseg,dofs : word;count : longint);    procedure dpmi_dosmemfillchar(seg,ofs : word;count : longint;c : char);    procedure dpmi_dosmemfillword(seg,ofs : word;count : longint;w : word);    const       { this procedures are assigned to the procedure which are needed }       { for the current mode to access DOS memory                      }       { It's strongly recommended to use this procedures!              }       dosmemput      : procedure(seg : word;ofs : word;var data;count : longint)=dpmi_dosmemput;       dosmemget      : procedure(seg : word;ofs : word;var data;count : longint)=dpmi_dosmemget;       dosmemmove     : procedure(sseg,sofs,dseg,dofs : word;count : longint)=dpmi_dosmemmove;       dosmemfillchar : procedure(seg,ofs : word;count : longint;c : char)=dpmi_dosmemfillchar;       dosmemfillword : procedure(seg,ofs : word;count : longint;w : word)=dpmi_dosmemfillword;  implementation{$ifndef go32v2}    { the following procedures copy from and to DOS memory without DPMI,      these are not necessary for go32v2, becuase that requires dpmi (PFV) }    procedure raw_dosmemput(seg : word;ofs : word;var data;count : longint);      begin         move(data,pointer($e0000000+seg*16+ofs)^,count);      end;    procedure raw_dosmemget(seg : word;ofs : word;var data;count : longint);      begin         move(pointer($e0000000+seg*16+ofs)^,data,count);      end;    procedure raw_dosmemmove(sseg,sofs,dseg,dofs : word;count : longint);      begin         move(pointer($e0000000+sseg*16+sofs)^,pointer($e0000000+dseg*16+dofs)^,count);      end;    procedure raw_dosmemfillchar(seg,ofs : word;count : longint;c : char);      begin         fillchar(pointer($e0000000+seg*16+ofs)^,count,c);      end;    procedure raw_dosmemfillword(seg,ofs : word;count : longint;w : word);      begin         fillword(pointer($e0000000+seg*16+ofs)^,count,w);      end;{$endif}    { the following procedures copy from and to DOS memory using DPMI }    procedure dpmi_dosmemput(seg : word;ofs : word;var data;count : longint);      begin         seg_move(get_ds,longint(@data),dosmemselector,seg*16+ofs,count);      end;    procedure dpmi_dosmemget(seg : word;ofs : word;var data;count : longint);      begin         seg_move(dosmemselector,seg*16+ofs,get_ds,longint(@data),count);      end;    procedure dpmi_dosmemmove(sseg,sofs,dseg,dofs : word;count : longint);      begin         seg_move(dosmemselector,sseg*16+sofs,dosmemselector,dseg*16+dofs,count);      end;    procedure dpmi_dosmemfillchar(seg,ofs : word;count : longint;c : char);      begin         seg_fillchar(dosmemselector,seg*16+ofs,count,c);      end;    procedure dpmi_dosmemfillword(seg,ofs : word;count : longint;w : word);      begin         seg_fillword(dosmemselector,seg*16+ofs,count,w);      end;    function global_dos_alloc(bytes : longint) : longint;      begin         asm            movl bytes,%ebx            orl  $0x10,%ebx             // round up            shrl $0x4,%ebx              // convert to Paragraphs            movl $0x100,%eax            // function 0x100            int  $0x31            shll $0x10,%eax             // return Segment in hi(Result)            movw %dx,%ax                // return Selector in lo(Result)            movl %eax,__result         end;      end;    function  global_dos_free(selector : word) : boolean;      begin         asm            movw Selector,%dx            movl $0x101,%eax            int  $0x31            setnc %al            movb %al,__RESULT         end;      end;    function realintr(intnr : word;var regs : trealregs) : boolean;      begin         regs.realsp:=0;         regs.realss:=0;         asm            movw  intnr,%bx            xorl  %ecx,%ecx            movl  regs,%edi            { es is always equal ds }            movl  $0x300,%eax            int   $0x31            setnc %al            movb  %al,__RESULT         end;      end;    procedure seg_fillchar(seg : word;ofs : longint;count : longint;c : char);      begin         asm            movl ofs,%edi            movl count,%ecx            movb c,%dl            { load es with selector }            pushw %es            movw seg,%ax            movw %ax,%es            { fill eax with duplicated c }            { so we can use stosl        }            movb %dl,%dh            movw %dx,%ax            shll $16,%eax            movw %dx,%ax            movl %ecx,%edx            shrl $2,%ecx            cld            rep            stosl            movl %edx,%ecx            andl $3,%ecx            rep            stosb            popw %es         end ['EAX','ECX','EDX','EDI'];      end;    procedure seg_fillword(seg : word;ofs : longint;count : longint;w : word);      begin         asm            movl ofs,%edi            movl count,%ecx            movw w,%dx            { load segment }            pushw %es            movw seg,%ax            movw %ax,%es            { fill eax }            movw %dx,%ax            shll $16,%eax            movw %dx,%ax            movl %ecx,%edx            shrl $1,%ecx            cld            rep            stosl            movl %edx,%ecx            andl $1,%ecx            rep            stosw            popw %es         end ['EAX','ECX','EDX','EDI'];      end;    procedure seg_move(sseg : word;source : longint;dseg : word;dest : longint;count : longint);      begin         if count=0 then           exit;         if (sseg<>dseg) or ((sseg=dseg) and (source>dest)) then           asm              pushw %es              pushw %ds              cld              movl count,%ecx              movl source,%esi              movl dest,%edi              movw dseg,%ax              movw %ax,%es              movw sseg,%ax              movw %ax,%ds              movl %ecx,%eax              shrl $2,%ecx              rep              movsl              movl %eax,%ecx              andl $3,%ecx              rep              movsb              popw %ds              popw %es           end ['ESI','EDI','ECX','EAX']         else if (source<dest) then           { copy backward for overlapping }           asm              pushw %es              pushw %ds              std              movl count,%ecx              movl source,%esi              movl dest,%edi              movw dseg,%ax              movw %ax,%es              movw sseg,%ax              movw %ax,%ds              addl %ecx,%esi              addl %ecx,%edi              movl %ecx,%eax              andl $3,%ecx              orl %ecx,%ecx              jz .LSEG_MOVE1              { calculate esi and edi}              decl %esi              decl %edi              rep              movsb              incl %esi              incl %edi           .LSEG_MOVE1:              subl $4,%esi              subl $4,%edi              movl %eax,%ecx              shrl $2,%ecx              rep              movsl              cld              popw %ds              popw %es           end ['ESI','EDI','ECX'];      end;    procedure outportb(port : word;data : byte);      begin         asm            movw port,%dx            movb data,%al            outb %al,%dx         end ['EAX','EDX'];      end;    procedure outportw(port : word;data : word);      begin         asm            movw port,%dx            movw data,%ax            outw %ax,%dx         end ['EAX','EDX'];      end;    procedure outportl(port : word;data : longint);      begin         asm            movw port,%dx            movl data,%eax            outl %eax,%dx         end ['EAX','EDX'];      end;    function inportb(port : word) : byte;      begin         asm            movw port,%dx            inb %dx,%al            movb %al,__RESULT         end ['EAX','EDX'];      end;    function inportw(port : word) : word;      begin         asm            movw port,%dx            inw %dx,%ax            movw %ax,__RESULT         end ['EAX','EDX'];      end;    function inportl(port : word) : longint;      begin         asm            movw port,%dx            inl %dx,%eax            movl %eax,__RESULT         end ['EAX','EDX'];      end;    function get_cs : word;      begin         asm            movw %cs,%ax            movw %ax,__RESULT;         end;      end;    function get_ss : word;      begin         asm            movw %ss,%ax            movw %ax,__RESULT;         end;      end;    function get_ds : word;      begin         asm            movw %ds,%ax            movw %ax,__RESULT;         end;      end;    var       int31error : word;    procedure test_int31(flag : longint);[alias : 'test_int31'];      begin         asm            pushl %ebx            movl  flag,%ebx            testb $1,%bl            jz    1f            movw  %ax,_INT31ERROR            xorl  %eax,%eax            jmp   2f            1:            movl  $1,%eax            2:            popl  %ebx         end;      end;    function set_pm_interrupt(vector : byte;const intaddr : tseginfo) : boolean;      begin         asm            movl intaddr,%eax            movl (%eax),%edx            movw 4(%eax),%cx            movl $0x205,%eax            movb vector,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function set_rm_interrupt(vector : byte;const intaddr : tseginfo) : boolean;      begin         asm            movl intaddr,%eax            movw (%eax),%dx            movw 4(%eax),%cx            movl $0x201,%eax            movb vector,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function set_pm_exception_handler(e : byte;const intaddr : tseginfo) : boolean;      begin         asm            movl intaddr,%eax            movl (%eax),%edx            movw 4(%eax),%cx            movl $0x212,%eax            movb e,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function set_exception_handler(e : byte;const intaddr : tseginfo) : boolean;      begin         asm            movl intaddr,%eax            movl (%eax),%edx            movw 4(%eax),%cx            movl $0x203,%eax            movb e,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function get_pm_exception_handler(e : byte;var intaddr : tseginfo) : boolean;      begin         asm            movl $0x210,%eax            movb e,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT            movl intaddr,%eax            movl %edx,(%eax)            movw %cx,4(%eax)         end;      end;    function get_exception_handler(e : byte;var intaddr : tseginfo) : boolean;      begin         asm            movl $0x202,%eax            movb e,%bl            int $0x31            pushf            call test_int31            movb %al,__RESULT            movl intaddr,%eax            movl %edx,(%eax)            movw %cx,4(%eax)         end;      end;    function get_pm_interrupt(vector : byte;var intaddr : tseginfo) : boolean;      begin         asm            movb vector,%bl            movl $0x204,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT            movl intaddr,%eax            movl %edx,(%eax)            movw %cx,4(%eax)         end;      end;    function get_rm_interrupt(vector : byte;var intaddr : tseginfo) : boolean;      begin         asm            movb vector,%bl            movl $0x200,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT            movl intaddr,%eax            movzwl %dx,%edx            movl %edx,(%eax)            movw %cx,4(%eax)         end;      end;    function free_rm_callback(var intaddr : tseginfo) : boolean;      begin         asm            movl intaddr,%eax            movw (%eax),%dx            movw 4(%eax),%cx            movl $0x304,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    { here we must use ___v2prt0_ds_alias instead of from v2prt0.s    because the exception processor sets the ds limit to $fff    at hardware exceptions }    function get_rm_callback(pm_func : pointer;const reg : trealregs;var rmcb : tseginfo) : boolean;      begin         asm            movl  pm_func,%esi            movl  reg,%edi            pushw %es{$ifdef GO32V2}            movw  ___v2prt0_ds_alias,%ax{$else GO32V2}            movw  %ds,%ax{$endif GO32V2}            movw  %ax,%es            pushw %ds            movw  %cs,%ax            movw  %ax,%ds            movl  $0x303,%eax            int   $0x31            popw  %ds            popw  %es            pushf            call test_int31            movb %al,__RESULT            movl  rmcb,%eax            movzwl %dx,%edx            movl  %edx,(%eax)            movw  %cx,4(%eax)         end;      end;    function allocate_ldt_descriptors(count : word) : word;      begin         asm            movw count,%cx            xorl %eax,%eax            int $0x31            movw %ax,__RESULT         end;      end;    function free_ldt_descriptor(d : word) : boolean;      begin         asm            movw d,%bx            movl $1,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function segment_to_descriptor(seg : word) : word;      begin         asm            movw seg,%bx            movl $2,%eax            int $0x31            movw %ax,__RESULT         end;      end;    function get_next_selector_increment_value : word;      begin         asm            movl $3,%eax            int $0x31            movw %ax,__RESULT         end;      end;    function get_segment_base_address(d : word) : longint;      begin         asm            movw d,%bx            movl $6,%eax            int $0x31            xorl %eax,%eax            movw %dx,%ax            shll $16,%ecx            orl %ecx,%eax            movl %eax,__RESULT         end;      end;    function get_page_size:longint;      begin        asm           movl $0x604,%eax           int $0x31           shll $16,%ebx           movw %cx,%bx           movl %ebx,__RESULT        end;      end;    function request_linear_region(linearaddr, size : longint;                                   var blockhandle : longint) : boolean;      var         pageofs : longint;      begin         pageofs:=linearaddr and $3ff;         linearaddr:=linearaddr-pageofs;         size:=size+pageofs;         asm            movl $0x504,%eax            movl linearaddr,%ebx            movl size,%ecx            movl $1,%edx            xorl %esi,%esi            int $0x31            pushf            call test_int31            movb %al,__RESULT            movl blockhandle,%eax            movl %esi,(%eax)            movl %ebx,pageofs         end;         if pageofs<>linearaddr then           request_linear_region:=false;      end;    function allocate_memory_block(size:longint):longint;      begin        asm          movl  $0x501,%eax          movl  size,%ecx          movl  %ecx,%ebx          shrl  $16,%ebx          andl  $65535,%ecx          int   $0x31          jnc   .Lallocate_mem_block_err          xorl  %ebx,%ebx          xorl  %ecx,%ecx       .Lallocate_mem_block_err:          shll  $16,%ebx          movw  %cx,%bx          shll  $16,%esi          movw  %di,%si          movl  %ebx,__RESULT        end;     end;    function free_memory_block(blockhandle : longint) : boolean;      begin         asm            movl blockhandle,%esi            movl %esi,%edi            shll $16,%esi            movl $0x502,%eax            int  $0x31            call test_int31            movb %al,__RESULT         end;      end;    function lock_linear_region(linearaddr, size : longint) : boolean;      begin          asm            movl  $0x600,%eax            movl  linearaddr,%ecx            movl  %ecx,%ebx            shrl  $16,%ebx            movl  size,%esi            movl  %esi,%edi            shrl  $16,%esi            int   $0x31            pushf            call test_int31            movb %al,__RESULT          end;      end;    function lock_data(var data;size : longint) : boolean;      var         linearaddr : longint;      begin         if get_run_mode <> 4 then           exit;         linearaddr:=longint(@data)+get_segment_base_address(get_ds);         lock_data:=lock_linear_region(linearaddr,size);      end;    function lock_code(functionaddr : pointer;size : longint) : boolean;      var         linearaddr : longint;      begin         if get_run_mode<>rm_dpmi then           exit;         linearaddr:=longint(functionaddr)+get_segment_base_address(get_cs);         lock_code:=lock_linear_region(linearaddr,size);      end;    function unlock_linear_region(linearaddr,size : longint) : boolean;      begin         asm            movl  $0x601,%eax            movl  linearaddr,%ecx            movl  %ecx,%ebx            shrl  $16,%ebx            movl  size,%esi            movl  %esi,%edi            shrl  $16,%esi            int   $0x31            pushf            call  test_int31            movb  %al,__RESULT         end;      end;    function unlock_data(var data;size : longint) : boolean;      var         linearaddr : longint;      begin         if get_run_mode<>rm_dpmi then           exit;         linearaddr:=longint(@data)+get_segment_base_address(get_ds);         unlock_data:=unlock_linear_region(linearaddr,size);      end;    function unlock_code(functionaddr : pointer;size : longint) : boolean;      var         linearaddr : longint;      begin         if get_run_mode <>rm_dpmi then           exit;         linearaddr:=longint(functionaddr)+get_segment_base_address(get_cs);         unlock_code:=unlock_linear_region(linearaddr,size);      end;    function set_segment_base_address(d : word;s : longint) : boolean;      begin         asm            movw d,%bx            leal s,%eax            movw (%eax),%dx            movw 2(%eax),%cx            movl $7,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function set_segment_limit(d : word;s : longint) : boolean;      begin         asm            movw d,%bx            leal s,%eax            movw (%eax),%dx            movw 2(%eax),%cx            movl $8,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function get_segment_limit(d : word) : longint;      begin         asm            movzwl d,%eax            lsl %eax,%eax            jz .L_ok            xorl %eax,%eax         .L_ok:            movl %eax,__RESULT         end;      end;    function create_code_segment_alias_descriptor(seg : word) : word;      begin         asm            movw seg,%bx            movl $0xa,%eax            int $0x31            movw %ax,__RESULT         end;      end;    function get_meminfo(var meminfo : tmeminfo) : boolean;      begin         asm            movl meminfo,%edi            movl $0x500,%eax            int $0x31            pushf            call test_int31            movb %al,__RESULT         end;      end;    function get_linear_addr(phys_addr : longint;size : longint) : longint;      begin         asm            movl phys_addr,%ebx            movl %ebx,%ecx            shrl $16,%ebx            movl size,%esi            movl %esi,%edi            shrl $16,%esi            movl $0x800,%eax            int $0x31            shll $16,%ebx            movw %cx,%bx            movl %ebx,__RESULT         end;      end;    procedure disable;assembler;      asm         cli      end;    procedure enable;assembler;      asm         sti      end;    function get_run_mode : word;      begin         asm            movw _run_mode,%ax            movw %ax,__RESULT         end ['EAX'];      end;    function map_device_in_memory_block(handle,offset,pagecount,device:longint):boolean;      begin         asm           movl device,%edx           movl handle,%esi           xorl %ebx,%ebx           movl pagecount,%ecx           movl $0x0508,%eax           int $0x31           setnc %al           movb %al,__RESULT         end;      end;    function get_core_selector : word;      begin         asm            movw _core_selector,%ax            movw %ax,__RESULT         end;      end;{$ifndef V0_6}    function transfer_buffer : longint;      begin         transfer_buffer := go32_info_block.linear_address_of_transfer_buffer;      end;    function tb_size : longint;      begin         tb_size := go32_info_block.size_of_transfer_buffer;      end;     procedure copytodos(var addr; len : longint);       begin          if len>tb_size then runerror(217);{$ifdef GO32V2}          seg_move(get_ds,longint(@addr),dosmemselector,transfer_buffer,len);{$else GO32V2}          move(addr,pointer(transfer_buffer)^,len);{$endif GO32V2}       end;     procedure copyfromdos(var addr; len : longint);       begin          if len > tb_size then runerror(217);{$ifdef GO32V2}          seg_move(dosmemselector,transfer_buffer,get_ds,longint(@addr),len);{$else GO32V2}          move(pointer(transfer_buffer)^,addr,len);{$endif GO32V2}       end;{$endif not V0_6}begin{$ifndef go32v2}   if not (get_run_mode=rm_dpmi) then     begin        dosmemget:=@raw_dosmemget;        dosmemput:=@raw_dosmemput;        dosmemmove:=@raw_dosmemmove;        dosmemfillchar:=@raw_dosmemfillchar;        dosmemfillword:=@raw_dosmemfillword;     end   else{$endif}     begin       dosmemselector:=get_core_selector;     end;end.{  $Log$  Revision 1.1.1.1  1998-03-25 11:18:41  root  * Restored version  Revision 1.8  1998/03/24 15:54:14  peter    - raw_ functions are not necessary for go32v2, $ifdef'd them  Revision 1.7  1998/03/24 09:33:59  peter    + new trealregs from the mailinglist    + 2 new functions get_page_size, map_device_in_mem_block  Revision 1.6  1998/02/01 09:32:21  florian    * some clean up  Revision 1.5  1998/01/26 11:56:27  michael  + Added log at the end  revision 1.4  date: 1997/12/12 13:14:37;  author: pierre;  state: Exp;  lines: +2 -1     + added handling of swap_vectors if under exceptions       i.e. swapvector is not dummy under go32v2     * bug in output, exceptions where not allways reset correctly       now the code in dpmiexcp is called from v2prt0.as exit routine     * in crt.pp corrected init_delay calibration loop       and added it for go32v2 also (was disabled before due to crashes !!)       the previous code did a wrong assumption on the time need to call       get_ticks compared to an internal loop without call  ----------------------------  revision 1.3  date: 1997/12/11 11:50:37;  author: pierre;  state: Exp;  lines: +2 -2    *  bug in get_linear_addr corrected       thanks to Raul who found this bug.  ----------------------------  revision 1.2  date: 1997/12/01 12:15:46;  author: michael;  state: Exp;  lines: +10 -3  + added copyright reference in header.  ----------------------------  revision 1.1  date: 1997/11/27 08:33:50;  author: michael;  state: Exp;  Initial revision  ----------------------------  revision 1.1.1.1  date: 1997/11/27 08:33:50;  author: michael;  state: Exp;  lines: +0 -0  FPC RTL CVS start  =============================================================================  History:       6th november 1996:         + dosmem* implemented}
 |