| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470 | {    Copyright (c) 1998-2002 by Florian Klaempfl    This unit implements some support functions    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published    by the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    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.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.****************************************************************************}{# This unit contains some generic support functions which are used   in the different parts of the compiler.}unit cutils;{$i fpcdefs.inc}interface  type    Tcharset=set of char;  var    internalerrorproc : procedure(i:longint);    {# Returns the minimal value between @var(a) and @var(b) }    function min(a,b : longint) : longint;{$ifdef USEINLINE}inline;{$endif}    function min(a,b : int64) : int64;{$ifdef USEINLINE}inline;{$endif}    {# Returns the maximum value between @var(a) and @var(b) }    function max(a,b : longint) : longint;{$ifdef USEINLINE}inline;{$endif}    function max(a,b : int64) : int64;{$ifdef USEINLINE}inline;{$endif}    {# Return value @var(i) aligned on @var(a) boundary }    function align(i,a:longint):longint;{$ifdef USEINLINE}inline;{$endif}    { if you have an address aligned using "oldalignment" and add an      offset of (a multiple of) offset to it, this function calculates      the new minimally guaranteed alignment    }    function newalignment(oldalignment: longint; offset: int64): longint;    {# Return @var(b) with the bit order reversed }    function reverse_byte(b: byte): byte;    function next_prime(l: longint): longint;    function used_align(varalign,minalign,maxalign:shortint):shortint;    function isbetteralignedthan(new, org, limit: cardinal): boolean;    function size_2_align(len : longint) : shortint;    function packedbitsloadsize(bitlen: int64) : int64;    procedure Replace(var s:string;s1:string;const s2:string);    procedure Replace(var s:AnsiString;s1:string;const s2:AnsiString);    procedure ReplaceCase(var s:string;const s1,s2:string);    procedure ReplaceCase(var s:ansistring;const s1,s2:ansistring);    Function MatchPattern(const pattern,what:string):boolean;    function upper(const c : char) : char;    function upper(const s : string) : string;    function upper(const s : ansistring) : ansistring;    function lower(const c : char) : char;    function lower(const s : string) : string;    function lower(const s : ansistring) : ansistring;    function rpos(const needle: char; const haystack: shortstring): longint; overload;    function rpos(const needle: shortstring; const haystack: shortstring): longint; overload;    function trimbspace(const s:string):string;    function trimspace(const s:string):string;    function space (b : longint): string;    function PadSpace(const s:string;len:longint):string;    function GetToken(var s:string;endchar:char):string;    procedure uppervar(var s : string);    function realtostr(e:extended):string;{$ifdef USEINLINE}inline;{$endif}    function tostr(i : qword) : string;{$ifdef USEINLINE}inline;{$endif}overload;    function tostr(i : int64) : string;{$ifdef USEINLINE}inline;{$endif}overload;    function tostr(i : longint) : string;{$ifdef USEINLINE}inline;{$endif}overload;    function tostr_with_plus(i : int64) : string;{$ifdef USEINLINE}inline;{$endif}    function DStr(l:longint):string;    {# Returns true if the string s is a number }    function is_number(const s : string) : boolean;{$ifdef USEINLINE}inline;{$endif}    {# Returns true if value is a power of 2, the actual       exponent value is returned in power.    }    function ispowerof2(value : int64;out power : longint) : boolean;    function nextpowerof2(value : int64; out power: longint) : int64;{$ifdef VER2_6}  { only 2.7.1+ has a popcnt function in the system unit }    function PopCnt(AValue : Byte): Byte;    function PopCnt(AValue : Word): Word;    function PopCnt(AValue : DWord): DWord;    function PopCnt(Const AValue : QWord): QWord;{$endif VER2_6}    function backspace_quote(const s:string;const qchars:Tcharset):string;    function octal_quote(const s:string;const qchars:Tcharset):string;    {# If the string is quoted, in accordance with pascal, it is       dequoted and returned in s, and the function returns true.       If it is not quoted, or if the quoting is bad, s is not touched,       and false is returned.    }    function DePascalQuote(var s: ansistring): Boolean;    function CompareStr(const S1, S2: string): Integer;    function CompareText(S1, S2: string): integer;    function CompareVersionStrings(s1,s2: string): longint;    { releases the string p and assignes nil to p }    { if p=nil then freemem isn't called          }    procedure stringdispose(var p : pshortstring);{$ifdef USEINLINE}inline;{$endif}    { allocates mem for a copy of s, copies s to this mem and returns }    { a pointer to this mem                                           }    function stringdup(const s : string) : pshortstring;{$ifdef USEINLINE}inline;{$endif}    {# Allocates memory for the string @var(s) and copies s as zero       terminated string to that allocated memory and returns a pointer       to that mem    }    function  strpnew(const s : string) : pchar;    function  strpnew(const s : ansistring) : pchar;    {# makes the character @var(c) lowercase, with spanish, french and german       character set    }    function lowercase(c : char) : char;    { makes zero terminated string to a pascal string }    { the data in p is modified and p is returned     }    function pchar2pshortstring(p : pchar) : pshortstring;    { inverse of pchar2pshortstring }    function pshortstring2pchar(p : pshortstring) : pchar;    { allocate a new pchar with the contents of a}    function ansistring2pchar(const a: ansistring) : pchar;    { Ansistring (pchar+length) support }    procedure ansistringdispose(var p : pchar;length : longint);    function compareansistrings(p1,p2 : pchar;length1,length2 : longint) : longint;    function concatansistrings(p1,p2 : pchar;length1,length2 : longint) : pchar;    {Lzw encode/decode to compress strings -> save memory.}    function minilzw_encode(const s:string):string;    function minilzw_decode(const s:string):string;    Function nextafter(x,y:double):double;  { hide Sysutils.ExecuteProcess in units using this one after SysUtils}  const    ExecuteProcess = 'Do not use' deprecated 'Use cfileutil.RequotedExecuteProcess instead, ExecuteProcess cannot deal with single quotes as used by Unix command lines';implementation    uses      SysUtils;    var      uppertbl,      lowertbl  : array[char] of char;    function min(a,b : longint) : longint;{$ifdef USEINLINE}inline;{$endif}    {      return the minimal of a and b    }      begin         if a<=b then           min:=a         else           min:=b;      end;    function min(a,b : int64) : int64;{$ifdef USEINLINE}inline;{$endif}    {      return the minimal of a and b    }      begin         if a<=b then           min:=a         else           min:=b;      end;    function max(a,b : longint) : longint;{$ifdef USEINLINE}inline;{$endif}    {      return the maximum of a and b    }      begin         if a>=b then           max:=a         else           max:=b;      end;    function max(a,b : int64) : int64;{$ifdef USEINLINE}inline;{$endif}    {      return the maximum of a and b    }      begin         if a>=b then           max:=a         else           max:=b;      end;    function newalignment(oldalignment: longint; offset: int64): longint;      var        localoffset: longint;      begin        localoffset:=longint(offset);        while (localoffset mod oldalignment)<>0 do          oldalignment:=oldalignment div 2;        newalignment:=oldalignment;      end;    function reverse_byte(b: byte): byte;      const        reverse_nible:array[0..15] of 0..15 =          (%0000,%1000,%0100,%1100,%0010,%1010,%0110,%1110,           %0001,%1001,%0101,%1101,%0011,%1011,%0111,%1111);      begin        reverse_byte:=(reverse_nible[b and $f] shl 4) or reverse_nible[b shr 4];      end;    function align(i,a:longint):longint;{$ifdef USEINLINE}inline;{$endif}    {      return value <i> aligned <a> boundary    }      begin        { for 0 and 1 no aligning is needed }        if a<=1 then          result:=i        else          begin            if i<0 then              result:=((i-a+1) div a) * a            else              result:=((i+a-1) div a) * a;          end;      end;    function size_2_align(len : longint) : shortint;      begin         if len>16 then           size_2_align:=32         else if len>8 then           size_2_align:=16         else if len>4 then           size_2_align:=8         else if len>2 then           size_2_align:=4         else if len>1 then           size_2_align:=2         else           size_2_align:=1;      end;    function packedbitsloadsize(bitlen: int64) : int64;      begin         case bitlen of           1,2,4,8:             result := 1;           { 10 bits can never be split over 3 bytes via 1-8-1, because it }           { always starts at a multiple of 10 bits. Same for the others.  }           3,5,6,7,9,10,12,16:             result := 2;  {$ifdef cpu64bitalu}           { performance penalty for unaligned 8 byte access is much   }           { higher than for unaligned 4 byte access, at least on ppc, }           { so use 4 bytes even in some cases where a value could     }           { always loaded using a single 8 byte load (e.g. in case of }           { 28 bit values)                                            }           11,13,14,15,17..32:             result := 4;           else             result := 8;  {$else cpu64bitalu}           else             result := 4;  {$endif cpu64bitalu}         end;      end;    function isbetteralignedthan(new, org, limit: cardinal): boolean;      var        cnt: cardinal;      begin        cnt:=2;        while (cnt <= limit) do          begin            if (org and (cnt-1)) > (new and (cnt-1)) then              begin                result:=true;                exit;              end            else if (org and (cnt-1)) < (new and (cnt-1)) then              begin                result:=false;                exit;              end;            cnt:=cnt*2;          end;        result:=false;      end;    function next_prime(l: longint): longint;      var        check, checkbound: longint;        ok: boolean;      begin        result:=l or 1;        while l<high(longint) do          begin            ok:=true;            checkbound:=trunc(sqrt(l));            check:=3;            while check<checkbound do              begin                if (l mod check) = 0 then                  begin                    ok:=false;                    break;                  end;                inc(check,2);              end;            if ok then              exit;            inc(l);          end;      end;    function used_align(varalign,minalign,maxalign:shortint):shortint;      begin        { varalign  : minimum alignment required for the variable          minalign  : Minimum alignment of this structure, 0 = undefined          maxalign  : Maximum alignment of this structure, 0 = undefined }        if (minalign>0) and           (varalign<minalign) then         used_align:=minalign        else         begin           if (maxalign>0) and              (varalign>maxalign) then            used_align:=maxalign           else            used_align:=varalign;         end;      end;    procedure Replace(var s:string;s1:string;const s2:string);      var         last,         i  : longint;      begin        s1:=upper(s1);        last:=0;        repeat          i:=pos(s1,upper(s));          if i=last then           i:=0;          if (i>0) then           begin             Delete(s,i,length(s1));             Insert(s2,s,i);             last:=i;           end;        until (i=0);      end;    procedure Replace(var s:AnsiString;s1:string;const s2:AnsiString);      var         last,         i  : longint;      begin        s1:=upper(s1);        last:=0;        repeat          i:=pos(s1,upper(s));          if i=last then           i:=0;          if (i>0) then           begin             Delete(s,i,length(s1));             Insert(s2,s,i);             last:=i;           end;        until (i=0);      end;    procedure ReplaceCase(var s:string;const s1,s2:string);      var         last,         i  : longint;      begin        last:=0;        repeat          i:=pos(s1,s);          if i=last then           i:=0;          if (i>0) then           begin             Delete(s,i,length(s1));             Insert(s2,s,i);             last:=i;           end;        until (i=0);      end;    procedure ReplaceCase(var s: ansistring; const s1, s2: ansistring);      var         last,         i  : longint;      begin        last:=0;        repeat          i:=pos(s1,s);          if i=last then           i:=0;          if (i>0) then           begin             Delete(s,i,length(s1));             Insert(s2,s,i);             last:=i;           end;        until (i=0);      end;    Function MatchPattern(const pattern,what:string):boolean;      var        found : boolean;        i1,i2 : longint;      begin        i1:=0;        i2:=0;        if pattern='' then          begin            result:=(what='');            exit;          end;        found:=true;        repeat          inc(i1);          if (i1>length(pattern)) then            break;          inc(i2);          if (i2>length(what)) then            break;          case pattern[i1] of            '?' :              found:=true;            '*' :              begin                found:=true;                if (i1=length(pattern)) then                 i2:=length(what)                else                 if (i1<length(pattern)) and (pattern[i1+1]<>what[i2]) then                  begin                    if i2<length(what) then                     dec(i1)                  end                else                 if i2>1 then                  dec(i2);              end;            else              found:=(pattern[i1]=what[i2]) or (what[i2]='?');          end;        until not found;        if found then          begin            found:=(i2>=length(what)) and                   (                    (i1>length(pattern)) or                    ((i1=length(pattern)) and                     (pattern[i1]='*'))                   );          end;        result:=found;      end;    function upper(const c : char) : char;    {      return uppercase of c    }      begin        upper:=uppertbl[c];      end;    function upper(const s : string) : string;    {      return uppercased string of s    }      var        i  : longint;      begin        for i:=1 to length(s) do          upper[i]:=uppertbl[s[i]];        upper[0]:=s[0];      end;    function upper(const s : ansistring) : ansistring;    {      return uppercased string of s    }      var        i  : longint;      begin        setlength(upper,length(s));        for i:=1 to length(s) do          upper[i]:=uppertbl[s[i]];      end;    function lower(const c : char) : char;    {      return lowercase of c    }      begin        lower:=lowertbl[c];      end;    function lower(const s : string) : string;    {      return lowercased string of s    }      var        i : longint;      begin        for i:=1 to length(s) do          lower[i]:=lowertbl[s[i]];        lower[0]:=s[0];      end;    function lower(const s : ansistring) : ansistring;    {      return lowercased string of s    }      var        i : longint;      begin        setlength(lower,length(s));        for i:=1 to length(s) do          lower[i]:=lowertbl[s[i]];      end;    procedure uppervar(var s : string);    {      uppercase string s    }      var         i : longint;      begin         for i:=1 to length(s) do          s[i]:=uppertbl[s[i]];      end;    procedure initupperlower;      var        c : char;      begin        for c:=#0 to #255 do         begin           lowertbl[c]:=c;           uppertbl[c]:=c;           case c of             'A'..'Z' :               lowertbl[c]:=char(byte(c)+32);             'a'..'z' :               uppertbl[c]:=char(byte(c)-32);           end;         end;      end;    function DStr(l:longint):string;      var        TmpStr : string[32];        i : longint;      begin        Str(l,TmpStr);        i:=Length(TmpStr);        while (i>3) do         begin           dec(i,3);           if TmpStr[i]<>'-' then            insert('.',TmpStr,i+1);         end;        DStr:=TmpStr;      end;    function rpos(const needle: char; const haystack: shortstring): longint;      begin        result:=length(haystack);        while (result>0) do          begin            if haystack[result]=needle then              exit;            dec(result);          end;      end;    function rpos(const needle: shortstring; const haystack: shortstring): longint;      begin        result:=0;        if (length(needle)=0) or           (length(needle)>length(haystack)) then          exit;        result:=length(haystack)-length(needle);        repeat          if (haystack[result]=needle[1]) and             (copy(haystack,result,length(needle))=needle) then            exit;          dec(result);        until result=0;      end;    function trimbspace(const s:string):string;    {      return s with all leading spaces and tabs removed    }      var        i,j : longint;      begin        j:=1;        i:=length(s);        while (j<i) and (s[j] in [#9,' ']) do         inc(j);        trimbspace:=Copy(s,j,i-j+1);      end;    function trimspace(const s:string):string;    {      return s with all leading and ending spaces and tabs removed    }      var        i,j : longint;      begin        i:=length(s);        while (i>0) and (s[i] in [#9,' ']) do         dec(i);        j:=1;        while (j<i) and (s[j] in [#9,' ']) do         inc(j);        trimspace:=Copy(s,j,i-j+1);      end;    function space (b : longint): string;      var       s: string;      begin        space[0] := chr(b);        s[0] := chr(b);        FillChar (S[1],b,' ');        space:=s;      end;    function PadSpace(const s:string;len:longint):string;    {      return s with spaces add to the end    }      begin         if length(s)<len then          PadSpace:=s+Space(len-length(s))         else          PadSpace:=s;      end;    function GetToken(var s:string;endchar:char):string;      var        i : longint;        quote : char;      begin        GetToken:='';        s:=TrimSpace(s);        if (length(s)>0) and           (s[1] in ['''','"']) then         begin           quote:=s[1];           i:=1;           while (i<length(s)) do            begin              inc(i);              if s[i]=quote then               begin                 { Remove double quote }                 if (i<length(s)) and                    (s[i+1]=quote) then                  begin                    Delete(s,i,1);                    inc(i);                  end                 else                  begin                    GetToken:=Copy(s,2,i-2);                    Delete(s,1,i);                    exit;                  end;               end;            end;           GetToken:=s;           s:='';         end        else         begin           i:=pos(EndChar,s);           if i=0 then            begin              GetToken:=s;              s:='';              exit;            end           else            begin              GetToken:=Copy(s,1,i-1);              Delete(s,1,i);              exit;            end;         end;      end;   function realtostr(e:extended):string;{$ifdef USEINLINE}inline;{$endif}     begin        str(e,result);     end;   function tostr(i : qword) : string;{$ifdef USEINLINE}inline;{$endif}overload;   {     return string of value i   }     begin        str(i,result);     end;   function tostr(i : int64) : string;{$ifdef USEINLINE}inline;{$endif}overload;   {     return string of value i   }     begin        str(i,result);     end;   function tostr(i : longint) : string;{$ifdef USEINLINE}inline;{$endif}overload;   {     return string of value i   }     begin        str(i,result);     end;   function tostr_with_plus(i : int64) : string;{$ifdef USEINLINE}inline;{$endif}   {     return string of value i, but always include a + when i>=0   }     begin        str(i,result);        if i>=0 then          result:='+'+result;     end;    function is_number(const s : string) : boolean;{$ifdef USEINLINE}inline;{$endif}    {      is string a correct number ?    }      var         w : integer;         l : longint;      begin         val(s,l,w);         // remove warning         l:=l;         is_number:=(w=0);      end;    function ispowerof2(value : int64;out power : longint) : boolean;    {      return if value is a power of 2. And if correct return the power    }      begin        if (value = 0) or (value and (value - 1) <> 0) then          exit(false);        power:=BsfQWord(value);        result:=true;      end;    function nextpowerof2(value : int64; out power: longint) : int64;    {      returns the power of 2 >= value    }      var        i : longint;      begin        result := 0;        power := -1;        if ((value <= 0) or            (value >= $4000000000000000)) then          exit;        result := 1;        for i:=0 to 63 do          begin            if result>=value then              begin                power := i;                exit;              end;            result:=result shl 1;          end;      end;{$ifdef VER2_6}    const      PopCntData : array[0..15] of byte = (0,1,1,2,1,2,2,3,1,2,2,3,2,3,3,4);    function PopCnt(AValue : Byte): Byte;      var        i : SizeInt;      begin        Result:=PopCntData[AValue and $f]+PopCntData[(AValue shr 4) and $f];      end;    function PopCnt(AValue : Word): Word;      var        i : SizeInt;      begin        Result:=0;        for i:=0 to 3 do          begin            inc(Result,PopCntData[AValue and $f]);            AValue:=AValue shr 4;          end;      end;    function PopCnt(AValue : DWord): DWord;      var        i : SizeInt;      begin        Result:=0;        for i:=0 to 7 do          begin            inc(Result,PopCntData[AValue and $f]);            AValue:=AValue shr 4;          end;      end;    function PopCnt(Const AValue : QWord): QWord;      begin        Result:=PopCnt(lo(AValue))+PopCnt(hi(AValue))      end; {$endif VER2_6}    function backspace_quote(const s:string;const qchars:Tcharset):string;    var i:byte;    begin      backspace_quote:='';      for i:=1 to length(s) do        begin          if (s[i]=#10) and (#10 in qchars) then            backspace_quote:=backspace_quote+'\n'          else if (s[i]=#13) and (#13 in qchars) then            backspace_quote:=backspace_quote+'\r'          else            begin              if s[i] in qchars then                backspace_quote:=backspace_quote+'\';              backspace_quote:=backspace_quote+s[i];            end;        end;    end;    function octal_quote(const s:string;const qchars:Tcharset):string;    var i:byte;    begin      octal_quote:='';      for i:=1 to length(s) do        begin          if s[i] in qchars then            begin              if ord(s[i])<64 then                octal_quote:=octal_quote+'\'+octstr(ord(s[i]),3)              else                octal_quote:=octal_quote+'\'+octstr(ord(s[i]),4);            end          else            octal_quote:=octal_quote+s[i];        end;    end;    function DePascalQuote(var s: ansistring): Boolean;      var        destPos, sourcePos, len: Integer;        t: string;        ch: Char;    begin      DePascalQuote:= false;      len:= length(s);      if (len >= 1) and (s[1] = '''') then        begin          {Remove quotes, exchange '' against ' }          destPos := 0;          sourcepos:=1;          while (sourcepos<len) do            begin              inc(sourcePos);              ch := s[sourcePos];              if ch = '''' then                begin                  inc(sourcePos);                  if (sourcePos <= len) and (s[sourcePos] = '''') then                    {Add the quote as part of string}                  else                    begin                      SetLength(t, destPos);                      s:= t;                      Exit(true);                    end;                end;              inc(destPos);              t[destPos] := ch;            end;        end;    end;    function pchar2pshortstring(p : pchar) : pshortstring;      var         w,i : longint;      begin         w:=strlen(p);         for i:=w-1 downto 0 do           p[i+1]:=p[i];         p[0]:=chr(w);         pchar2pshortstring:=pshortstring(p);      end;    function pshortstring2pchar(p : pshortstring) : pchar;      var         w,i : longint;      begin         w:=length(p^);         for i:=1 to w do           p^[i-1]:=p^[i];         p^[w]:=#0;         pshortstring2pchar:=pchar(p);      end;    function ansistring2pchar(const a: ansistring) : pchar;      var        len: ptrint;      begin        len:=length(a);        getmem(result,len+1);        if (len<>0) then          move(a[1],result[0],len);        result[len]:=#0;      end;    function lowercase(c : char) : char;       begin          case c of             #65..#90 : c := chr(ord (c) + 32);             #154 : c:=#129;  { german }             #142 : c:=#132;  { german }             #153 : c:=#148;  { german }             #144 : c:=#130;  { french }             #128 : c:=#135;  { french }             #143 : c:=#134;  { swedish/norge (?) }             #165 : c:=#164;  { spanish }             #228 : c:=#229;  { greek }             #226 : c:=#231;  { greek }             #232 : c:=#227;  { greek }          end;          lowercase := c;       end;    function strpnew(const s : string) : pchar;      var         p : pchar;      begin         getmem(p,length(s)+1);         move(s[1],p^,length(s));         p[length(s)]:=#0;         result:=p;      end;    function strpnew(const s: ansistring): pchar;      var         p : pchar;      begin        getmem(p,length(s)+1);        move(s[1],p^,length(s)+1);        result:=p;      end;    procedure stringdispose(var p : pshortstring);{$ifdef USEINLINE}inline;{$endif}      begin         if assigned(p) then           begin             freemem(p);             p:=nil;           end;      end;    function stringdup(const s : string) : pshortstring;{$ifdef USEINLINE}inline;{$endif}      begin         getmem(result,length(s)+1);         result^:=s;      end;    function CompareStr(const S1, S2: string): Integer;      var        count, count1, count2: integer;      begin        result := 0;        Count1 := Length(S1);        Count2 := Length(S2);        if Count1>Count2 then          Count:=Count2        else          Count:=Count1;        result := CompareChar(S1[1],S2[1], Count);        if result=0 then          result:=Count1-Count2;      end;    function CompareText(S1, S2: string): integer;      begin        UpperVar(S1);        UpperVar(S2);        Result:=CompareStr(S1,S2);      end;    function CompareVersionStrings(s1,s2: string): longint;      var        start1, start2,        i1, i2,        num1,num2,        res,        err        : longint;      begin        i1:=1;        i2:=1;        repeat          start1:=i1;          start2:=i2;          while (i1<=length(s1)) and                (s1[i1] in ['0'..'9']) do             inc(i1);          while (i2<=length(s2)) and                (s2[i2] in ['0'..'9']) do             inc(i2);          { one of the strings misses digits -> other is the largest version }          if i1=start1 then            if i2=start2 then              exit(0)            else              exit(-1)          else if i2=start2 then            exit(1);          { get version number part }          val(copy(s1,start1,i1-start1),num1,err);          val(copy(s2,start2,i2-start2),num2,err);          { different -> done }          res:=num1-num2;          if res<>0 then            exit(res);          { if one of the two is at the end while the other isn't, add a '.0' }          if (i1>length(s1)) and             (i2<=length(s1)) then            s1:=s1+'.0'          else if i2>length(s2) then            s2:=s2+'.0';          { compare non-numerical characters normally }          while (i1<=length(s1)) and                not(s1[i1] in ['0'..'9']) and                (i2<=length(s2)) and                not(s2[i2] in ['0'..'9']) do            begin              res:=ord(s1[i1])-ord(s2[i2]);              if res<>0 then                exit(res);              inc(i1);              inc(i2);            end;          { both should be digits again now, otherwise pick the one with the            digits as the largest (it more likely means that the input was            ill-formatted though) }          if (i1<=length(s1)) and             not(s1[i1] in ['0'..'9']) then            exit(-1);          if (i2<=length(s2)) and             not(s2[i2] in ['0'..'9']) then            exit(1);        until false;      end;{*****************************************************************************                               Ansistring (PChar+Length)*****************************************************************************}    procedure ansistringdispose(var p : pchar;length : longint);      begin         if assigned(p) then           begin             freemem(p);             p:=nil;           end;      end;    { enable ansistring comparison }    { 0 means equal }    { 1 means p1 > p2 }    { -1 means p1 < p2 }    function compareansistrings(p1,p2 : pchar;length1,length2 :  longint) : longint;      var         i,j : longint;      begin         compareansistrings:=0;         j:=min(length1,length2);         i:=0;         while (i<j) do          begin            if p1[i]>p2[i] then             begin               compareansistrings:=1;               exit;             end            else             if p1[i]<p2[i] then              begin                compareansistrings:=-1;                exit;              end;            inc(i);          end;         if length1>length2 then          compareansistrings:=1         else          if length1<length2 then           compareansistrings:=-1;      end;    function concatansistrings(p1,p2 : pchar;length1,length2 : longint) : pchar;      var         p : pchar;      begin         getmem(p,length1+length2+1);         move(p1[0],p[0],length1);         move(p2[0],p[length1],length2+1);         concatansistrings:=p;      end;{*****************************************************************************                       Ultra basic KISS Lzw (de)compressor*****************************************************************************}    {This is an extremely basic implementation of the Lzw algorithm. It     compresses 7-bit ASCII strings into 8-bit compressed strings.     The Lzw dictionary is preinitialized with 0..127, therefore this     part of the dictionary does not need to be stored in the arrays.     The Lzw code size is allways 8 bit, so we do not need complex code     that can write partial bytes.}    function minilzw_encode(const s:string):string;    var t,u,i:byte;        c:char;        data:array[128..255] of char;        previous:array[128..255] of byte;        lzwptr:byte;        next_avail:set of 0..255;    label l1;    begin      minilzw_encode:='';      fillchar(data,sizeof(data),#0);      fillchar(previous,sizeof(previous),#0);      if s<>'' then        begin          lzwptr:=127;          t:=byte(s[1]);          i:=2;          u:=128;          next_avail:=[];          while i<=length(s) do            begin              c:=s[i];              if not(t in next_avail) or (u>lzwptr) then goto l1;              while (previous[u]<>t) or (data[u]<>c) do                begin                  inc(u);                  if u>lzwptr then goto l1;                end;              t:=u;              inc(i);              continue;            l1:              {It's a pity that we still need those awfull tricks               with this modern compiler. Without this performance               of the entire procedure drops about 3 times.}              inc(minilzw_encode[0]);              minilzw_encode[length(minilzw_encode)]:=char(t);              if lzwptr=255 then                begin                  lzwptr:=127;                  next_avail:=[];                end              else                begin                  inc(lzwptr);                  data[lzwptr]:=c;                  previous[lzwptr]:=t;                  include(next_avail,t);                end;              t:=byte(c);              u:=128;              inc(i);            end;          inc(minilzw_encode[0]);          minilzw_encode[length(minilzw_encode)]:=char(t);        end;    end;    function minilzw_decode(const s:string):string;    var oldc,newc,c:char;        i,j:byte;        data:array[128..255] of char;        previous:array[128..255] of byte;        lzwptr:byte;        t:string;    begin      minilzw_decode:='';      fillchar(data,sizeof(data),#0);      fillchar(previous,sizeof(previous),#0);      if s<>'' then        begin          lzwptr:=127;          oldc:=s[1];          c:=oldc;          i:=2;          minilzw_decode:=oldc;          while i<=length(s) do            begin              newc:=s[i];              if byte(newc)>lzwptr then                begin                  t:=c;                  c:=oldc;                end              else                begin                  c:=newc;                  t:='';                end;              while c>=#128 do                begin                  inc(t[0]);                  t[length(t)]:=data[byte(c)];                  byte(c):=previous[byte(c)];                end;              inc(minilzw_decode[0]);              minilzw_decode[length(minilzw_decode)]:=c;              for j:=length(t) downto 1 do                begin                  inc(minilzw_decode[0]);                  minilzw_decode[length(minilzw_decode)]:=t[j];                end;              if lzwptr=255 then                lzwptr:=127              else                begin                  inc(lzwptr);                  previous[lzwptr]:=byte(oldc);                  data[lzwptr]:=c;                end;              oldc:=newc;              inc(i);            end;        end;    end;    procedure defaulterror(i:longint);      begin        writeln('Internal error ',i);        runerror(255);      end;    Function Nextafter(x,y:double):double;    // Returns the double precision number closest to x in    // the direction toward y.    // Initial direct translation by Soeren Haastrup from    // www.netlib.org/fdlibm/s_nextafter.c according to    // ====================================================    // Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.    // Developed at SunSoft, a Sun Microsystems, Inc. business.    // Permission to use, copy, modify, and distribute this    // software is freely granted, provided that this notice    // is preserved.    // ====================================================    // and with all signaling policies preserved as is.    type      {$if defined(ENDIAN_LITTLE) and not defined(FPC_DOUBLE_HILO_SWAPPED)}        twoword=record                  lo,hi:longword; // Little Endian split of a double.                end;      {$else}        twoword=record                  hi,lo:longword; // Big Endian split of a double.                end;      {$endif}    var        hx,hy,ix,iy:longint;        lx,ly:longword;    Begin    hx:=twoword(x).hi;    // high and low words of x and y    lx:=twoword(x).lo;    hy:=twoword(y).hi;    ly:=twoword(y).lo;    ix:=hx and $7fffffff;  // absolute values    iy:=hy and $7fffffff;    // Case x=NAN or y=NAN    if ( (ix>=$7ff00000) and ((longword(ix-$7ff00000) or lx) <> 0) )        or ( (iy>=$7ff00000) and ((longword(iy-$7ff00000) OR ly) <> 0) )    then exit(x+y);    // Case x=y    if x=y then exit(x); // (implies Nextafter(0,-0) is 0 and not -0...)    // Case x=0    if (longword(ix) or lx)=0    then begin          twoword(x).hi:=hy and $80000000;  // return +-minimalSubnormal          twoword(x).lo:=1;          y:=x*x;    // set underflow flag (ignored in FPC as default)          if y=x          then exit(y)          else exit(x);        end;    // all other cases    if hx>=0  // x>0    then begin          if (hx>hy) or ( (hx=hy) and (lx>ly) ) // x>y , return x-ulp          then begin                if (lx=0) then hx:=hx-1;                lx:=lx-1;              end          else begin                      // x<y, return x+ulp                lx:=lx+1;                if lx=0 then hx:=hx+1;              end        end    else begin // x<0          if (hy>=0) or (hx>=hy) or ( (hx=hy) and (lx>ly)) // x<y, return x-ulp          then begin                if (lx=0) then hx:=hx-1;                lx:=lx-1;              end          else begin            // x>y , return x+ulp                lx:=lx+1;                if lx=0 then hx:=hx+1;              end        end;    // finally check if overflow or underflow just happend    hy:=hx and $7ff00000;    if (hy>= $7ff00000) then exit(x+x); // overflow and signal    if (hy<$0010000)                    // underflow    then begin          y:=x*x;              // raise underflow flag          if y<>x          then begin                twoword(y).hi:=hx;                twoword(y).lo:=lx;                exit(y);              end        end;    twoword(x).hi:=hx;    twoword(x).lo:=lx;    nextafter:=x;    end;initialization  internalerrorproc:=@defaulterror;  initupperlower;end.
 |