| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561 | {    $Id$    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by the Free Pascal development team.    Watcom    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 system;INTERFACE{$ifndef NO_EXCEPTIONS_IN_SYSTEM}{$define EXCEPTIONS_IN_SYSTEM}{$endif NO_EXCEPTIONS_IN_SYSTEM}{ include system-independent routine headers }{$include systemh.inc}{ include heap support headers }{$include heaph.inc}{Platform specific information}type  THandle = Longint;  TThreadID = THandle;  const LineEnding = #13#10;{ LFNSupport is a variable here, defined below!!! } DirectorySeparator = '\'; DriveSeparator = ':'; PathSeparator = ';';{ FileNameCaseSensitive is defined separately below!!! } maxExitCode = 255; MaxPathLen = 256; const{ Default filehandles }  UnusedHandle    = -1;  StdInputHandle  = 0;  StdOutputHandle = 1;  StdErrorHandle  = 2;  FileNameCaseSensitive : boolean = false;  CtrlZMarksEOF: boolean = true; (* #26 is considered as end of file *)  sLineBreak = LineEnding;  DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsCRLF;{ Default memory segments (Tp7 compatibility) }  seg0040 = $0040;  segA000 = $A000;  segB000 = $B000;  segB800 = $B800;var{ Mem[] support }  mem  : array[0..$7fffffff] of byte absolute $0:$0;  memw : array[0..$7fffffff div sizeof(word)] of word absolute $0:$0;  meml : array[0..$7fffffff div sizeof(longint)] of longint absolute $0:$0;{ C-compatible arguments and environment }  argc  : longint;  argv  : ppchar;  envp  : ppchar;  dos_argv0 : pchar;{$ifndef RTLLITE}{ System info }  LFNSupport : boolean;{$ELSE RTLLITE}Const  LFNSupport = false;{$endif RTLLITE}{  necessary for objects.pas, should be removed (at least from the interface  to the implementation)}  type    trealregs=record      realedi,realesi,realebp,realres,      realebx,realedx,realecx,realeax : longint;      realflags,      reales,realds,realfs,realgs,      realip,realcs,realsp,realss  : word;    end;  function  do_write(h:longint;addr:pointer;len : longint) : longint;  function  do_read(h:longint;addr:pointer;len : longint) : longint;  procedure syscopyfromdos(addr : sizeuint; len : longint);  procedure syscopytodos(addr : sizeuint; len : longint);  procedure sysrealintr(intnr : word;var regs : trealregs);  var tb:longint;      tb_segment:word;  const tb_offset=0;        tb_size=8192;IMPLEMENTATION{ include system independent routines }{$include system.inc}const  carryflag = 1;type  tseginfo=packed record    offset  : pointer;    segment : word;  end;var  old_int00 : tseginfo;cvar;  old_int75 : tseginfo;cvar;{$asmmode ATT}{*****************************************************************************                             Watcom Helpers*****************************************************************************}function far_strlen(selector : word;linear_address : sizeuint) : longint;assembler;asm        movl linear_address,%edx        movl %edx,%ecx        movw selector,%gs.Larg19:        movb %gs:(%edx),%al        testb %al,%al        je .Larg20        incl %edx        jmp .Larg19.Larg20:        movl %edx,%eax        subl %ecx,%eaxend;function get_ds : word;assembler;asm        movw    %ds,%axend;function get_cs : word;assembler;asm        movw    %cs,%axend;function dos_selector : word; assembler;asm   movw %ds,%ax  { no separate selector needed }end;procedure alloc_tb; assembler;{ allocate 8kB real mode transfer buffer }asm   pushl %ebx   movw $0x100,%ax   movw $512,%bx   int $0x31   movw %ax,tb_segment   shll $16,%eax   shrl $12,%eax   movl %eax,tb   popl %ebxend;procedure sysseg_move(sseg : word;source : sizeuint;dseg : word;dest : sizeuint;count : longint);begin   if count=0 then     exit;   if (sseg<>dseg) or ((sseg=dseg) and (source>dest)) then     asm        pushl %esi        pushl %edi        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        popl %edi        popl %esi     end   else if (source<dest) then     { copy backward for overlapping }     asm        pushl %esi        pushl %edi        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        popl %edi        popl %esi     end;end;var psp_selector:word; external name 'PSP_SELECTOR';procedure setup_arguments;type  arrayword = array [0..255] of word;var  proxy_s  : string[50];  proxy_argc,proxy_seg,proxy_ofs,lin : longint;  rm_argv  : ^arrayword;  argv0len : longint;  useproxy : boolean;  hp       : ppchar;  doscmd   : string[129];  { Dos commandline copied from PSP, max is 128 chars +1 for terminating zero }  arglen,  count   : longint;  argstart,  pc,arg  : pchar;  quote   : char;  argvlen : longint;  function atohex(s : pchar) : longint;  var    rv : longint;    v  : byte;  begin    rv:=0;    while (s^<>#0) do     begin       v:=byte(s^)-byte('0');       if (v > 9) then         dec(v,7);       v:=v and 15; { in case it's lower case }       rv:=(rv shl 4) or v;       inc(longint(s));     end;    atohex:=rv;  end;  procedure allocarg(idx,len:longint);  var oldargvlen:longint;  begin    if idx>=argvlen then     begin       oldargvlen:=argvlen;       argvlen:=(idx+8) and (not 7);       sysreallocmem(argv,argvlen*sizeof(pointer));       fillchar(argv[oldargvlen],(argvlen-oldargvlen)*sizeof(pointer),0);       argv[idx]:=nil;     end;     ArgV [Idx] := SysAllocMem (Succ (Len));  end;begin  count:=0;  argc:=1;  argv:=nil;  argvlen:=0;  { load commandline from psp }  sysseg_move(psp_selector, 128, get_ds, longint(@doscmd), 128);  doscmd[length(doscmd)+1]:=#0;{$IfDef SYSTEM_DEBUG_STARTUP}  Writeln(stderr,'Dos command line is #',doscmd,'# size = ',length(doscmd));{$EndIf }  { create argv[0] }  argv0len:=strlen(dos_argv0);  allocarg(count,argv0len);  move(dos_argv0^,argv[count]^,argv0len);  inc(count);  { setup cmdline variable }  cmdline:=Getmem(argv0len+length(doscmd)+2);  move(dos_argv0^,cmdline^,argv0len);  cmdline[argv0len]:=' ';  inc(argv0len);  move(doscmd[1],cmdline[argv0len],length(doscmd));  cmdline[argv0len+length(doscmd)+1]:=#0;  { parse dos commandline }  pc:=@doscmd[1];  while pc^<>#0 do   begin     { skip leading spaces }     while pc^ in [#1..#32] do      inc(pc);     if pc^=#0 then      break;     { calc argument length }     quote:=' ';     argstart:=pc;     arglen:=0;     while (pc^<>#0) do      begin        case pc^ of          #1..#32 :            begin              if quote<>' ' then               inc(arglen)              else               break;            end;          '"' :            begin              if quote<>'''' then               begin                 if pchar(pc+1)^<>'"' then                  begin                    if quote='"' then                     quote:=' '                    else                     quote:='"';                  end                 else                  inc(pc);               end              else               inc(arglen);            end;          '''' :            begin              if quote<>'"' then               begin                 if pchar(pc+1)^<>'''' then                  begin                    if quote=''''  then                     quote:=' '                    else                     quote:='''';                  end                 else                  inc(pc);               end              else               inc(arglen);            end;          else            inc(arglen);        end;        inc(pc);      end;     { copy argument }     allocarg(count,arglen);     quote:=' ';     pc:=argstart;     arg:=argv[count];     while (pc^<>#0) do      begin        case pc^ of          #1..#32 :            begin              if quote<>' ' then               begin                 arg^:=pc^;                 inc(arg);               end              else               break;            end;          '"' :            begin              if quote<>'''' then               begin                 if pchar(pc+1)^<>'"' then                  begin                    if quote='"' then                     quote:=' '                    else                     quote:='"';                  end                 else                  inc(pc);               end              else               begin                 arg^:=pc^;                 inc(arg);               end;            end;          '''' :            begin              if quote<>'"' then               begin                 if pchar(pc+1)^<>'''' then                  begin                    if quote=''''  then                     quote:=' '                    else                     quote:='''';                  end                 else                  inc(pc);               end              else               begin                 arg^:=pc^;                 inc(arg);               end;            end;          else            begin              arg^:=pc^;              inc(arg);            end;        end;        inc(pc);      end;     arg^:=#0; {$IfDef SYSTEM_DEBUG_STARTUP}     Writeln(stderr,'dos arg ',count,' #',arglen,'#',argv[count],'#'); {$EndIf SYSTEM_DEBUG_STARTUP}     inc(count);   end;  argc:=count;  { check for !proxy for long commandlines passed using environment }  hp:=envp;  useproxy:=false;  while assigned(hp^) do   begin     if (hp^[0]=' ') then      begin        proxy_s:=strpas(hp^);        if Copy(proxy_s,1,7)=' !proxy' then         begin           proxy_s[13]:=#0;           proxy_s[18]:=#0;           proxy_s[23]:=#0;           argv[2]:=@proxy_s[9];           argv[3]:=@proxy_s[14];           argv[4]:=@proxy_s[19];           useproxy:=true;           break;         end;      end;     inc(hp);   end;  { check for !proxy for long commandlines passed using commandline }  if (not useproxy) and     (argc > 1) and (far_strlen(get_ds,longint(argv[1])) = 6)  then   begin     move(argv[1]^,proxy_s[1],6);     proxy_s[0] := #6;     if (proxy_s = '!proxy') then      useproxy:=true;   end;  { use proxy when found }  if useproxy then   begin     proxy_argc:=atohex(argv[2]);     proxy_seg:=atohex(argv[3]);     proxy_ofs:=atohex(argv[4]);{$IfDef SYSTEM_DEBUG_STARTUP}     Writeln(stderr,'proxy command line found');     writeln(stderr,'argc: ',proxy_argc,' seg: ',proxy_seg,' ofs: ',proxy_ofs);{$EndIf SYSTEM_DEBUG_STARTUP}     rm_argv:=SysGetmem(proxy_argc*sizeof(word));     sysseg_move(dos_selector,proxy_seg*16+proxy_ofs, get_ds,longint(rm_argv),proxy_argc*sizeof(word));     for count:=0 to proxy_argc - 1 do      begin        lin:=proxy_seg*16+rm_argv^[count];        arglen:=far_strlen(dos_selector,lin);        allocarg(count,arglen);        sysseg_move(dos_selector,lin,get_ds,longint(argv[count]),arglen+1);{$IfDef SYSTEM_DEBUG_STARTUP}        Writeln(stderr,'arg ',count,' #',rm_argv^[count],'#',arglen,'#',argv[count],'#');{$EndIf SYSTEM_DEBUG_STARTUP}    end;     SysFreemem(rm_argv);     argc:=proxy_argc;   end;  { create an nil entry }  allocarg(argc,0);  { free unused memory }  sysreallocmem(argv,(argc+1)*sizeof(pointer));end;function strcopy(dest,source : pchar) : pchar;assembler;var  saveeax,saveesi,saveedi : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi{$ifdef REGCALL}        movl    %eax,saveeax        movl    %edx,%edi{$else}        movl    source,%edi{$endif}        testl   %edi,%edi        jz      .LStrCopyDone        leal    3(%edi),%ecx        andl    $-4,%ecx        movl    %edi,%esi        subl    %edi,%ecx{$ifdef REGCALL}        movl    %eax,%edi{$else}        movl    dest,%edi{$endif}        jz      .LStrCopyAligned.LStrCopyAlignLoop:        movb    (%esi),%al        incl    %edi        incl    %esi        testb   %al,%al        movb    %al,-1(%edi)        jz      .LStrCopyDone        decl    %ecx        jnz     .LStrCopyAlignLoop        .balign  16.LStrCopyAligned:        movl    (%esi),%eax        movl    %eax,%edx        leal    0x0fefefeff(%eax),%ecx        notl    %edx        addl    $4,%esi        andl    %edx,%ecx        andl    $0x080808080,%ecx        jnz     .LStrCopyEndFound        movl    %eax,(%edi)        addl    $4,%edi        jmp     .LStrCopyAligned.LStrCopyEndFound:        testl   $0x0ff,%eax        jz      .LStrCopyByte        testl   $0x0ff00,%eax        jz      .LStrCopyWord        testl   $0x0ff0000,%eax        jz      .LStrCopy3Bytes        movl    %eax,(%edi)        jmp     .LStrCopyDone.LStrCopy3Bytes:        xorb     %dl,%dl        movw     %ax,(%edi)        movb     %dl,2(%edi)        jmp     .LStrCopyDone.LStrCopyWord:        movw    %ax,(%edi)        jmp     .LStrCopyDone.LStrCopyByte:        movb    %al,(%edi).LStrCopyDone:{$ifdef REGCALL}        movl    saveeax,%eax{$else}        movl    dest,%eax{$endif}        movl    saveedi,%edi        movl    saveesi,%esiend;var    env_selector:word; external name 'ENV_SELECTOR';    env_size:longint; external name 'ENV_SIZE';procedure setup_environment;var env_count : longint;    dos_env,cp : pchar;begin  env_count:=0;  dos_env:=getmem(env_size);  sysseg_move(env_selector,$0, get_ds, longint(dos_env), env_size);  cp:=dos_env;  while cp ^ <> #0 do    begin    inc(env_count);    while (cp^ <> #0) do inc(longint(cp)); { skip to NUL }    inc(longint(cp)); { skip to next character }    end;  envp := sysgetmem((env_count+1) * sizeof(pchar));  if (envp = nil) then exit;  cp:=dos_env;  env_count:=0;  while cp^ <> #0 do   begin     envp[env_count] := sysgetmem(strlen(cp)+1);     strcopy(envp[env_count], cp);{$IfDef SYSTEM_DEBUG_STARTUP}     Writeln(stderr,'env ',env_count,' = "',envp[env_count],'"');{$EndIf SYSTEM_DEBUG_STARTUP}     inc(env_count);     while (cp^ <> #0) do      inc(longint(cp)); { skip to NUL }     inc(longint(cp)); { skip to next character }   end;  envp[env_count]:=nil;  longint(cp):=longint(cp)+3;  dos_argv0 := sysgetmem(strlen(cp)+1);  if (dos_argv0 = nil) then halt;  strcopy(dos_argv0, cp);end;procedure syscopytodos(addr : sizeuint; len : longint);begin   if len > tb_size then     HandleError(217);   sysseg_move(get_ds,addr,dos_selector,tb,len);end;procedure syscopyfromdos(addr : sizeuint; len : longint);begin   if len > tb_size then     HandleError(217);   sysseg_move(dos_selector,tb,get_ds,addr,len);end;procedure sysrealintr(intnr : word;var regs : trealregs);begin   regs.realsp:=0;   regs.realss:=0;   asm      pushl %edi      pushl %ebx      pushw %fs      movw  intnr,%bx      xorl  %ecx,%ecx      movl  regs,%edi      movw  $0x300,%ax      int   $0x31      popw  %fs      popl %ebx      popl %edi   end;end;procedure set_pm_interrupt(vector : byte;const intaddr : tseginfo);begin  asm        pushl %ebx        movl intaddr,%eax        movl (%eax),%edx        movw 4(%eax),%cx        movl $0x205,%eax        movb vector,%bl        int $0x31        popl %ebx  end;end;procedure get_pm_interrupt(vector : byte;var intaddr : tseginfo);begin  asm        pushl %ebx        movb    vector,%bl        movl    $0x204,%eax        int     $0x31        movl    intaddr,%eax        movl    %edx,(%eax)        movw    %cx,4(%eax)        popl %ebx  end;end;procedure getinoutres(def : word);var  regs : trealregs;begin  regs.realeax:=$5900;  regs.realebx:=$0;  sysrealintr($21,regs);  InOutRes:=lo(regs.realeax);  case InOutRes of   19 : InOutRes:=150;   21 : InOutRes:=152;   32 : InOutRes:=5;  end;  if InOutRes=0 then    InOutRes:=Def;end;   { Keep Track of open files }   const      max_files = 50;   var      openfiles : array [0..max_files-1] of boolean;{$ifdef SYSTEMDEBUG}      opennames : array [0..max_files-1] of pchar;   const      free_closed_names : boolean = true;{$endif SYSTEMDEBUG}{*****************************************************************************                         System Dependent Exit code*****************************************************************************}procedure ___exit(exitcode:longint);cdecl;external name '___exit';procedure do_close(handle : longint);forward;Procedure system_exit;var  h : byte;begin  for h:=0 to max_files-1 do    if openfiles[h] then      begin{$ifdef SYSTEMDEBUG}         writeln(stderr,'file ',opennames[h],' not closed at exit');{$endif SYSTEMDEBUG}         if h>=5 then           do_close(h);      end;  { halt is not allways called !! }  { not on normal exit !! PM }  set_pm_interrupt($00,old_int00);{$ifndef EXCEPTIONS_IN_SYSTEM}  set_pm_interrupt($75,old_int75);{$endif EXCEPTIONS_IN_SYSTEM}  ___exit(exitcode);end;procedure new_int00;begin  HandleError(200);end;{$ifndef EXCEPTIONS_IN_SYSTEM}procedure new_int75;begin  asm        xorl    %eax,%eax        outb    %al,$0x0f0        movb    $0x20,%al        outb    %al,$0x0a0        outb    %al,$0x020  end;  HandleError(200);end;{$endif EXCEPTIONS_IN_SYSTEM}var  __stkbottom : pointer;//###########external name '__stkbottom';{*****************************************************************************                              ParamStr/Randomize*****************************************************************************}function paramcount : longint;begin  paramcount := argc - 1;end;function paramstr(l : longint) : string;begin  if (l>=0) and (l+1<=argc) then   paramstr:=strpas(argv[l])  else   paramstr:='';end;procedure randomize;var  hl   : longint;  regs : trealregs;begin  regs.realeax:=$2c00;  sysrealintr($21,regs);  hl:=lo(regs.realedx);  randseed:=hl*$10000+ lo(regs.realecx);end;{*****************************************************************************      OS Memory allocation / deallocation ****************************************************************************}function ___sbrk(size:longint):pointer;cdecl; external name '___sbrk';function SysOSAlloc(size: ptrint): pointer;assembler;asm{$ifdef SYSTEMDEBUG}        cmpb    $1,accept_sbrk        je      .Lsbrk        movl    $0,%eax        jmp     .Lsbrk_fail      .Lsbrk:{$endif}        movl    size,%eax        pushl   %eax        call    ___sbrk        addl    $4,%esp{$ifdef SYSTEMDEBUG}      .Lsbrk_fail:{$endif}end;{ define HAS_SYSOSFREE}procedure SysOSFree(p: pointer; size: ptrint);beginend;{ include standard heap management }{$include heap.inc}{****************************************************************************                        Low level File Routines ****************************************************************************}procedure AllowSlash(p:pchar);var  i : longint;begin{ allow slash as backslash }  for i:=0 to strlen(p) do   if p[i]='/' then p[i]:='\';end;procedure do_close(handle : longint);var  regs : trealregs;begin  if Handle<=4 then   exit;  regs.realebx:=handle;  if handle<max_files then    begin       openfiles[handle]:=false;{$ifdef SYSTEMDEBUG}       if assigned(opennames[handle]) and free_closed_names then         begin            sysfreememsize(opennames[handle],strlen(opennames[handle])+1);            opennames[handle]:=nil;         end;{$endif SYSTEMDEBUG}    end;  regs.realeax:=$3e00;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;procedure do_erase(p : pchar);var  regs : trealregs;begin  AllowSlash(p);  syscopytodos(longint(p),strlen(p)+1);  regs.realedx:=tb_offset;  regs.realds:=tb_segment;  if LFNSupport then   regs.realeax:=$7141  else   regs.realeax:=$4100;  regs.realesi:=0;  regs.realecx:=0;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;procedure do_rename(p1,p2 : pchar);var  regs : trealregs;begin  AllowSlash(p1);  AllowSlash(p2);  if strlen(p1)+strlen(p2)+3>tb_size then   HandleError(217);  sysseg_move(get_ds,sizeuint(p2),dos_selector,tb,strlen(p2)+1);  sysseg_move(get_ds,sizeuint(p1),dos_selector,tb+strlen(p2)+2,strlen(p1)+1);  regs.realedi:=tb_offset;  regs.realedx:=tb_offset + strlen(p2)+2;  regs.realds:=tb_segment;  regs.reales:=tb_segment;  if LFNSupport then   regs.realeax:=$7156  else   regs.realeax:=$5600;  regs.realecx:=$ff;            { attribute problem here ! }  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;function do_write(h:longint;addr:pointer;len : longint) : longint;var  regs      : trealregs;  size,  writesize : longint;begin  writesize:=0;  while len > 0 do   begin     if len>tb_size then      size:=tb_size     else      size:=len;     syscopytodos(ptrint(addr)+writesize,size);     regs.realecx:=size;     regs.realedx:=tb_offset;     regs.realds:=tb_segment;     regs.realebx:=h;     regs.realeax:=$4000;     sysrealintr($21,regs);     if (regs.realflags and carryflag) <> 0 then      begin        GetInOutRes(lo(regs.realeax));        exit(writesize);      end;     inc(writesize,lo(regs.realeax));     dec(len,lo(regs.realeax));     { stop when not the specified size is written }     if lo(regs.realeax)<size then      break;   end;  Do_Write:=WriteSize;end;function do_read(h:longint;addr:pointer;len : longint) : longint;var  regs     : trealregs;  size,  readsize : longint;begin  readsize:=0;  while len > 0 do   begin     if len>tb_size then      size:=tb_size     else      size:=len;     regs.realecx:=size;     regs.realedx:=tb_offset;     regs.realds:=tb_segment;     regs.realebx:=h;     regs.realeax:=$3f00;     sysrealintr($21,regs);     if (regs.realflags and carryflag) <> 0 then      begin        GetInOutRes(lo(regs.realeax));        do_read:=0;        exit;      end;     syscopyfromdos(ptrint(addr)+readsize,lo(regs.realeax));     inc(readsize,lo(regs.realeax));     dec(len,lo(regs.realeax));     { stop when not the specified size is read }     if lo(regs.realeax)<size then      break;   end;  do_read:=readsize;end;function do_filepos(handle : longint) : longint;var  regs : trealregs;begin  regs.realebx:=handle;  regs.realecx:=0;  regs.realedx:=0;  regs.realeax:=$4201;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   Begin     GetInOutRes(lo(regs.realeax));     do_filepos:=0;   end  else   do_filepos:=lo(regs.realedx) shl 16+lo(regs.realeax);end;procedure do_seek(handle,pos : longint);var  regs : trealregs;begin  regs.realebx:=handle;  regs.realecx:=pos shr 16;  regs.realedx:=pos and $ffff;  regs.realeax:=$4200;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;function do_seekend(handle:longint):longint;var  regs : trealregs;begin  regs.realebx:=handle;  regs.realecx:=0;  regs.realedx:=0;  regs.realeax:=$4202;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   Begin     GetInOutRes(lo(regs.realeax));     do_seekend:=0;   end  else   do_seekend:=lo(regs.realedx) shl 16+lo(regs.realeax);end;function do_filesize(handle : longint) : longint;var  aktfilepos : longint;begin  aktfilepos:=do_filepos(handle);  do_filesize:=do_seekend(handle);  do_seek(handle,aktfilepos);end;{ truncate at a given position }procedure do_truncate (handle,pos:longint);var  regs : trealregs;begin  do_seek(handle,pos);  regs.realecx:=0;  regs.realedx:=tb_offset;  regs.realds:=tb_segment;  regs.realebx:=handle;  regs.realeax:=$4000;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;const  FileHandleCount : longint = 20;function Increase_file_handle_count : boolean;var  regs : trealregs;begin  Inc(FileHandleCount,10);  regs.realebx:=FileHandleCount;  regs.realeax:=$6700;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   begin    Increase_file_handle_count:=false;    Dec (FileHandleCount, 10);   end  else    Increase_file_handle_count:=true;end;function dos_version : word;var  regs   : trealregs;begin  regs.realeax := $3000;  sysrealintr($21,regs);  dos_version := regs.realeaxend;procedure do_open(var f;p:pchar;flags:longint);{  filerec and textrec have both handle and mode as the first items so  they could use the same routine for opening/creating.  when (flags and $100)   the file will be append  when (flags and $1000)  the file will be truncate/rewritten  when (flags and $10000) there is no check for close (needed for textfiles)}var  regs   : trealregs;  action : longint;  Avoid6c00 : boolean;begin  AllowSlash(p);{ check if Extended Open/Create API is safe to use }  Avoid6c00 := lo(dos_version) < 7;{ close first if opened }  if ((flags and $10000)=0) then   begin     case filerec(f).mode of      fminput,fmoutput,fminout : Do_Close(filerec(f).handle);      fmclosed : ;     else      begin        inoutres:=102; {not assigned}        exit;      end;     end;   end;{ reset file handle }  filerec(f).handle:=UnusedHandle;  action:=$1;{ convert filemode to filerec modes }  case (flags and 3) of   0 : filerec(f).mode:=fminput;   1 : filerec(f).mode:=fmoutput;   2 : filerec(f).mode:=fminout;  end;  if (flags and $1000)<>0 then   action:=$12; {create file function}{ empty name is special }  if p[0]=#0 then   begin     case FileRec(f).mode of       fminput :         FileRec(f).Handle:=StdInputHandle;       fminout, { this is set by rewrite }       fmoutput :         FileRec(f).Handle:=StdOutputHandle;       fmappend :         begin           FileRec(f).Handle:=StdOutputHandle;           FileRec(f).mode:=fmoutput; {fool fmappend}         end;     end;     exit;   end;{ real dos call }  syscopytodos(longint(p),strlen(p)+1);{$ifndef RTLLITE}  if LFNSupport then   regs.realeax := $716c                           { Use LFN Open/Create API }  else   regs.realeax:=$6c00;{$endif RTLLITE}   if Avoid6c00 then     regs.realeax := $3d00 + (flags and $ff)      { For now, map to Open API }   else     regs.realeax := $6c00;                   { Use Extended Open/Create API }  if byte(regs.realeax shr 8) = $3d then    begin  { Using the older Open or Create API's }      if (action and $00f0) <> 0 then        regs.realeax := $3c00;                   { Map to Create/Replace API }      regs.realds := tb_segment;      regs.realedx := tb_offset;    end  else    begin  { Using LFN or Extended Open/Create API }      regs.realedx := action;            { action if file does/doesn't exist }      regs.realds := tb_segment;      regs.realesi := tb_offset;      regs.realebx := $2000 + (flags and $ff);              { file open mode }    end;  regs.realecx := $20;                                     { file attributes }  sysrealintr($21,regs);{$ifndef RTLLITE}  if (regs.realflags and carryflag) <> 0 then    if lo(regs.realeax)=4 then      if Increase_file_handle_count then        begin          { Try again }          if LFNSupport then            regs.realeax := $716c                    {Use LFN Open/Create API}          else            if Avoid6c00 then              regs.realeax := $3d00+(flags and $ff) {For now, map to Open API}            else              regs.realeax := $6c00;            {Use Extended Open/Create API}          if byte(regs.realeax shr 8) = $3d then            begin  { Using the older Open or Create API's }              if (action and $00f0) <> 0 then                regs.realeax := $3c00;             {Map to Create/Replace API}              regs.realds := tb_segment;              regs.realedx := tb_offset;            end          else            begin  { Using LFN or Extended Open/Create API }              regs.realedx := action;      {action if file does/doesn't exist}              regs.realds := tb_segment;              regs.realesi := tb_offset;              regs.realebx := $2000+(flags and $ff);          {file open mode}            end;          regs.realecx := $20;                               {file attributes}          sysrealintr($21,regs);        end;{$endif RTLLITE}  if (regs.realflags and carryflag) <> 0 then    begin      GetInOutRes(lo(regs.realeax));      exit;    end  else    begin      filerec(f).handle:=lo(regs.realeax);{$ifndef RTLLITE}      { for systems that have more then 20 by default ! }      if lo(regs.realeax)>FileHandleCount then        FileHandleCount:=lo(regs.realeax);{$endif RTLLITE}    end;  if lo(regs.realeax)<max_files then    begin{$ifdef SYSTEMDEBUG}       if openfiles[lo(regs.realeax)] and          assigned(opennames[lo(regs.realeax)]) then         begin            Writeln(stderr,'file ',opennames[lo(regs.realeax)],'(',lo(regs.realeax),') not closed but handle reused!');            sysfreememsize(opennames[lo(regs.realeax)],strlen(opennames[lo(regs.realeax)])+1);         end;{$endif SYSTEMDEBUG}       openfiles[lo(regs.realeax)]:=true;{$ifdef SYSTEMDEBUG}       opennames[lo(regs.realeax)] := sysgetmem(strlen(p)+1);       move(p^,opennames[lo(regs.realeax)]^,strlen(p)+1);{$endif SYSTEMDEBUG}    end;{ append mode }  if ((flags and $100) <> 0) and   (FileRec (F).Handle <> UnusedHandle) then   begin     do_seekend(filerec(f).handle);     filerec(f).mode:=fmoutput; {fool fmappend}   end;end;function do_isdevice(handle:THandle):boolean;var  regs : trealregs;begin  regs.realebx:=handle;  regs.realeax:=$4400;  sysrealintr($21,regs);  do_isdevice:=(regs.realedx and $80)<>0;  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;{*****************************************************************************                           UnTyped File Handling*****************************************************************************}{$i file.inc}{*****************************************************************************                           Typed File Handling*****************************************************************************}{$i typefile.inc}{*****************************************************************************                           Text File Handling*****************************************************************************}{$i text.inc}{*****************************************************************************                           Generic Handling*****************************************************************************}{$ifdef TEST_GENERIC}{$i generic.inc}{$endif TEST_GENERIC}{*****************************************************************************                           Directory Handling*****************************************************************************}procedure DosDir(func:byte;const s:string);var  buffer : array[0..255] of char;  regs   : trealregs;begin  move(s[1],buffer,length(s));  buffer[length(s)]:=#0;  AllowSlash(pchar(@buffer));  { True DOS does not like backslashes at end    Win95 DOS accepts this !!    but "\" and "c:\" should still be kept and accepted hopefully PM }  if (length(s)>0) and (buffer[length(s)-1]='\') and     Not ((length(s)=1) or ((length(s)=3) and (s[2]=':'))) then    buffer[length(s)-1]:=#0;  syscopytodos(longint(@buffer),length(s)+1);  regs.realedx:=tb_offset;  regs.realds:=tb_segment;  if LFNSupport then   regs.realeax:=$7100+func  else   regs.realeax:=func shl 8;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   GetInOutRes(lo(regs.realeax));end;procedure mkdir(const s : string);[IOCheck];begin  If (s='') or (InOutRes <> 0) then   exit;  DosDir($39,s);end;procedure rmdir(const s : string);[IOCheck];begin  if (s = '.' ) then    InOutRes := 16;  If (s='') or (InOutRes <> 0) then   exit;  DosDir($3a,s);end;procedure chdir(const s : string);[IOCheck];var  regs : trealregs;begin  If (s='') or (InOutRes <> 0) then   exit;{ First handle Drive changes }  if (length(s)>=2) and (s[2]=':') then   begin     regs.realedx:=(ord(s[1]) and (not 32))-ord('A');     regs.realeax:=$0e00;     sysrealintr($21,regs);     regs.realeax:=$1900;     sysrealintr($21,regs);     if byte(regs.realeax)<>byte(regs.realedx) then      begin        Inoutres:=15;        exit;      end;     { DosDir($3b,'c:') give Path not found error on       pure DOS PM }     if length(s)=2 then       exit;   end;{ do the normal dos chdir }  DosDir($3b,s);end;procedure getdir(drivenr : byte;var dir : shortstring);var  temp : array[0..255] of char;  i    : longint;  regs : trealregs;begin  regs.realedx:=drivenr;  regs.realesi:=tb_offset;  regs.realds:=tb_segment;  if LFNSupport then   regs.realeax:=$7147  else   regs.realeax:=$4700;  sysrealintr($21,regs);  if (regs.realflags and carryflag) <> 0 then   Begin     GetInOutRes(lo(regs.realeax));     Dir := char (DriveNr + 64) + ':\';     exit;   end  else   syscopyfromdos(longint(@temp),251);{ conversion to Pascal string including slash conversion }  i:=0;  while (temp[i]<>#0) do   begin     if temp[i]='/' then      temp[i]:='\';     dir[i+4]:=temp[i];     inc(i);   end;  dir[2]:=':';  dir[3]:='\';  dir[0]:=char(i+3);{ upcase the string }  if not FileNameCaseSensitive then   dir:=upcase(dir);  if drivenr<>0 then   { Drive was supplied. We know it }   dir[1]:=char(65+drivenr-1)  else   begin   { We need to get the current drive from DOS function 19H  }   { because the drive was the default, which can be unknown }     regs.realeax:=$1900;     sysrealintr($21,regs);     i:= (regs.realeax and $ff) + ord('A');     dir[1]:=chr(i);   end;end;{*****************************************************************************                         SystemUnit Initialization*****************************************************************************}function CheckLFN:boolean;var  regs     : TRealRegs;  RootName : pchar;begin{ Check LFN API on drive c:\ }  RootName:='C:\';  syscopytodos(longint(RootName),strlen(RootName)+1);{ Call 'Get Volume Information' ($71A0) }  regs.realeax:=$71a0;  regs.reales:=tb_segment;  regs.realedi:=tb_offset;  regs.realecx:=32;  regs.realds:=tb_segment;  regs.realedx:=tb_offset;  regs.realflags:=carryflag;  sysrealintr($21,regs);{ If carryflag=0 and LFN API bit in ebx is set then use Long file names }  CheckLFN:=(regs.realflags and carryflag=0) and (regs.realebx and $4000=$4000);end;{$ifdef  EXCEPTIONS_IN_SYSTEM}{$define IN_SYSTEM}{$i dpmiexcp.pp}{$endif  EXCEPTIONS_IN_SYSTEM}procedure SysInitStdIO;begin  OpenStdIO(Input,fmInput,StdInputHandle);  OpenStdIO(Output,fmOutput,StdOutputHandle);  OpenStdIO(ErrOutput,fmOutput,StdErrorHandle);  OpenStdIO(StdOut,fmOutput,StdOutputHandle);  OpenStdIO(StdErr,fmOutput,StdErrorHandle);end;function GetProcessID: SizeUInt;begin GetProcessID := 1;end;var  temp_int : tseginfo;Begin  alloc_tb;  StackLength := InitialStkLen;  StackBottom := __stkbottom;  { To be set if this is a GUI or console application }  IsConsole := TRUE;  { To be set if this is a library and not a program  }  IsLibrary := FALSE;{ save old int 0 and 75 }  get_pm_interrupt($00,old_int00);  get_pm_interrupt($75,old_int75);  temp_int.segment:=get_cs;  temp_int.offset:=@new_int00;  set_pm_interrupt($00,temp_int);{$ifndef EXCEPTIONS_IN_SYSTEM}  temp_int.offset:=@new_int75;  set_pm_interrupt($75,temp_int);{$endif EXCEPTIONS_IN_SYSTEM}{ Setup heap }  InitHeap;  SysInitExceptions;{ Setup stdin, stdout and stderr }  SysInitStdIO;{ Setup environment and arguments }  Setup_Environment;  Setup_Arguments;{ Use LFNSupport LFN }  LFNSupport:=CheckLFN;  if LFNSupport then   FileNameCaseSensitive:=true;{ Reset IO Error }  InOutRes:=0;  ThreadID := 1;{$ifdef  EXCEPTIONS_IN_SYSTEM}  InitDPMIExcp;  InstallDefaultHandlers;{$endif  EXCEPTIONS_IN_SYSTEM}{$ifdef HASVARIANT}  initvariantmanager;{$endif HASVARIANT}{$ifdef HASWIDESTRING}  initwidestringmanager;{$endif HASWIDESTRING}End.{  $Log$  Revision 1.24  2005-05-12 20:29:04  michael  + Added maxpathlen constant (maximum length of filename path)  Revision 1.23  2005/05/10 21:45:08  hajny    * fix for potential SIGSEGV during argv allocation  Revision 1.22  2005/04/13 20:10:50  florian    + TThreadID  Revision 1.21  2005/04/03 21:10:59  hajny    * EOF_CTRLZ conditional define replaced with CtrlZMarksEOF, #26 handling made more consistent (fix for bug 2453)  Revision 1.20  2005/02/14 17:13:32  peter    * truncate log  Revision 1.19  2005/02/01 20:22:50  florian    * improved widestring infrastructure manager}
 |