1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by the Free Pascal development team
- See the file COPYING.FPC, included in this distribution,
- for details about the copyright.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- **********************************************************************}
- {****************************************************************************
- subroutines for string handling
- ****************************************************************************}
- {$ifndef FPC_HAS_SHORTSTR_SETLENGTH}
- {$define FPC_HAS_SHORTSTR_SETLENGTH}
- procedure fpc_Shortstr_SetLength(var s:shortstring;len:SizeInt);[Public,Alias : 'FPC_SHORTSTR_SETLENGTH']; compilerproc;
- begin
- if Len>255 then
- Len:=255;
- s[0]:=chr(len);
- end;
- {$endif FPC_HAS_SHORTSTR_SETLENGTH}
- {$ifndef FPC_HAS_SHORTSTR_COPY}
- {$define FPC_HAS_SHORTSTR_COPY}
- function fpc_shortstr_copy(const s : shortstring;index : SizeInt;count : SizeInt): shortstring;compilerproc;
- begin
- if count<0 then
- count:=0;
- if index>1 then
- dec(index)
- else
- index:=0;
- if index>length(s) then
- count:=0
- else
- if count>length(s)-index then
- count:=length(s)-index;
- fpc_shortstr_Copy[0]:=chr(Count);
- fpc_shortstr_shortstr_intern_charmove(s,Index+1,fpc_shortstr_Copy,1,Count);
- end;
- {$endif FPC_HAS_SHORTSTR_COPY}
- {$ifndef FPC_HAS_SHORTSTR_DELETE}
- {$define FPC_HAS_SHORTSTR_DELETE}
- procedure {$ifdef VER3_0}delete{$else}fpc_shortstr_delete{$endif}(var s : shortstring;index : SizeInt;count : SizeInt);
- begin
- if index<=0 then
- exit;
- if (Index<=Length(s)) and (Count>0) then
- begin
- if Count>length(s)-Index then
- Count:=length(s)-Index+1;
- s[0]:=Chr(length(s)-Count);
- if Index<=Length(s) then
- fpc_shortstr_shortstr_intern_charmove(s,Index+Count,s,Index,Length(s)-Index+1);
- end;
- end;
- {$endif FPC_HAS_SHORTSTR_DELETE}
- {$ifndef FPC_HAS_SHORTSTR_INSERT}
- {$define FPC_HAS_SHORTSTR_INSERT}
- procedure {$ifdef ver3_0}insert{$else}fpc_shortstr_insert{$endif}(const source : shortstring;var s : shortstring;index : SizeInt);
- var
- cut,srclen,indexlen : SizeInt;
- begin
- if index<1 then
- index:=1;
- if index>length(s) then
- begin
- index:=length(s)+1;
- if index>high(s) then
- exit;
- end;
- indexlen:=Length(s)-Index+1;
- srclen:=length(Source);
- if sizeInt(length(source))+sizeint(length(s))>=sizeof(s) then
- begin
- cut:=sizeInt(length(source))+sizeint(length(s))-sizeof(s)+1;
- if cut>indexlen then
- begin
- dec(srclen,cut-indexlen);
- indexlen:=0;
- end
- else
- dec(indexlen,cut);
- end;
- fpc_shortstr_shortstr_intern_charmove(s,Index,s,Index+srclen,indexlen);
- fpc_shortstr_shortstr_intern_charmove(Source,1,s,Index,srclen);
- s[0]:=chr(index+srclen+indexlen-1);
- end;
- {$endif FPC_HAS_SHORTSTR_INSERT}
- {$ifndef FPC_HAS_SHORTSTR_INSERT_CHAR}
- {$define FPC_HAS_SHORTSTR_INSERT_CHAR}
- procedure {$ifdef ver3_0}insert{$else}fpc_shortstr_insert_char{$endif}(source : Char;var s : shortstring;index : SizeInt);
- var
- indexlen : SizeInt;
- begin
- if index<1 then
- index:=1;
- if index>length(s) then
- begin
- index:=length(s)+1;
- if index>high(s) then
- exit;
- end;
- indexlen:=Length(s)-Index+1;
- if (sizeint(length(s))+1=sizeof(s)) and (indexlen>0) then
- dec(indexlen);
- fpc_shortstr_shortstr_intern_charmove(s,Index,s,Index+1,indexlen);
- s[Index]:=Source;
- s[0]:=chr(index+indexlen);
- end;
- {$endif FPC_HAS_SHORTSTR_INSERT_CHAR}
- {$ifndef FPC_HAS_SHORTSTR_POS_SHORTSTR}
- {$define FPC_HAS_SHORTSTR_POS_SHORTSTR}
- function pos(const substr : shortstring;const s : shortstring; Offset : Sizeint = 1):SizeInt;
- var
- i,MaxLen : SizeInt;
- pc : pchar;
- begin
- Pos:=0;
- if (Length(SubStr)>0) and (Offset>0) and (Offset<=Length(S)) then
- begin
- MaxLen:=sizeint(Length(s))-Length(SubStr);
- i:=Offset-1;
- pc:=@s[Offset];
- while (i<=MaxLen) do
- begin
- inc(i);
- if (SubStr[1]=pc^) and
- (CompareChar(Substr[1],pc^,Length(SubStr))=0) then
- begin
- Pos:=i;
- exit;
- end;
- inc(pc);
- end;
- end;
- end;
- {$endif FPC_HAS_SHORTSTR_POS_SHORTSTR}
- {$ifndef FPC_HAS_SHORTSTR_POS_CHAR}
- {$define FPC_HAS_SHORTSTR_POS_CHAR}
- {Faster when looking for a single char...}
- function pos(c:char;const s:shortstring; Offset : Sizeint = 1 ):SizeInt;
- var
- i : SizeInt;
- pc : pchar;
- begin
- Pos:=0;
- if (Offset<1) or (Offset>Length(S)) then
- exit;
- pc:=@s[Offset];
- for i:=Offset to length(s) do
- begin
- if pc^=c then
- begin
- pos:=i;
- exit;
- end;
- inc(pc);
- end;
- pos:=0;
- end;
- {$endif FPC_HAS_SHORTSTR_POS_CHAR}
- function fpc_char_copy(c:char;index : SizeInt;count : SizeInt): shortstring;compilerproc;
- begin
- if (index=1) and (Count>0) then
- fpc_char_Copy:=c
- else
- fpc_char_Copy:='';
- end;
- function pos(const substr : shortstring;c:char; Offset : Sizeint = 1): SizeInt;
- begin
- if (length(substr)=1) and (substr[1]=c) and (Offset=1) then
- Pos:=1
- else
- Pos:=0;
- end;
- {$if not defined(FPC_UPCASE_CHAR) or not defined(FPC_LOWERCASE_CHAR)}
- {$ifdef IBM_CHAR_SET}
- const
- UpCaseTbl : shortstring[7]=#154#142#153#144#128#143#165;
- LoCaseTbl : shortstring[7]=#129#132#148#130#135#134#164;
- {$endif}
- {$endif}
- {$ifndef FPC_UPCASE_CHAR}
- {$define FPC_UPCASE_CHAR}
- function upcase(c : char) : char;
- {$IFDEF IBM_CHAR_SET}
- var
- i : ObjpasInt;
- {$ENDIF}
- begin
- if (c in ['a'..'z']) then
- upcase:=char(byte(c)-32)
- else
- {$IFDEF IBM_CHAR_SET}
- begin
- i:=Pos(c,LoCaseTbl);
- if i>0 then
- upcase:=UpCaseTbl[i]
- else
- upcase:=c;
- end;
- {$ELSE}
- upcase:=c;
- {$ENDIF}
- end;
- {$endif FPC_UPCASE_CHAR}
- {$ifndef FPC_UPCASE_SHORTSTR}
- {$define FPC_UPCASE_SHORTSTR}
- function upcase(const s : shortstring) : shortstring;
- var
- i : ObjpasInt;
- begin
- upcase[0]:=s[0];
- for i := 1 to length (s) do
- upcase[i] := upcase (s[i]);
- end;
- {$endif FPC_UPCASE_SHORTSTR}
- {$ifndef FPC_LOWERCASE_CHAR}
- {$define FPC_LOWERCASE_CHAR}
- function lowercase(c : char) : char;overload;
- {$IFDEF IBM_CHAR_SET}
- var
- i : ObjpasInt;
- {$ENDIF}
- begin
- if (c in ['A'..'Z']) then
- lowercase:=char(byte(c)+32)
- else
- {$IFDEF IBM_CHAR_SET}
- begin
- i:=Pos(c,UpCaseTbl);
- if i>0 then
- lowercase:=LoCaseTbl[i]
- else
- lowercase:=c;
- end;
- {$ELSE}
- lowercase:=c;
- {$ENDIF}
- end;
- {$endif FPC_LOWERCASE_CHAR}
- {$ifndef FPC_LOWERCASE_SHORTSTR}
- {$define FPC_LOWERCASE_SHORTSTR}
- function lowercase(const s : shortstring) : shortstring; overload;
- var
- i : ObjpasInt;
- begin
- lowercase [0]:=s[0];
- for i:=1 to length(s) do
- lowercase[i]:=lowercase (s[i]);
- end;
- {$endif FPC_LOWERCASE_SHORTSTR}
- const
- HexTbl : array[0..15] of char='0123456789ABCDEF';
- function hexstr(val : longint;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- hexstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- hexstr[i]:=hextbl[val and $f];
- val:=val shr 4;
- end;
- end;
- function octstr(val : longint;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- octstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- octstr[i]:=hextbl[val and 7];
- val:=val shr 3;
- end;
- end;
- function binstr(val : longint;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- binstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- binstr[i]:=char(48+val and 1);
- val:=val shr 1;
- end;
- end;
- function hexstr(val : int64;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- hexstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- hexstr[i]:=hextbl[val and $f];
- val:=val shr 4;
- end;
- end;
- function octstr(val : int64;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- octstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- octstr[i]:=hextbl[val and 7];
- val:=val shr 3;
- end;
- end;
- function binstr(val : int64;cnt : byte) : shortstring;
- var
- i : ObjpasInt;
- begin
- binstr[0]:=char(cnt);
- for i:=cnt downto 1 do
- begin
- binstr[i]:=char(48+val and 1);
- val:=val shr 1;
- end;
- end;
- {$ifndef FPC_HAS_QWORD_HEX_SHORTSTR}
- {$define FPC_HAS_QWORD_HEX_SHORTSTR}
- Function hexStr(Val:qword;cnt:byte):shortstring;
- begin
- hexStr:=hexStr(int64(Val),cnt);
- end;
- {$endif FPC_HAS_QWORD_HEX_SHORTSTR}
- {$ifndef FPC_HAS_QWORD_OCT_SHORTSTR}
- {$define FPC_HAS_QWORD_OCT_SHORTSTR}
- Function OctStr(Val:qword;cnt:byte):shortstring;
- begin
- OctStr:=OctStr(int64(Val),cnt);
- end;
- {$endif FPC_HAS_QWORD_OCT_SHORTSTR}
- {$ifndef FPC_HAS_QWORD_BIN_SHORTSTR}
- {$define FPC_HAS_QWORD_BIN_SHORTSTR}
- Function binStr(Val:qword;cnt:byte):shortstring;
- begin
- binStr:=binStr(int64(Val),cnt);
- end;
- {$endif FPC_HAS_QWORD_BIN_SHORTSTR}
- {$ifndef FPC_HAS_HEXSTR_POINTER_SHORTSTR}
- {$define FPC_HAS_HEXSTR_POINTER_SHORTSTR}
- function hexstr(val : pointer) : shortstring;
- var
- i : ObjpasInt;
- v : ptruint;
- begin
- v:=ptruint(val);
- hexstr[0]:=chr(sizeof(pointer)*2);
- for i:=sizeof(pointer)*2 downto 1 do
- begin
- hexstr[i]:=hextbl[v and $f];
- v:=v shr 4;
- end;
- end;
- {$endif FPC_HAS_HEXSTR_POINTER_SHORTSTR}
- {$ifndef FPC_HAS_SPACE_SHORTSTR}
- {$define FPC_HAS_SPACE_SHORTSTR}
- function space (b : byte): shortstring;
- begin
- space[0] := chr(b);
- FillChar (Space[1],b,' ');
- end;
- {$endif FPC_HAS_SPACE_SHORTSTR}
- {*****************************************************************************
- Str() Helpers
- *****************************************************************************}
- procedure fpc_shortstr_SInt(v : valSInt;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_SINT']; compilerproc;
- begin
- int_str(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- procedure fpc_shortstr_UInt(v : valUInt;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_UINT']; compilerproc;
- begin
- int_str_unsigned(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- {$ifndef CPU64}
- procedure fpc_shortstr_qword(v : qword;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_QWORD']; compilerproc;
- begin
- int_str_unsigned(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- procedure fpc_shortstr_int64(v : int64;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_INT64']; compilerproc;
- begin
- int_str(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- {$endif CPU64}
- {$if defined(CPU16) or defined(CPU8)}
- procedure fpc_shortstr_longword(v : longword;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_LONGWORD']; compilerproc;
- begin
- int_str_unsigned(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- procedure fpc_shortstr_longint(v : longint;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_LONGINT']; compilerproc;
- begin
- int_str(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- procedure fpc_shortstr_word(v : word;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_WORD']; compilerproc;
- begin
- int_str_unsigned(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- procedure fpc_shortstr_smallint(v : smallint;len : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_SMALLINT']; compilerproc;
- begin
- int_str(v,s);
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- {$endif CPU16 or CPU8}
- { fpc_shortstr_sInt must appear before this file is included, because }
- { it's used inside real2str.inc and otherwise the searching via the }
- { compilerproc name will fail (JM) }
- {$ifndef FPUNONE}
- {$ifdef FLOAT_ASCII_FALLBACK}
- {$I real2str.inc}
- {$else not FLOAT_ASCII_FALLBACK}
- {$I flt_conv.inc}
- {$endif FLOAT_ASCII_FALLBACK}
- {$endif}
- {$ifndef FPUNONE}
- procedure fpc_shortstr_float(d : ValReal;len,fr,rt : SizeInt;out s : shortstring);[public,alias:'FPC_SHORTSTR_FLOAT']; compilerproc;
- begin
- str_real(len,fr,d,treal_type(rt),s);
- end;
- {$endif}
- {$ifndef FPC_STR_ENUM_INTERN}
- function fpc_shortstr_enum_intern(ordinal,len:sizeint;typinfo,ord2strindex:pointer;out s:shortstring): longint;
- { The following contains the TTypeInfo/TTypeData records from typinfo.pp
- specialized for the tkEnumeration case (and stripped of unused things). }
- type
- PPstring=^Pstring;
- Penum_typeinfo=^Tenum_typeinfo;
- Tenum_typeinfo={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- kind:TTypeKind; { always tkEnumeration }
- num_chars:byte;
- chars:array[0..0] of char; { variable length with size of num_chars }
- end;
- {$push}
- {$packrecords c}
- Penum_typedata=^Tenum_typedata;
- Tenum_typedata={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- ordtype:byte;
- case TTypeKind of
- tkInteger,tkChar,tkEnumeration,tkBool,tkWChar,tkSet: (
- MinValue,MaxValue : Longint;
- case byte of
- tkEnumeration: (
- BaseTypeRef : pointer
- );
- {$ifndef VER3_0}
- {tkBool with OrdType=otSQWord }
- tkInt64:
- (MinInt64Value, MaxInt64Value: Int64);
- {tkBool with OrdType=otUQWord }
- tkQWord:
- (MinQWordValue, MaxQWordValue: QWord);
- {$endif VER3_0}
- );
- { more data here, but not needed }
- end;
- { Pascal data types for the ordinal enum value to string table. It consists of a header
- that indicates what type of data the table stores, either a direct lookup table (when
- o = lookup) or a set of ordered (ordinal value, string) tuples (when o = search). }
- { A single entry in the set of ordered tuples }
- Psearch_data=^Tsearch_data;
- Tsearch_data={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- value:longint;
- name:Pstring;
- end;
- Penum_ord_to_string=^Tenum_ord_to_string;
- Tenum_ord_to_string={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- o:(lookup,search);
- case integer of
- 0: (lookup_data:array[0..0] of Pstring);
- 1: (num_entries:longint;
- search_data:array[0..0] of Tsearch_data);
- end;
- {$pop}
- var
- enum_o2s : Penum_ord_to_string;
- header:Penum_typeinfo;
- body:Penum_typedata;
- res:Pshortstring;
- sorted_data:Psearch_data;
- spaces,i,m,h,l:longint;
- begin
- { set default return value }
- fpc_shortstr_enum_intern:=107;
- enum_o2s:=Penum_ord_to_string(ord2strindex);
- { depending on the type of table in ord2strindex retrieve the data }
- if (enum_o2s^.o=lookup) then
- begin
- { direct lookup table }
- header:=Penum_typeinfo(typinfo);
- { calculate address of enum rtti body: add the actual size of the
- enum_rtti_header, and then align. Use an alignment of 1 (which
- does nothing) in case FPC_REQUIRES_PROPER_ALIGNMENT is not set
- to avoid the need for an if in this situation }
- {$ifdef VER3_0}
- body:=Penum_typedata(aligntoptr(pointer(header) + 2 * sizeof(byte) { kind, num_chars } + header^.num_chars));
- {$else VER3_0}
- body:=Penum_typedata(aligntoqword(pointer(header) + 2 * sizeof(byte) { kind, num_chars } + header^.num_chars));
- {$endif VER3_0}
- with body^ do
- begin
- { Bounds check for the ordinal value for this enum }
- if (ordinal<minvalue) or (ordinal>maxvalue) then
- exit;
- { make the ordinal index for lookup zero-based }
- dec(ordinal,minvalue);
- end;
- { temporarily disable range checking because of the access to the array[0..0]
- member of Tenum_ord_to_string_lookup }
- {$push}{$R-}
- res:=enum_o2s^.lookup_data[ordinal];
- {$pop}
- if (not assigned(res)) then
- exit;
- s:=res^;
- end
- else
- begin
- { The compiler did generate a sorted array of (ordvalue,Pstring) tuples }
- sorted_data:=@enum_o2s^.search_data;
- { Use a binary search to get the string }
- l:=0;
- { temporarily disable range checking because of the access to the array[0..0]
- member of Tenum_ord_to_string_search }
- {$push}{$R-}
- h:=enum_o2s^.num_entries-1;
- repeat
- m:=(l+h) div 2;
- if ordinal>sorted_data[m].value then
- l:=m+1
- else if ordinal<sorted_data[m].value then
- h:=m-1
- else
- break;
- if l>h then
- exit; { Ordinal value not found? Exit }
- until false;
- {$pop}
- s:=sorted_data[m].name^;
- end;
- { Pad the string with spaces if necessary }
- if (len>length(s)) then
- begin
- spaces:=len-length(s);
- for i:=1 to spaces do
- s[length(s)+i]:=' ';
- inc(byte(s[0]),spaces);
- end;
- fpc_shortstr_enum_intern:=0;
- end;
- procedure fpc_shortstr_enum(ordinal,len:sizeint;typinfo,ord2strindex:pointer;out s:shortstring);[public,alias:'FPC_SHORTSTR_ENUM'];compilerproc;
- var
- res: longint;
- begin
- res:=fpc_shortstr_enum_intern(ordinal,len,typinfo,ord2strindex,s);
- if (res<>0) then
- runerror(107);
- end;
- { also define alias for internal use in the system unit }
- procedure fpc_shortstr_enum(ordinal,len:sizeint;typinfo,ord2strindex:pointer;out s:shortstring);external name 'FPC_SHORTSTR_ENUM';
- {$endif FPC_SHORTSTR_ENUM_INTERN}
- procedure fpc_shortstr_bool(b : boolean;len:sizeint;out s:shortstring);[public,alias:'FPC_SHORTSTR_BOOL'];compilerproc;
- begin
- if b then
- s:='TRUE'
- else
- s:='FALSE';
- if length(s)<len then
- s:=space(len-length(s))+s;
- end;
- { also define alias for internal use in the system unit }
- procedure fpc_shortstr_bool(b : boolean;len:sizeint;out s:shortstring);external {$ifndef cpujvm}name 'FPC_SHORTSTR_BOOL'{$endif};
- procedure fpc_shortstr_currency({$ifdef cpujvm}constref{$endif} c : currency; len,f : SizeInt; out s : shortstring);[public,alias:'FPC_SHORTSTR_CURRENCY']; compilerproc;
- const
- MinLen = 8; { Minimal string length in scientific format }
- var
- buf : array[1..19] of char;
- i,j,k,reslen,tlen,sign,r,point : ObjpasInt;
- ic : qword;
- begin
- fillchar(buf,length(buf),'0');
- { default value for length is -32767 }
- if len=-32767 then
- len:=25;
- if PInt64(@c)^ >= 0 then
- begin
- ic:=QWord(PInt64(@c)^);
- sign:=0;
- end
- else
- begin
- sign:=1;
- ic:=QWord(-PInt64(@c)^);
- end;
- { converting to integer string }
- tlen:=0;
- repeat
- Inc(tlen);
- buf[tlen]:=Chr(ic mod 10 + $30);
- ic:=ic div 10;
- until ic = 0;
- { calculating:
- reslen - length of result string,
- r - rounding or appending zeroes,
- point - place of decimal point }
- reslen:=tlen;
- if f <> 0 then
- Inc(reslen); { adding decimal point length }
- if f < 0 then
- begin
- { scientific format }
- Inc(reslen,5); { adding length of sign and exponent }
- if len < MinLen then
- len:=MinLen;
- r:=reslen-len;
- if reslen < len then
- reslen:=len;
- if r > 0 then
- begin
- reslen:=len;
- point:=tlen - r;
- end
- else
- point:=tlen;
- end
- else
- begin
- { fixed format }
- Inc(reslen, sign);
- { prepending fractional part with zeroes }
- while tlen < 5 do
- begin
- Inc(reslen);
- Inc(tlen);
- buf[tlen]:='0';
- end;
- { Currency have 4 digits in fractional part }
- r:=4 - f;
- point:=f;
- if point <> 0 then
- begin
- if point > 4 then
- point:=4;
- Inc(point);
- end;
- Dec(reslen,r);
- end;
- { rounding string if r > 0 }
- if r > 0 then
- begin
- k := 0;
- i := r+2;
- if i > tlen then
- i := tlen+1;
- if buf[i-2] >= '5' then
- begin
- if buf[i-1] < '9' then
- buf[i-1] := chr(ord(buf[i-1])+1)
- else
- begin
- buf[i-1] := '0';
- k := 1;
- end;
- end;
- If (k=1) and (buf[i-1]='0') then
- begin
- { 1.9996 rounded to two decimal digits after the decimal separator must result in
- 2.00, i.e. the rounding is propagated
- }
- while buf[i]='9' do
- begin
- buf[i]:='0';
- inc(i);
- end;
- buf[i]:=chr(Ord(buf[i])+1);
- { did we add another digit? This happens when rounding
- e.g. 99.9996 to two decimal digits after the decimal separator which should result in
- 100.00
- }
- if i>tlen then
- begin
- inc(reslen);
- inc(tlen);
- end;
- end;
- end;
- { preparing result string }
- if reslen<len then
- reslen:=len;
- if reslen>High(s) then
- begin
- if r < 0 then
- Inc(r, reslen - High(s));
- reslen:=High(s);
- end;
- SetLength(s,reslen);
- j:=reslen;
- if f<0 then
- begin
- { writing power of 10 part }
- if PInt64(@c)^ = 0 then
- k:=0
- else
- k:=tlen-5;
- if k >= 0 then
- s[j-2]:='+'
- else
- begin
- s[j-2]:='-';
- k:=-k;
- end;
- s[j]:=Chr(k mod 10 + $30);
- Dec(j);
- s[j]:=Chr(k div 10 + $30);
- Dec(j,2);
- s[j]:='E';
- Dec(j);
- end;
- { writing extra zeroes if r < 0 }
- while r < 0 do
- begin
- s[j]:='0';
- Dec(j);
- Inc(r);
- end;
- { writing digits and decimal point }
- for i:=r + 1 to tlen do
- begin
- Dec(point);
- if point = 0 then
- begin
- s[j]:='.';
- Dec(j);
- end;
- s[j]:=buf[i];
- Dec(j);
- end;
- { writing sign }
- if sign = 1 then
- begin
- s[j]:='-';
- Dec(j);
- end;
- { writing spaces }
- while j > 0 do
- begin
- s[j]:=' ';
- Dec(j);
- end;
- end;
- {
- Array Of Char Str() helpers
- }
- procedure fpc_chararray_sint(v : valsint;len : SizeInt;out a:array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- procedure fpc_chararray_uint(v : valuint;len : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str_unsigned(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$ifndef CPU64}
- procedure fpc_chararray_qword(v : qword;len : SizeInt;out a : array of char);compilerproc;
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- begin
- runerror(219);
- end;
- {$else EXCLUDE_COMPLEX_PROCS}
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str_unsigned(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$endif EXCLUDE_COMPLEX_PROCS}
- procedure fpc_chararray_int64(v : int64;len : SizeInt;out a : array of char);compilerproc;
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- begin
- runerror(219);
- end;
- {$else EXCLUDE_COMPLEX_PROCS}
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$endif EXCLUDE_COMPLEX_PROCS}
- {$endif CPU64}
- {$if defined(CPU16) or defined(CPU8)}
- procedure fpc_chararray_longword(v : longword;len : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str_unsigned(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- procedure fpc_chararray_longint(v : longint;len : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- procedure fpc_chararray_word(v : word;len : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str_unsigned(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- procedure fpc_chararray_smallint(v : smallint;len : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- int_str(v,ss);
- if length(ss)<len then
- ss:=space(len-length(ss))+ss;
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$endif CPU16 or CPU8}
- {$ifndef FPUNONE}
- procedure fpc_chararray_Float(d : ValReal;len,fr,rt : SizeInt;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- str_real(len,fr,d,treal_type(rt),ss);
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$endif}
- {$ifndef FPC_STR_ENUM_INTERN}
- { currently, the avr code generator fails on this procedure, so we disable it,
- this is not a good solution but fixing compilation of this procedure for
- avr is hard, requires significant changes to the register allocator to take
- care of different register classes }
- procedure fpc_chararray_enum(ordinal,len:sizeint;typinfo,ord2strindex:pointer;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- runerror(219);
- {$else EXCLUDE_COMPLEX_PROCS}
- fpc_shortstr_enum(ordinal,len,typinfo,ord2strindex,ss);
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- {$endif EXCLUDE_COMPLEX_PROCS}
- end;
- {$endif not FPC_STR_ENUM_INTERN}
- procedure fpc_chararray_bool(b : boolean;len:sizeint;out a : array of char);compilerproc;
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- fpc_shortstr_bool(b,len,ss);
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$ifndef FPC_HAS_CHARARRAY_CURRENCY}
- {$define FPC_HAS_CHARARRAY_CURRENCY}
- procedure fpc_chararray_Currency(c : Currency;len,fr : SizeInt;out a : array of char);compilerproc;
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- begin
- runerror(217);
- end;
- {$else EXCLUDE_COMPLEX_PROCS}
- var
- ss : shortstring;
- maxlen : SizeInt;
- begin
- str(c:len:fr,ss);
- if length(ss)<high(a)+1 then
- maxlen:=length(ss)
- else
- maxlen:=high(a)+1;
- fpc_shortstr_chararray_intern_charmove(ss,a,maxlen);
- end;
- {$endif EXCLUDE_COMPLEX_PROCS}
- {$endif FPC_HAS_CHARARRAY_CURRENCY}
- {*****************************************************************************
- Val() Functions
- *****************************************************************************}
- Function InitVal(const s:shortstring;out negativ:boolean;out base:byte):ValSInt;
- var
- Code : SizeInt;
- begin
- code:=1;
- negativ:=false;
- base:=10;
- if length(s)=0 then
- begin
- InitVal:=code;
- Exit;
- end;
- {Skip Spaces and Tab}
- while (code<=length(s)) and (s[code] in [' ',#9]) do
- inc(code);
- {Sign}
- case s[code] of
- '-' : begin
- negativ:=true;
- inc(code);
- end;
- '+' : inc(code);
- end;
- {Base}
- if code<=length(s) then
- begin
- case s[code] of
- '$',
- 'X',
- 'x' : begin
- base:=16;
- inc(code);
- end;
- '%' : begin
- base:=2;
- inc(code);
- end;
- '&' : begin
- Base:=8;
- inc(code);
- end;
- '0' : begin
- if (code < length(s)) and (s[code+1] in ['x', 'X']) then
- begin
- inc(code, 2);
- base := 16;
- end;
- end;
- end;
- end;
- { strip leading zeros }
- while ((code < length(s)) and (s[code] = '0')) do begin
- inc(code);
- end;
- InitVal:=code;
- end;
- Function fpc_Val_SInt_ShortStr(DestSize: SizeInt; Const S: ShortString; out Code: ValSInt): ValSInt; [public, alias:'FPC_VAL_SINT_SHORTSTR']; compilerproc;
- var
- temp, prev, maxPrevValue, maxNewValue: ValUInt;
- base,u : byte;
- negative : boolean;
- begin
- fpc_Val_SInt_ShortStr := 0;
- Temp:=0;
- Code:=InitVal(s,negative,base);
- if Code>length(s) then
- exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- maxPrevValue := ValUInt(MaxUIntValue) div ValUInt(Base);
- if (base = 10) then
- maxNewValue := MaxSIntValue + ord(negative)
- else
- maxNewValue := MaxUIntValue;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- Prev := Temp;
- Temp := Temp*ValUInt(base);
- If (u >= base) or
- (ValUInt(maxNewValue-u) < Temp) or
- (prev > maxPrevValue) Then
- Begin
- fpc_Val_SInt_ShortStr := 0;
- Exit
- End;
- Temp:=Temp+u;
- inc(code);
- end;
- code := 0;
- fpc_Val_SInt_ShortStr := ValSInt(Temp);
- If Negative Then
- fpc_Val_SInt_ShortStr := -fpc_Val_SInt_ShortStr;
- If Not(Negative) and (base <> 10) Then
- {sign extend the result to allow proper range checking}
- Case DestSize of
- 1: fpc_Val_SInt_ShortStr := shortint(fpc_Val_SInt_ShortStr);
- 2: fpc_Val_SInt_ShortStr := smallint(fpc_Val_SInt_ShortStr);
- {$ifdef cpu64}
- 4: fpc_Val_SInt_ShortStr := longint(fpc_Val_SInt_ShortStr);
- {$endif cpu64}
- End;
- end;
- {$ifndef FPC_HAS_INT_VAL_SINT_SHORTSTR}
- {$define FPC_HAS_INT_VAL_SINT_SHORTSTR}
- { we need this for fpc_Val_SInt_Ansistr and fpc_Val_SInt_WideStr because }
- { we have to pass the DestSize parameter on (JM) }
- Function int_Val_SInt_ShortStr(DestSize: SizeInt; Const S: ShortString; out Code: ValSInt): ValSInt; [external name 'FPC_VAL_SINT_SHORTSTR'];
- {$endif FPC_HAS_INT_VAL_SINT_SHORTSTR}
- Function fpc_Val_UInt_Shortstr(Const S: ShortString; out Code: ValSInt): ValUInt; [public, alias:'FPC_VAL_UINT_SHORTSTR']; compilerproc;
- var
- base,u : byte;
- negative : boolean;
- begin
- fpc_Val_UInt_Shortstr:=0;
- Code:=InitVal(s,negative,base);
- If Negative or (Code>length(s)) Then
- Exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- If (u>=base) or
- (ValUInt(MaxUIntValue-u) div ValUInt(Base)<fpc_val_uint_shortstr) then
- begin
- fpc_Val_UInt_Shortstr:=0;
- exit;
- end;
- fpc_Val_UInt_Shortstr:=fpc_Val_UInt_Shortstr*ValUInt(base) + u;
- inc(code);
- end;
- code := 0;
- end;
- {$ifndef CPU64}
- Function fpc_val_int64_shortstr(Const S: ShortString; out Code: ValSInt): Int64; [public, alias:'FPC_VAL_INT64_SHORTSTR']; compilerproc;
- var u : sizeuint;
- temp, prev, maxprevvalue, maxnewvalue : qword;
- base : byte;
- negative : boolean;
- const maxint64=qword($7fffffffffffffff);
- maxqword=qword($ffffffffffffffff);
- begin
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- runerror(219);
- {$else EXCLUDE_COMPLEX_PROCS}
- fpc_val_int64_shortstr := 0;
- Temp:=0;
- Code:=InitVal(s,negative,base);
- if Code>length(s) then
- exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- maxprevvalue := maxqword div base;
- if (base = 10) then
- maxnewvalue := maxint64 + ord(negative)
- else
- maxnewvalue := maxqword;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- Prev:=Temp;
- Temp:=Temp*qword(base);
- If (u >= base) or
- (qword(maxnewvalue-u) < temp) or
- (prev > maxprevvalue) Then
- Begin
- fpc_val_int64_shortstr := 0;
- Exit
- End;
- Temp:=Temp+u;
- inc(code);
- end;
- code:=0;
- fpc_val_int64_shortstr:=int64(Temp);
- If Negative Then
- fpc_val_int64_shortstr:=-fpc_val_int64_shortstr;
- {$endif EXCLUDE_COMPLEX_PROCS}
- end;
- Function fpc_val_qword_shortstr(Const S: ShortString; out Code: ValSInt): QWord; [public, alias:'FPC_VAL_QWORD_SHORTSTR']; compilerproc;
- var u : sizeuint;
- base : byte;
- negative : boolean;
- const maxqword=qword($ffffffffffffffff);
- begin
- fpc_val_qword_shortstr:=0;
- Code:=InitVal(s,negative,base);
- If Negative or (Code>length(s)) Then
- Exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- If (u>=base) or
- ((QWord(maxqword-u) div QWord(base))<fpc_val_qword_shortstr) then
- Begin
- fpc_val_qword_shortstr := 0;
- Exit
- End;
- fpc_val_qword_shortstr:=fpc_val_qword_shortstr*QWord(base) + u;
- inc(code);
- end;
- code := 0;
- end;
- {$endif CPU64}
- {$if defined(CPU16) or defined(CPU8)}
- Function fpc_val_longint_shortstr(Const S: ShortString; out Code: ValSInt): LongInt; [public, alias:'FPC_VAL_LONGINT_SHORTSTR']; compilerproc;
- var u, temp, prev, maxprevvalue, maxnewvalue : longword;
- base : byte;
- negative : boolean;
- const maxlongint=longword($7fffffff);
- maxlongword=longword($ffffffff);
- begin
- fpc_val_longint_shortstr := 0;
- Temp:=0;
- Code:=InitVal(s,negative,base);
- if Code>length(s) then
- exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- maxprevvalue := maxlongword div base;
- if (base = 10) then
- maxnewvalue := maxlongint + ord(negative)
- else
- maxnewvalue := maxlongword;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- Prev:=Temp;
- Temp:=Temp*longword(base);
- If (u >= base) or
- (longword(maxnewvalue-u) < temp) or
- (prev > maxprevvalue) Then
- Begin
- fpc_val_longint_shortstr := 0;
- Exit
- End;
- Temp:=Temp+u;
- inc(code);
- end;
- code:=0;
- fpc_val_longint_shortstr:=longint(Temp);
- If Negative Then
- fpc_val_longint_shortstr:=-fpc_val_longint_shortstr;
- end;
- Function fpc_val_longword_shortstr(Const S: ShortString; out Code: ValSInt): LongWord; [public, alias:'FPC_VAL_LONGWORD_SHORTSTR']; compilerproc;
- var u, prev: LongWord;
- base : byte;
- negative : boolean;
- const maxlongword=longword($ffffffff);
- begin
- fpc_val_longword_shortstr:=0;
- Code:=InitVal(s,negative,base);
- If Negative or (Code>length(s)) Then
- Exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- prev := fpc_val_longword_shortstr;
- If (u>=base) or
- ((LongWord(maxlongword-u) div LongWord(base))<prev) then
- Begin
- fpc_val_longword_shortstr := 0;
- Exit
- End;
- fpc_val_longword_shortstr:=fpc_val_longword_shortstr*LongWord(base) + u;
- inc(code);
- end;
- code := 0;
- end;
- Function fpc_val_smallint_shortstr(Const S: ShortString; out Code: ValSInt): SmallInt; [public, alias:'FPC_VAL_SMALLINT_SHORTSTR']; compilerproc;
- var u, temp, prev, maxprevvalue, maxnewvalue : word;
- base : byte;
- negative : boolean;
- const maxlongint=longword($7fffffff);
- maxlongword=longword($ffffffff);
- begin
- fpc_val_smallint_shortstr := 0;
- Temp:=0;
- Code:=InitVal(s,negative,base);
- if Code>length(s) then
- exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- maxprevvalue := maxlongword div base;
- if (base = 10) then
- maxnewvalue := maxlongint + ord(negative)
- else
- maxnewvalue := maxlongword;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- Prev:=Temp;
- Temp:=Temp*longword(base);
- If (u >= base) or
- (longword(maxnewvalue-u) < temp) or
- (prev > maxprevvalue) Then
- Begin
- fpc_val_smallint_shortstr := 0;
- Exit
- End;
- Temp:=Temp+u;
- inc(code);
- end;
- code:=0;
- fpc_val_smallint_shortstr:=longint(Temp);
- If Negative Then
- fpc_val_smallint_shortstr:=-fpc_val_smallint_shortstr;
- end;
- Function fpc_val_word_shortstr(Const S: ShortString; out Code: ValSInt): Word; [public, alias:'FPC_VAL_WORD_SHORTSTR']; compilerproc;
- var u, prev: word;
- base : byte;
- negative : boolean;
- const maxlongword=longword($ffffffff);
- begin
- fpc_val_word_shortstr:=0;
- Code:=InitVal(s,negative,base);
- If Negative or (Code>length(s)) Then
- Exit;
- if (s[Code]=#0) then
- begin
- if (Code>1) and (s[Code-1]='0') then
- Code:=0;
- exit;
- end;
- while Code<=Length(s) do
- begin
- case s[Code] of
- '0'..'9' : u:=Ord(S[Code])-Ord('0');
- 'A'..'F' : u:=Ord(S[Code])-(Ord('A')-10);
- 'a'..'f' : u:=Ord(S[Code])-(Ord('a')-10);
- #0 : break;
- else
- u:=16;
- end;
- prev := fpc_val_word_shortstr;
- If (u>=base) or
- ((LongWord(maxlongword-u) div LongWord(base))<prev) then
- Begin
- fpc_val_word_shortstr := 0;
- Exit
- End;
- fpc_val_word_shortstr:=fpc_val_word_shortstr*LongWord(base) + u;
- inc(code);
- end;
- code := 0;
- end;
- {$endif CPU16 or CPU8}
- {$ifdef FLOAT_ASCII_FALLBACK}
- {$ifndef FPUNONE}
- const
- {$ifdef FPC_HAS_TYPE_EXTENDED}
- valmaxexpnorm=4932;
- mantissabits=64;
- {$else}
- {$ifdef FPC_HAS_TYPE_DOUBLE}
- valmaxexpnorm=308;
- mantissabits=53;
- {$else}
- {$ifdef FPC_HAS_TYPE_SINGLE}
- valmaxexpnorm=38;
- mantissabits=24;
- {$else}
- {$error Unknown floating point precision }
- {$endif}
- {$endif}
- {$endif}
- {$endif}
- {$ifndef FPUNONE}
- (******************
- Derived from: ".\Free Pascal\source\rtl\inc\genmath.inc"
- Origin: "fast 10^n routine"
- function FPower10(val: Extended; Power: Longint): Extended;
- Changes:
- > adapted to "ValReal", so float can be single/double/extended
- > slightly changed arrays [redundant 58+2 float constants gone away]
- > added some checks etc..
- Notes:
- > denormalization and overflow should go smooth if corresponding
- FPU exceptions are masked [no external care needed by now]
- > adaption to real48 and real128 is not hard if one needed
- ******************)
- //
- function mul_by_power10(x:ValReal;power:integer):ValReal;
- //
- // result:=X*(10^power)
- //
- // Routine achieves result with no more than 3 floating point mul/div's.
- // Up to ABS(power)=31, only 1 floating point mul/div is needed.
- //
- // Limitations:
- // for ValReal=extended : power=-5119..+5119
- // for ValReal=double : power=-319..+319
- // for ValReal=single : power=-63..+63
- //
- // If "power" is beyond this limits, routine gives up and returns 0/+INF/-INF.
- // This is not generally correct, but should be ok when routine is used only
- // as "VAL"-helper, since "x" exponent is reasonably close to 0 in this case.
- //
- //==================================
- {$IF DECLARED(C_HIGH_EXPBITS_5TO8)}
- {$ERROR C_HIGH_EXPBITS_5TO8 declared somewhere in scope}
- {$ENDIF}
- {$IF DECLARED(C_HIGH_EXPBITS_9ANDUP)}
- {$ERROR C_HIGH_EXPBITS_9ANDUP declared somewhere in scope}
- {$ENDIF}
- {$IF SIZEOF(ValReal)=10}
- //==================================
- // assuming "type ValReal=extended;"
- //
- const
- C_MAX_POWER = 5119;
- C_HIGH_EXPBITS_5TO8 = 15;
- C_HIGH_EXPBITS_9ANDUP = 9;
- {$ELSEIF SIZEOF(ValReal)=8}
- //==================================
- // assuming "type ValReal=double;"
- //
- const
- C_MAX_POWER = 319;
- C_HIGH_EXPBITS_5TO8 = 9;
- {$ELSEIF SIZEOF(ValReal)=4}
- //==================================
- // assuming "type ValReal=single;"
- //
- const
- C_MAX_POWER = 63;
- {$ELSE}
- //==================================
- // assuming "ValReal=?"
- //
- {$ERROR Unsupported ValReal type}
- {$ENDIF}
- //==================================
- const
- C_INFTYP = ValReal( 1.0/0.0);
- C_INFTYM = ValReal(-1.0/0.0);
- mul_expbits_0_to_4:packed array[0..31]of ValReal=(
- 1E0, 1E1, 1E2, 1E3,
- 1E4, 1E5, 1E6, 1E7,
- 1E8, 1E9, 1E10, 1E11,
- 1E12, 1E13, 1E14, 1E15,
- 1E16, 1E17, 1E18, 1E19,
- 1E20, 1E21, 1E22, 1E23,
- 1E24, 1E25, 1E26, 1E27,
- 1E28, 1E29, 1E30, 1E31);
- {$IF DECLARED(C_HIGH_EXPBITS_5TO8)}
- mul_expbits_5_to_8:packed array[1..C_HIGH_EXPBITS_5TO8] of ValReal=(
- 1E32, 1E64, 1E96, 1E128,
- 1E160, 1E192, 1E224, 1E256, 1E288
- {$IF DECLARED(C_HIGH_EXPBITS_9ANDUP)},
- 1E320, 1E352, 1E384, 1E416, 1E448, 1E480
- {$ENDIF});
- {$ELSE}
- mul_expbits_5_to_8:ValReal=1E32;
- {$ENDIF}
- {$IF DECLARED(C_HIGH_EXPBITS_9ANDUP)}
- mul_expbits_9_and_up:packed array[1..C_HIGH_EXPBITS_9ANDUP] of ValReal=(
- 1E512, 1E1024, 1E1536, 1E2048,
- 1E2560, 1E3072, 1E3584, 1E4096,
- 1E4608);
- {$ENDIF}
- begin
- if power=0 then mul_by_power10:=x else
- if power<-C_MAX_POWER then mul_by_power10:=0 else
- if power>C_MAX_POWER then
- if x<0 then mul_by_power10:=C_INFTYM else
- if x>0 then mul_by_power10:=C_INFTYP else mul_by_power10:=0
- else
- if power<0 then
- begin
- power:=-power;
- mul_by_power10:=x/mul_expbits_0_to_4[power and $1F];
- power:=(power shr 5);
- if power=0 then exit;
- {$IF DECLARED(C_HIGH_EXPBITS_5TO8)}
- if power and $F<>0 then
- mul_by_power10:=
- mul_by_power10/mul_expbits_5_to_8[power and $F];
- {$ELSE} // "single", power<>0, so always div
- mul_by_power10:=mul_by_power10/mul_expbits_5_to_8;
- {$ENDIF}
- {$IF DECLARED(C_HIGH_EXPBITS_9ANDUP)}
- power:=(power shr 4);
- if power<>0 then
- mul_by_power10:=
- mul_by_power10/mul_expbits_9_and_up[power];
- {$ENDIF}
- end
- else
- begin
- mul_by_power10:=x*mul_expbits_0_to_4[power and $1F];
- power:=(power shr 5);
- if power=0 then exit;
- {$IF DECLARED(C_HIGH_EXPBITS_5TO8)}
- if power and $F<>0 then
- mul_by_power10:=
- mul_by_power10*mul_expbits_5_to_8[power and $F];
- {$ELSE} // "single", power<>0, so always mul
- mul_by_power10:=mul_by_power10*mul_expbits_5_to_8;
- {$ENDIF}
- {$IF DECLARED(C_HIGH_EXPBITS_9ANDUP)}
- power:=(power shr 4);
- if power<>0 then
- mul_by_power10:=
- mul_by_power10*mul_expbits_9_and_up[power];
- {$ENDIF}
- end;
- end;
- Function fpc_Val_Real_ShortStr(const s : shortstring; out Code : ValSInt): ValReal; [public, alias:'FPC_VAL_REAL_SHORTSTR']; compilerproc;
- var
- hd,
- sign : valreal;
- esign,
- exponent,
- expstart,
- decpoint : SizeInt;
- nint,
- nlz,
- explimit,
- explastdigit: SizeInt;
- begin
- fpc_Val_Real_ShortStr:=0.0;
- code:=1;
- exponent:=0;
- decpoint:=0;
- esign:=1;
- hd:=0.0;
- nlz:=0;
- nint:=0;
- sign:=1;
- while (code<=length(s)) and (s[code] in [' ',#9]) do
- inc(code);
- if code<=length(s) then
- case s[code] of
- '+' : inc(code);
- '-' : begin
- sign:=-1;
- inc(code);
- end;
- end;
- { leading zeroes do not influence result, skip all but one of them }
- expstart:=code;
- while (code<Length(s)) and (s[code]='0') do
- inc(code);
- if (code>expstart) then
- dec(code);
- expstart:=code;
- while (Code<=Length(s)) do
- begin
- case s[code] of
- '0':
- begin
- if (hd=0) then
- inc(nlz,ord(decpoint<>0))
- else
- inc(nint,ord(decpoint=0));
- hd:=hd*10;
- end;
- '1'..'9':
- begin
- if (decpoint=0) then
- inc(nint);
- hd:=hd*10+(ord(s[code])-ord('0'));
- end;
- '.':
- if decpoint=0 then
- decpoint:=code
- else
- exit;
- else
- break;
- end;
- inc(code);
- end;
- { must have seen at least one digit }
- if (code-expstart)<1+ord(decpoint<>0) then
- exit;
- if decpoint<>0 then
- decpoint:=code-decpoint-1;
- { Exponent ? }
- if (length(s)>=code) and (s[code] in ['e','E']) then
- begin
- inc(code);
- if Length(s) >= code then
- case s[code] of
- '+': inc(code);
- '-': begin
- esign:=-1;
- inc(code);
- end;
- end;
- expstart:=code;
- { Limit the exponent, accounting for digits in integer part of mantissa
- and leading zeros in fractional part, e.g 100.0e306 = 1.0e308, etc. }
- if (esign<0) then
- explimit:=valmaxexpnorm+mantissabits-1+nint
- else if (nint>0) then
- explimit:=valmaxexpnorm+1-nint
- else
- explimit:=valmaxexpnorm+1+nlz;
- explastdigit:=(explimit mod 10)+ord('0');
- explimit:=explimit div 10;
- while (length(s)>=code) and (s[code] in ['0'..'9']) do
- begin
- { Check commented out: since this code is used by compiler, it would error out
- e.g. if compiling '1e3000' for non-x86 target. OTOH silently treating it
- as infinity isn't a good option either. }
- (*
- if (exponent>explimit) or
- ((exponent=explimit) and (ord(s[code])>explastdigit)) then
- begin
- { ignore exponent overflow for zero mantissa }
- if hd<>0.0 then
- exit;
- end
- else *)
- exponent:=exponent*10+(ord(s[code])-ord('0'));
- inc(code);
- end;
- if code=expstart then
- exit;
- end;
- { Not all characters are read ? }
- if length(s)>=code then
- exit;
- { adjust exponent based on decimal point }
- dec(exponent,decpoint*esign);
- if (exponent<0) then
- begin
- esign:=-1;
- exponent:=-exponent;
- end;
- { evaluate sign }
- { (before exponent, because the exponent may turn it into a denormal) }
- fpc_Val_Real_ShortStr:=hd*sign;
- { Calculate Exponent }
- hd:=1.0;
- { the magnitude range maximum (normal) is lower in absolute value than the }
- { the magnitude range minimum (denormal). E.g. an extended value can go }
- { up to 1E4932, but "down" to 1E-4951. So make sure that we don't try to }
- { calculate 1E4951 as factor, since that would overflow and result in 0. }
- if (exponent>valmaxexpnorm-2) then
- begin
- hd:=mul_by_power10(hd,valmaxexpnorm-2);
- if esign>0 then
- fpc_Val_Real_ShortStr:=fpc_Val_Real_ShortStr*hd
- else
- fpc_Val_Real_ShortStr:=fpc_Val_Real_ShortStr/hd;
- dec(exponent,valmaxexpnorm-2);
- hd:=1.0;
- end;
- hd:=mul_by_power10(hd,exponent);
- if esign>0 then
- fpc_Val_Real_ShortStr:=fpc_Val_Real_ShortStr*hd
- else
- fpc_Val_Real_ShortStr:=fpc_Val_Real_ShortStr/hd;
- { success ! }
- code:=0;
- end;
- {$endif}
- {$else not FLOAT_ASCII_FALLBACK}
- {$ifndef FPUNONE}
- Function fpc_Val_Real_ShortStr(const s : shortstring; out Code : ValSInt): ValReal; [public, alias:'FPC_VAL_REAL_SHORTSTR']; compilerproc;
- begin
- fpc_Val_Real_ShortStr := val_real( s, code );
- end;
- {$endif FPUNONE}
- {$endif FLOAT_ASCII_FALLBACK}
- {$ifndef FPC_STR_ENUM_INTERN}
- function fpc_val_enum_shortstr(str2ordindex:pointer;const s:shortstring;out code:valsint):longint; [public, alias:'FPC_VAL_ENUM_SHORTSTR']; compilerproc;
- function string_compare(const s1,s2:shortstring):sizeint;
- {We cannot use the > and < operators to compare a string here, because we if the string is
- not found in the enum, we need to return the position of error in "code". Code equals the
- highest matching character of all string compares, which is only known inside the string
- comparison.}
- var i,l:byte;
- c1,c2:char;
- begin
- l:=length(s1);
- if length(s1)>length(s2) then
- l:=length(s2);
- i:=1;
- while i<=l do
- begin
- c1:=s1[i];
- c2:=s2[i];
- if c1<>c2 then
- break;
- inc(i);
- end;
- if i>code then
- code:=i;
- if i<=l then
- string_compare:=byte(c1)-byte(c2)
- else
- string_compare:=length(s1)-length(s2);
- end;
- type Psorted_array=^Tsorted_array;
- Tsorted_array={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- o:longint;
- s:Pstring;
- end;
- Pstring_to_ord=^Tstring_to_ord;
- Tstring_to_ord={$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}packed{$endif}record
- count:longint;
- data:array[0..0] of Tsorted_array;
- end;
- var l,h,m:cardinal;
- c:sizeint;
- sorted_array:^Tsorted_array;
- spaces:byte;
- t:shortstring;
- begin
- {Val for numbers accepts spaces at the start, so lets do the same
- for enums. Skip spaces at the start of the string.}
- spaces:=1;
- code:=1;
- while (spaces<=length(s)) and (s[spaces]=' ') do
- inc(spaces);
- t:=upcase(copy(s,spaces,255));
- sorted_array:=pointer(@Pstring_to_ord(str2ordindex)^.data);
- {Use a binary search to get the string.}
- l:=1;
- h:=Pstring_to_ord(str2ordindex)^.count;
- repeat
- m:=(l+h) div 2;
- c:=string_compare(t,upcase(sorted_array[m-1].s^));
- if c>0 then
- l:=m+1
- else if c<0 then
- h:=m-1
- else
- break;
- if l>h then
- begin
- {Not found...}
- inc(code,spaces-1); {Add skipped spaces again.}
- {The result of val in case of error is undefined, don't assign a function result.}
- exit;
- end;
- until false;
- code:=0;
- fpc_val_enum_shortstr:=sorted_array[m-1].o;
- end;
- {Redeclare fpc_val_enum_shortstr for internal use in the system unit.}
- function fpc_val_enum_shortstr(str2ordindex:pointer;const s:shortstring;out code:valsint):longint;external name 'FPC_VAL_ENUM_SHORTSTR';
- {$endif FPC_STR_ENUM_INTERN}
- function fpc_Val_Currency_ShortStr(const s : shortstring; out Code : ValSInt): currency; [public, alias:'FPC_VAL_CURRENCY_SHORTSTR']; compilerproc;
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- begin
- runerror(217);
- end;
- {$else EXCLUDE_COMPLEX_PROCS}
- const
- MinInt64 : Int64 =-$8000000000000000;
- MinInt64Edge : Int64 = (-$8000000000000000 + 10) div 10;
- var
- { to enable taking the address on the JVM target }
- res : array[0..0] of Int64;
- i,j,power,sign,len : longint;
- FracOverflow : boolean;
- begin
- fpc_Val_Currency_ShortStr:=0;
- res[0]:=0;
- len:=Length(s);
- Code:=1;
- sign:=-1;
- power:=0;
- while True do
- if Code > len then
- exit
- else
- if s[Code] in [' ', #9] then
- Inc(Code)
- else
- break;
- { Read sign }
- case s[Code] of
- '+' : begin
- Inc(Code);
- end;
- '-' : begin
- sign:=+1;
- Inc(Code);
- end;
- end;
- { Read digits }
- FracOverflow:=False;
- i:=0;
- while Code <= len do
- begin
- case s[Code] of
- '0'..'9':
- begin
- j:=Ord(s[code])-Ord('0');
- { check overflow }
- if (res[0] >= MinInt64Edge) or (res[0] >= (MinInt64 + j) div 10) then
- begin
- res[0]:=res[0]*10 - j;
- Inc(i);
- end
- else
- if power = 0 then
- { exit if integer part overflow }
- exit
- else
- begin
- if not FracOverflow and (j >= 5) and (res[0] > MinInt64) then
- { round if first digit of fractional part overflow }
- Dec(res[0]);
- FracOverflow:=True;
- end;
- end;
- '.':
- begin
- if power = 0 then
- begin
- power:=1;
- i:=0;
- end
- else
- exit;
- end;
- else
- break;
- end;
- Inc(Code);
- end;
- if (i = 0) and (power = 0) then
- exit;
- if power <> 0 then
- power:=i;
- power:=4 - power;
- { Exponent? }
- if Code <= len then
- if s[Code] in ['E', 'e'] then
- begin
- Inc(Code);
- if Code > len then
- exit;
- i:=1;
- case s[Code] of
- '+':
- Inc(Code);
- '-':
- begin
- i:=-1;
- Inc(Code);
- end;
- end;
- { read exponent }
- j:=0;
- while Code <= len do
- if s[Code] in ['0'..'9'] then
- begin
- if j > 4951 then
- exit;
- j:=j*10 + (Ord(s[code])-Ord('0'));
- Inc(Code);
- end
- else
- exit;
- power:=power + j*i;
- end
- else
- exit;
- if power > 0 then
- begin
- for i:=1 to power do
- if res[0] >= MinInt64 div 10 then
- res[0]:=res[0]*10
- else
- exit;
- end
- else
- for i:=1 to -power do
- begin
- if res[0] >= MinInt64 + 5 then
- Dec(res[0], 5);
- res[0]:=res[0] div 10;
- end;
- if sign <> 1 then
- if res[0] > MinInt64 then
- res[0]:=res[0]*sign
- else
- exit;
- fpc_Val_Currency_ShortStr:=PCurrency(@res[0])^;
- Code:=0;
- end;
- {$endif EXCLUDE_COMPLEX_PROCS}
- {$ifndef FPC_HAS_SETSTRING_SHORTSTR}
- {$define FPC_HAS_SETSTRING_SHORTSTR}
- Procedure {$ifdef FPC_HAS_CPSTRING}fpc_setstring_shortstr{$else}SetString{$endif}(Out S : Shortstring; Buf : PChar; Len : SizeInt); {$ifdef FPC_HAS_CPSTRING} compilerproc; {$endif FPC_HAS_CPSTRING}
- begin
- If Len > High(S) then
- Len := High(S);
- SetLength(S,Len);
- If Buf<>Nil then
- begin
- Move (Buf[0],S[1],Len);
- end;
- end;
- {$endif FPC_HAS_SETSTRING_SHORTSTR}
- {$ifndef FPC_HAS_COMPARETEXT_SHORTSTR}
- {$define FPC_HAS_COMPARETEXT_SHORTSTR}
- function ShortCompareText(const S1, S2: shortstring): SizeInt;
- var
- c1, c2: Byte;
- i: SizeInt;
- L1, L2, Count: SizeInt;
- P1, P2: PChar;
- begin
- L1 := Length(S1);
- L2 := Length(S2);
- if L1 > L2 then
- Count := L2
- else
- Count := L1;
- i := 0;
- P1 := @S1[1];
- P2 := @S2[1];
- while i < count do
- begin
- c1 := byte(p1^);
- c2 := byte(p2^);
- if c1 <> c2 then
- begin
- if c1 in [97..122] then
- Dec(c1, 32);
- if c2 in [97..122] then
- Dec(c2, 32);
- if c1 <> c2 then
- Break;
- end;
- Inc(P1); Inc(P2); Inc(I);
- end;
- if i < count then
- ShortCompareText := c1 - c2
- else
- ShortCompareText := L1 - L2;
- end;
- {$endif FPC_HAS_COMPARETEXT_SHORTSTR}
|