| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Michael Van Canneyt,    member of 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. **********************************************************************}{  This file contains the implementation of the LongString type,  and all things that are needed for it.  LongSTring is defined as a 'silent' pansichar :  a pansichar that points to :  @   : Longint for size  @+4 : Unused byte;  @+5 : AnsiString;   So LS[i] is converted to the address @LS+4+i.  pansichar[0]-pansichar[3] : Longint Size  pansichar [4] : Unused  pansichar[5] : AnsiString;}{$ifdef lstrings_unit}{ Compile as a separate unit - development only}{$IFNDEF FPC_DOTTEDUNITS}unit lstrings;{$ENDIF FPC_DOTTEDUNITS}InterfaceType longstring = pansichar;     ShortString = string;{ Internal functions, will not appear in systemh.inc }Function  NewLongString (Len : Longint) : LongString;Procedure DisposeLongString (Var S : LongString; Len : Longint);Procedure Long_String_Concat (Var S1 : LongString; Const S2 : LongString; maxlen : Longint);Procedure Long_ShortString_Concat (Var S1: LongString; Const S2 : ShortString; maxlen : Longint);Procedure Long_To_ShortString (Var S1 : ShortString; Const S2 : LongString; Maxlen : Longint);Procedure Short_To_LongString (Var S1 : LongString; Const S2 : ShortString; Maxlen : Longint);Function  LongCompare (Const S1,S2 : Longstring): Longint;Function  LongCompare (Const S1 : LongString; Const S2 : ShortString): Longint;{ Public functions, Will end up in systemh.inc }Procedure SetLength (Var S : LongString; l : Longint);Procedure Write_Text_LongString (Len : Longint; T : Textrec; Var S : LongString);Function  Length (Const S : LongString) : Longint;Function  Copy (Const S : LongString; Index,Size : Longint) : LongString;Function  Pos (Const Substr : LongString; Const Source : Longstring) : Longint;Procedure Insert (Const Source : LongString; Var S : LongString; Index : Longint);Procedure Delete (Var S : LongString; Index,Size: Longint);Procedure Val (Const S : LongString; var R : real; Var Code : Integer);{Procedure Val (Const S : LongString; var D : Double; Var Code : Integer);}Procedure Val (Const S : LongString; var E : Extended; Code : Integer);Procedure Val (Const S : LongString; var C : Cardinal; Code : Integer);Procedure Val (Const S : LongString; var L : Longint; Var Code : Integer);Procedure Val (Const S : LongString; var W : Word; Var Code : Integer);Procedure Val (Const S : LongString; var I : Integer; Var Code : Integer);Procedure Val (Const S : LongString; var B : Byte; Var Code : Integer);Procedure Val (Const S : LongString; var SI : ShortInt; Var  Code : Integer);Procedure Str (Const R : Real;Len, fr : longint; Var S : LongString);{Procedure Str (Const D : Double;Len,fr : longint; Var S : LongString);}Procedure Str (Const E : Extended;Len,fr : longint; Var S : LongString);Procedure Str (Const C : Cardinal;len : Longint; Var S : LongString);Procedure Str (Const L : LongInt;len : longint; Var S : LongString);Procedure Str (Const W : Word;len : longint; Var S : LongString);Procedure Str (Const I : Integer;len : Longint; Var S : LongString);Procedure Str (Const B : Byte; Len : longint; Var S : LongString);Procedure Str (Const SI : ShortInt; Len : longint; Var S : LongString);Implementation{$endif}Type PLongint = ^Longint;{ ---------------------------------------------------------------------  Internal functions, not in interface.  ---------------------------------------------------------------------}Function  NewLongString (Len : Longint) : LongString;{  Allocate a new string on the heap.  initialize it to zero length}Var P : Pointer;begin  GetMem(P,Len+5);  If P<>Nil then     begin     PLongint(P)^:=0;     pansichar(P+4)^:=#0;     end;  NewLongString:=P;end;Procedure DisposeLongString (Var S : LongString; Len : Longint);{  DeAllocates a LongString From the heap.}begin  FreeMem (Pointer(S),Len+5);end;Procedure Long_String_Concat (Var S1 : LongString; Const S2 : LongString; maxlen : Longint);{  Concatenates 2 LongStrings : S1+S2  If maxlen<>-1 then the result has maximal length maxlen.}Var Size : Longint;begin  Size:=PLongint(S2)^;  If maxlen<>-1 then    if Size+PLongint(S1)^>MaxLen then      Size:=Maxlen-PLongint(S1)^;  If Size<=0 then exit;  Move (pansichar(S2)[5],ansipchar(S1)[PLongint(S1)^+5],Size);  PLongint(S1)^:=PLongint(S1)^+Size;end;Procedure Long_ShortString_Concat (Var S1: LongString; Const S2 : ShortString; maxlen : Longint);{  Concatenates a long with a short string; : S2 + S2  If maxlen<>-1 then the result has maximal length maxlen.}Var Size : Longint;begin  Size:=Byte(S2[0]);  if MaxLen<>-1 then    if Size+PLongint(S1)^>Maxlen then      Size:=Maxlen-PLongint(S1)^;  If Size<=0 then exit;  Move (S2[1],PAnsiChar(S1)[PLongint(S1)^+5],Size);  PLongint(S1)^:=PLongint(S1)^+Size;end;Procedure Long_To_ShortString (Var S1 : ShortString; Const S2 : LongString; Maxlen : Longint);{ Converts a LongString to a longstring; if maxlen<>-1, the resulting string has maximal length maxlen else a default length of 255 is taken.}Var Size : Longint;begin  Size:=PLongint(S2)^;  if maxlen=-1 then maxlen:=255;  If Size>maxlen then Size:=maxlen;  Move (PAnsiChar(S2)[5],S1[1],Size);  S1[0]:=chr(Size);end;Procedure Short_To_LongString (Var S1 : LongString; Const S2 : ShortString; Maxlen : Longint);{ Converts a ShortString to a LongString; if maxlen<>-1 then the resulting string has length maxlen.}Var Size : Longint;begin  Size:=Byte(S2[0]);  if maxlen=-1 then maxlen:=255;  If Size>maxlen then Size:=maxlen;  Move (S2[1],PAnsiChar(S1)[5],Size);  PLongint(S1)^:=Size;end;Function LongCompare (Const S1,S2 : Longstring): Longint;{  Compares 2 longStrings;  The result is   <0 if S1<S2   0 if S1=S2   >0 if S1>S2}Var i,MaxI,Temp : Longint;begin Temp:=0; i:=1; MaxI:=PLongint(S1)^; if MaxI>PLOngint(S2)^ then MaxI:=PLongint(S2)^; While (i<=MaxI) and (Temp=0) do   begin   Temp:= Byte( PAnsiChar(S1)[i+4] ) - Byte( PAnsiChar(S2)[I+4] );   inc(i);   end; if temp=0 then temp:=Plongint(S1)^-PLongint(S2)^; LongCompare:=Temp;end;Function LongCompare (Const S1 : LongString; Const S2 : ShortString): Longint;{  Compares a longString with a ShortString;  The result is   <0 if S1<S2   0 if S1=S2   >0 if S1>S2}Var i,MaxI,Temp : Longint;begin Temp:=0; i:=1; MaxI:=PLongint(S1)^; if MaxI>byte(S2[0]) then MaxI:=Byte(S2[0]); While (i<=MaxI) and (Temp=0) do   begin   Temp:=(Byte(PAnsiChar(S1)[i+4])-Byte(S2[I]));   inc(i);   end; LongCompare:=Temp;end;Procedure Write_Text_LongString (Len : Longint; T : TextRec; Var S : LongString);{ Writes a LongString to the Text file T}beginend;{ ---------------------------------------------------------------------   Public functions, In interface.  ---------------------------------------------------------------------}Function Length (Const S : LongString) : Longint;begin  Length:=PLongint(S)^;end;Procedure SetLength (Var S : LongString; l : Longint);begin  PLongint(S)^:=l;end;Function Copy (Const S : LongString; Index,Size : Longint) : LongString;var ResultAddress : PAnsiChar;begin  ResultAddress:=NewLongString (Size);  if ResultAddress=Nil then    {We're in deep shit here !!}    exit;  dec(index);  if PLongint(S)^<Index+Size then    Size:=PLongint(S)^-Index;  if Size>0 then    Move (PAnsiChar(S)[Index+5],ResultAddress[5],Size)  Else    Size:=0;  PLongint(ResultAddress)^:=Size;  Copy:=ResultAddressend;Function Pos (Const Substr : LongString; Const Source : Longstring) : Longint;var i,j : longint;    e : boolean;    s : longstring;begin i := 0; j := 0; e := true; if Plongint(substr)^=0 then e := false; while (e) and (i <= length (Source) - length (substr)) do   begin   inc (i);   s :=copy(Source,i,length(Substr));   if LongCompare(substr,s)=0 then     begin     j := i;     e := false;     end;   DisposeLongString(s,length(Substr));   end; pos := j;end;Procedure Val (Const S : LongString; var R : real; Var Code : Integer);Var SS : String;begin Long_To_ShortString (SS,S,255); System.Val(SS,R,Code);end;{Procedure Val (Const S : LongString; var D : Double; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); Val(SS,D,Code);end;}Procedure Val (Const S : LongString; var E : Extended; Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,E,Code);end;Procedure Val (Const S : LongString; var C : Cardinal; Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,C,Code);end;Procedure Val (Const S : LongString; var L : Longint; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,L,Code);end;Procedure Val (Const S : LongString; var W : Word; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,W,Code);end;Procedure Val (Const S : LongString; var I : Integer; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,I,Code);end;Procedure Val (Const S : LongString; var B : Byte; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,B,Code);end;Procedure Val (Const S : LongString; var SI : ShortInt; Var Code : Integer);Var SS : ShortString;begin Long_To_ShortString (SS,S,255); System.Val(SS,SI,Code);end;Procedure Str (Const R : Real;Len,fr : Longint; Var S : LongString);Var SS : ShortString;begin {int_Str_Real (R,Len,fr,SS);} Short_To_LongString (S,SS,255);end;{Procedure Str (Const D : Double;Len,fr: Longint; Var S : LongString);Var SS : ShortString;begin {int_Str_Double (D,Len,fr,SS);} Short_To_LongString (S,SS,255);end;}Procedure Str (Const E : Extended;Lenf,Fr: Longint; Var S : LongString);Var SS : ShortString;begin {int_Str_Extended (E,Len,fr,SS);} Short_To_LongString (S,SS,255);end;Procedure Str (Const C : Cardinal;Len : Longint; Var S : LongString);beginend;Procedure Str (Const L : Longint; Len : Longint; Var S : LongString);Var SS : ShortString;begin {int_Str_Longint (L,Len,fr,SS);} Short_To_LongString (S,SS,255);end;Procedure Str (Const W : Word;Len : Longint; Var S : LongString);beginend;Procedure Str (Const I : Integer;Len : Longint; Var S : LongString);beginend;Procedure Str (Const B : Byte; Len : Longint; Var S : LongString);beginend;Procedure Str (Const SI : ShortInt; Len : Longint; Var S : LongString);beginend;Procedure Delete (Var S : LongString; Index,Size: Longint);begin  if index<=0 then    begin    Size:=Size+index-1;    index:=1;    end;  if (Index<=PLongint(s)^) and (Size>0) then    begin    if Size+Index>PLongint(s)^ then      Size:=PLongint(s)^-Index+1;    PLongint(s)^:=PLongint(s)^-Size;    if Index<=Length(s) then      Move(PAnsiChar(s)[Index+Size+4],PAnsiChar(s)[Index+4],Length(s)-Index+1);    end;end;Procedure Insert (Const Source : LongString; Var S : LongString; Index : Longint);var s3,s4 : PAnsiChar;begin  if index <= 0 then index := 1;  s3 := longString(copy (s, index, length(s)));  if index > PLongint(s)^ then index := PLongint(S)^+1;  PLongint(s)^ := index - 1;  s4 :=PAnsiChar ( NewLongString (Plongint(Source)^) );  Long_String_Concat(LongString(s4),Source,-1);  Long_String_Concat(LongString(S4),LongString(s3),-1);  Long_String_Concat(S,LongString(S4),-1);  DisposeLongstring(LongString(S3),PLongint(S3)^);  DisposeLongString(LongString(S4),PLongint(S4)^);end;{$ifdef lstrings_unit}end.
 |