| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by the Free Pascal development team.    Processor dependent implementation for the system unit for    intel i386+    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. **********************************************************************}{****************************************************************************                               Primitives****************************************************************************}var  has_sse_support,has_mmx_support,os_supports_sse : boolean;{$asmmode intel}function cpuid_support : boolean;assembler;  {    Check if the ID-flag can be changed, if changed then CpuID is supported.    Tested under go32v1 and Linux on c6x86 with CpuID enabled and disabled (PFV)  }  asm    push   ebx    pushf    pushf    pop     eax    mov     ebx,eax    xor     eax,200000h    push    eax    popf    pushf    pop     eax    popf    and     eax,200000h    and     ebx,200000h    cmp     eax,ebx    setnz   al    pop     ebx  end;{$asmmode ATT}function sse_support : boolean;  var     _edx : longint;  begin    if cpuid_support then     begin        asm          pushl %ebx          movl $1,%eax          cpuid          movl %edx,_edx          popl %ebx                  end;        sse_support:=((_edx and $2000000)<>0) and os_supports_sse;     end    else     { a cpu with without cpuid instruction supports never sse }     sse_support:=false;  end;{ returns true, if the processor supports the mmx instructions }function mmx_support : boolean;  var     _edx : longint;  begin     if cpuid_support then       begin          asm             pushl %ebx             movl $1,%eax             cpuid             movl %edx,_edx             popl %ebx                      end;          mmx_support:=(_edx and $800000)<>0;       end     else       { a cpu with without cpuid instruction supports never mmx }       mmx_support:=false;  end;{$define USE_FASTMOVE}{$i fastmove.inc}procedure fpc_cpuinit;  begin    { because of the brain dead sse detection on x86, this test is post poned to fpc_cpucodeinit which      must be implemented OS dependend (FK)    has_sse_support:=sse_support;    has_mmx_support:=mmx_support;    setup_fastmove;    }    os_supports_sse:=false;  end;{$ifndef FPC_SYSTEM_HAS_MOVE}{$define FPC_SYSTEM_HAS_MOVE}procedure Move(const source;var dest;count:SizeInt);[public, alias: 'FPC_MOVE'];assembler;var  saveesi,saveedi : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi{$ifdef REGCALL}        movl    %eax,%esi        movl    %edx,%edi        movl    %ecx,%edx{$else}        movl    dest,%edi        movl    source,%esi        movl    count,%edx{$endif}        movl    %edi,%eax{ check for zero or negative count }        cmpl    $0,%edx        jle     .LMoveEnd{ Check for back or forward }        sub     %esi,%eax        jz      .LMoveEnd               { Do nothing when source=dest }        jc      .LFMove                 { Do forward, dest<source }        cmp     %edx,%eax        jb      .LBMove                 { Dest is in range of move, do backward }{ Forward Copy }.LFMove:        cld        cmpl    $15,%edx        jl      .LFMove1        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%edx        rep        movsb        movl    %edx,%ecx        andl    $3,%edx        shrl    $2,%ecx        rep        movsl.LFMove1:        movl    %edx,%ecx        rep        movsb        jmp .LMoveEnd{ Backward Copy }.LBMove:        std        addl    %edx,%esi        addl    %edx,%edi        movl    %edi,%ecx        decl    %esi        decl    %edi        cmpl    $15,%edx        jl      .LBMove1        negl    %ecx            { Align on 32bits }        andl    $3,%ecx        subl    %ecx,%edx        rep        movsb        movl    %edx,%ecx        andl    $3,%edx        shrl    $2,%ecx        subl    $3,%esi        subl    $3,%edi        rep        movsl        addl    $3,%esi        addl    $3,%edi.LBMove1:        movl    %edx,%ecx        rep        movsb        cld.LMoveEnd:        movl    saveedi,%edi        movl    saveesi,%esiend;{$endif FPC_SYSTEM_HAS_MOVE}{$ifndef FPC_SYSTEM_HAS_FILLCHAR}{$define FPC_SYSTEM_HAS_FILLCHAR}Procedure FillChar(var x;count:SizeInt;value:byte);assembler;asm        {A push is prefered over a local variable because a local         variable causes the compiler to generate a stackframe.}        cld{$ifdef REGCALL}        push    %edi        movl    %eax,%edi        movzbl  %cl,%eax        movl    %edx,%ecx{$else}        movl    x,%edi        movl    count,%ecx        movzbl  value,%eax        movl    %ecx,%edx{$endif}{ check for zero or negative count }        or      %ecx,%ecx        jle     .LFillEnd        cmpl    $7,%ecx        jl      .LFill1        imul    $0x01010101,%eax { Expand al into a 4 subbytes of eax}        shrl    $2,%ecx        andl    $3,%edx        rep        stosl        movl    %edx,%ecx.LFill1:        rep        stosb.LFillEnd:{$ifdef REGCALL}        pop %edi{$endif}end;{$endif FPC_SYSTEM_HAS_FILLCHAR}{$ifndef FPC_SYSTEM_HAS_FILLWORD}{$define FPC_SYSTEM_HAS_FILLWORD}procedure fillword(var x;count : SizeInt;value : word);assembler;var  saveedi : longint;asm        movl    %edi,saveedi{$ifdef REGCALL}        movl    %eax,%edi        movzwl  %cx,%eax        movl    %edx,%ecx{$else}        movl    x,%edi        movl    count,%ecx        movzwl  value,%eax{$endif}{ check for zero or negative count }        cmpl    $0,%ecx        jle     .LFillWordEnd        movl    %eax,%edx        shll    $16,%eax        orl     %edx,%eax        movl    %ecx,%edx        shrl    $1,%ecx        cld        rep        stosl        movl    %edx,%ecx        andl    $1,%ecx        rep        stosw.LFillWordEnd:        movl    saveedi,%ediend;{$endif FPC_SYSTEM_HAS_FILLWORD}{$ifndef FPC_SYSTEM_HAS_FILLDWORD}{$define FPC_SYSTEM_HAS_FILLDWORD}procedure filldword(var x;count : SizeInt;value : dword);assembler;var  saveedi : longint;asm        movl    %edi,saveedi{$ifdef REGCALL}        movl    %eax,%edi        movl    %ecx,%eax        movl    %edx,%ecx{$else}        movl    x,%edi        movl    count,%ecx        movl    value,%eax{$endif}{ check for zero or negative count }        cmpl    $0,%ecx        jle     .LFillDWordEnd        cld        rep        stosl.LFillDWordEnd:        movl    saveedi,%ediend;{$endif FPC_SYSTEM_HAS_FILLDWORD}{$ifndef FPC_SYSTEM_HAS_INDEXBYTE}{$define FPC_SYSTEM_HAS_INDEXBYTE}function IndexByte(Const buf;len:SizeInt;b:byte):SizeInt; assembler;var  saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %ebx,saveebx        movl    buf,%edi       // Load String        movb    b,%bl        movl    len,%ecx       // Load len        xorl    %eax,%eax        testl   %ecx,%ecx        jz      .Lcharposnotfound        cld        movl    %ecx,%edx      // Copy for easy manipulation        movb    %bl,%al        repne        scasb        jne     .Lcharposnotfound        incl    %ecx        subl    %ecx,%edx        movl    %edx,%eax        jmp     .Lready.Lcharposnotfound:        movl    $-1,%eax.Lready:        movl    saveedi,%edi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_FILLDWORD}{$ifndef FPC_SYSTEM_HAS_INDEXWORD}{$define FPC_SYSTEM_HAS_INDEXWORD}function Indexword(Const buf;len:SizeInt;b:word):SizeInt; assembler;var  saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %ebx,saveebx        movl    Buf,%edi       // Load String        movw    b,%bx        movl    Len,%ecx       // Load len        xorl    %eax,%eax        testl   %ecx,%ecx        jz      .Lcharposnotfound        cld        movl    %ecx,%edx      // Copy for easy manipulation        movw    %bx,%ax        repne        scasw        jne     .Lcharposnotfound        incl    %ecx        subl    %ecx,%edx        movl    %edx,%eax        jmp     .Lready.Lcharposnotfound:        movl    $-1,%eax.Lready:        movl    saveedi,%edi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_INDEXWORD}{$ifndef FPC_SYSTEM_HAS_INDEXDWORD}{$define FPC_SYSTEM_HAS_INDEXDWORD}function IndexDWord(Const buf;len:SizeInt;b:DWord):SizeInt; assembler;var  saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %ebx,saveebx{$ifdef REGCALL}        movl    %eax,%edi        movl    %ecx,%ebx        movl    %edx,%ecx{$else}        movl    Len,%ecx       // Load len        movl    Buf,%edi       // Load String        movl    b,%ebx{$endif}        xorl    %eax,%eax        testl   %ecx,%ecx        jz      .Lcharposnotfound        cld        movl    %ecx,%edx      // Copy for easy manipulation        movl    %ebx,%eax        repne        scasl        jne     .Lcharposnotfound        incl    %ecx        subl    %ecx,%edx        movl    %edx,%eax        jmp     .Lready.Lcharposnotfound:        movl    $-1,%eax.Lready:        movl    saveedi,%edi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_INDEXDWORD}{$ifndef FPC_SYSTEM_HAS_COMPAREBYTE}{$define FPC_SYSTEM_HAS_COMPAREBYTE}function CompareByte(Const buf1,buf2;len:SizeInt):SizeInt; assembler;var  saveesi,saveedi : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi        cld{$ifdef REGCALL}        movl    %eax,%edi        movl    %edx,%esi        movl    %ecx,%eax{$else}        movl    len,%eax        movl    buf2,%esi       { Load params}        movl    buf1,%edi{$endif}        testl   %eax,%eax       {We address -1(%esi), so we have to deal with len=0}        je      .LCmpbyteExit        cmpl    $7,%eax         {<7 not worth aligning and go through all trouble}        jl      .LCmpbyte2        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx            { calc bytes to align   (%edi and 3) xor 3= -%edi and 3}        andl    $3,%ecx        subl    %ecx,%eax       { Subtract from number of bytes to go}        orl     %ecx,%ecx        rep        cmpsb                   {The actual 32-bit Aligning}        jne     .LCmpbyte3        movl    %eax,%ecx       {bytes to do, divide by 4}        andl    $3,%eax         {remainder}        shrl    $2,%ecx         {The actual division}        orl     %ecx,%ecx       {Sets zero flag if ecx=0 -> no cmp}        rep        cmpsl        je      .LCmpbyte2       { All equal? then to the left over bytes}        movl    $4,%eax         { Not equal. Rescan the last 4 bytes bytewise}        subl    %eax,%esi        subl    %eax,%edi.LCmpbyte2:        movl    %eax,%ecx       {bytes still to (re)scan}        orl     %eax,%eax       {prevent disaster in case %eax=0}        rep        cmpsb.LCmpbyte3:        movzbl  -1(%esi),%ecx        movzbl  -1(%edi),%eax      // Compare failing (or equal) position        subl    %ecx,%eax.LCmpbyteExit:        movl    saveedi,%edi        movl    saveesi,%esiend;{$endif FPC_SYSTEM_HAS_COMPAREBYTE}{$ifndef FPC_SYSTEM_HAS_COMPAREWORD}{$define FPC_SYSTEM_HAS_COMPAREWORD}function CompareWord(Const buf1,buf2;len:SizeInt):SizeInt; assembler;var  saveesi,saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi        movl    %ebx,saveebx        cld{$ifdef REGCALL}        movl    %eax,%edi        movl    %edx,%esi        movl    %ecx,%eax{$else}        movl    len,%eax        movl    buf2,%esi       { Load params}        movl    buf1,%edi{$endif}        testl   %eax,%eax       {We address -2(%esi), so we have to deal with len=0}        je      .LCmpwordExit        cmpl    $5,%eax         {<5 (3 bytes align + 4 bytes cmpsl = 4 words}        jl      .LCmpword2      { not worth aligning and go through all trouble}        movl    (%edi),%ebx     // Compare alignment bytes.        cmpl    (%esi),%ebx        jne     .LCmpword2      // Aligning will go wrong already. Max 2 words will be scanned Branch NOW        shll    $1,%eax         {Convert word count to bytes}        movl    %edi,%edx       { Align comparing is already done, so simply add}        negl    %edx            { calc bytes to align  -%edi and 3}        andl    $3,%edx        addl    %edx,%esi       { Skip max 3 bytes alignment}        addl    %edx,%edi        subl    %edx,%eax       { Subtract from number of bytes to go}        movl    %eax,%ecx       { Make copy of bytes to go}        andl    $3,%eax         { Calc remainder (mod 4) }        andl    $1,%edx         { %edx is 1 if array not 2-aligned, 0 otherwise}        shrl    $2,%ecx         { divide bytes to go by 4, DWords to go}        orl     %ecx,%ecx       { Sets zero flag if ecx=0 -> no cmp}        rep                     { Compare entire DWords}        cmpsl        je      .LCmpword2a     { All equal? then to the left over bytes}        movl    $4,%eax         { Not equal. Rescan the last 4 bytes bytewise}        subl    %eax,%esi       { Go back one DWord}        subl    %eax,%edi        incl    %eax            {if not odd then this does nothing, else it makes                                  sure that adding %edx increases from 2 to 3 words}.LCmpword2a:        subl    %edx,%esi       { Subtract alignment}        subl    %edx,%edi        addl    %edx,%eax        shrl    $1,%eax.LCmpword2:        movl    %eax,%ecx       {words still to (re)scan}        orl     %eax,%eax       {prevent disaster in case %eax=0}        rep        cmpsw.LCmpword3:        movzwl  -2(%esi),%ecx        movzwl  -2(%edi),%eax    // Compare failing (or equal) position        subl    %ecx,%eax        // calculate end result..LCmpwordExit:        movl    saveedi,%edi        movl    saveesi,%esi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_COMPAREWORD}{$ifndef FPC_SYSTEM_HAS_COMPAREDWORD}{$define FPC_SYSTEM_HAS_COMPAREDWORD}function CompareDWord(Const buf1,buf2;len:SizeInt):SizeInt; assembler;var  saveesi,saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi        cld{$ifdef REGCALL}        movl    %eax,%edi        movl    %edx,%esi{$else}        movl    len,%ecx        movl    buf2,%esi       { Load params}        movl    buf1,%edi{$endif}        testl   %ecx,%ecx         je      .LCmpDwordExit        xorl    %eax,%eax        rep                     { Compare entire DWords}        cmpsl        movl    -4(%edi),%edi        // Compare failing (or equal) position        subl    -4(%esi),%edi        // calculate end result.        setb    %dl        seta    %cl        addb    %cl,%al        subb    %dl,%al        movsbl  %al,%eax.LCmpDwordExit:        movl    saveedi,%edi        movl    saveesi,%esiend;{$endif FPC_SYSTEM_HAS_COMPAREDWORD}{$ifndef FPC_SYSTEM_HAS_INDEXCHAR0}{$define FPC_SYSTEM_HAS_INDEXCHAR0}function IndexChar0(Const buf;len:SizeInt;b:Char):SizeInt; assembler;var  saveesi,saveebx : longint;asm        movl    %esi,saveesi        movl    %ebx,saveebx// Can't use scasb, or will have to do it twice, think this//   is faster for small "len"{$ifdef REGCALL}        movl    %eax,%esi        // Load address        movzbl  %cl,%ebx          // Load searchpattern{$else}        movl    Buf,%esi        // Load address        movl    len,%edx        // load maximal searchdistance        movzbl  b,%ebx          // Load searchpattern{$endif}        testl   %edx,%edx        je      .LFound        xorl    %ecx,%ecx       // zero index in Buf        xorl    %eax,%eax       // To make DWord compares possible.LLoop:        movb    (%esi),%al      // Load byte        cmpb    %al,%bl        je      .LFound         //  byte the same?        incl    %ecx        incl    %esi        cmpl    %edx,%ecx       // Maximal distance reached?        je      .LNotFound        testl   %eax,%eax       // Nullchar = end of search?        jne     .LLoop.LNotFound:        movl    $-1,%ecx        // Not found return -1.LFound:        movl    %ecx,%eax        movl    saveesi,%esi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_INDEXCHAR0}{****************************************************************************                                 String****************************************************************************}{$ifndef FPC_SYSTEM_HAS_FPC_SHORTSTR_ASSIGN}{$define FPC_SYSTEM_HAS_FPC_SHORTSTR_ASSIGN}function fpc_shortstr_to_shortstr(len:longint; const sstr: shortstring): shortstring; [public,alias: 'FPC_SHORTSTR_TO_SHORTSTR']; compilerproc;begin  asm        cld        movl    __RESULT,%edi        movl    sstr,%esi        xorl    %eax,%eax        movl    len,%ecx        lodsb        cmpl    %ecx,%eax        jbe     .LStrCopy1        movl    %ecx,%eax.LStrCopy1:        stosb        cmpl    $7,%eax        jl      .LStrCopy2        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        rep        movsb        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        rep        movsl.LStrCopy2:        movl    %eax,%ecx        rep        movsb  end ['ESI','EDI','EAX','ECX'];end;procedure fpc_shortstr_assign(len:longint;sstr,dstr:pointer);[public,alias:'FPC_SHORTSTR_ASSIGN'];begin  asm        pushl   %eax        pushl   %ecx        cld        movl    dstr,%edi        movl    sstr,%esi        xorl    %eax,%eax        movl    len,%ecx        lodsb        cmpl    %ecx,%eax        jbe     .LStrCopy1        movl    %ecx,%eax.LStrCopy1:        stosb        cmpl    $7,%eax        jl      .LStrCopy2        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        rep        movsb        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        rep        movsl.LStrCopy2:        movl    %eax,%ecx        rep        movsb        popl    %ecx        popl    %eax  end ['ESI','EDI'];end;{$endif FPC_SYSTEM_HAS_FPC_SHORTSTR_ASSIGN}{$ifndef FPC_SYSTEM_HAS_FPC_SHORTSTR_CONCAT}{$define FPC_SYSTEM_HAS_FPC_SHORTSTR_CONCAT}function fpc_shortstr_concat(const s1,s2:shortstring):shortstring;compilerproc;begin  asm        movl    __RESULT,%edi        movl    %edi,%ebx        movl    s1,%esi         { first string }        lodsb        andl    $0x0ff,%eax        stosb        cmpl    $7,%eax        jl      .LStrConcat1        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        rep        movsb        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        rep        movsl.LStrConcat1:        movl    %eax,%ecx        rep        movsb        movl    s2,%esi       { second string }        movzbl  (%ebx),%ecx        negl    %ecx        addl    $0x0ff,%ecx        lodsb        cmpl    %ecx,%eax        jbe     .LStrConcat2        movl    %ecx,%eax.LStrConcat2:        addb    %al,(%ebx)        cmpl    $7,%eax        jl      .LStrConcat3        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        rep        movsb        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        rep        movsl.LStrConcat3:        movl    %eax,%ecx        rep        movsb  end ['EBX','ECX','EAX','ESI','EDI'];end;{$endif FPC_SYSTEM_HAS_FPC_SHORTSTR_CONCAT}{$ifndef FPC_SYSTEM_HAS_FPC_SHORTSTR_APPEND_SHORTSTR}{$define FPC_SYSTEM_HAS_FPC_SHORTSTR_APPEND_SHORTSTR}procedure fpc_shortstr_append_shortstr(var s1:shortstring;const s2:shortstring);compilerproc;    [public,alias:'FPC_SHORTSTR_APPEND_SHORTSTR'];begin  asm        movl    s1,%edi        movl    s2,%esi        movl    %edi,%ebx        movzbl  (%edi),%ecx        movl    __HIGH(s1),%eax        lea     1(%edi,%ecx),%edi        negl    %ecx        addl    %eax,%ecx        // no need to zero eax, high(s1) <= 255        lodsb        cmpl    %ecx,%eax        jbe     .LStrConcat1        movl    %ecx,%eax.LStrConcat1:        addb    %al,(%ebx)        cmpl    $7,%eax        jl      .LStrConcat2        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        rep        movsb        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        rep        movsl.LStrConcat2:        movl    %eax,%ecx        rep        movsb  end ['EBX','ECX','EAX','ESI','EDI'];end;{$endif FPC_SYSTEM_HAS_FPC_SHORTSTR_APPEND_SHORTSTR}{$ifndef FPC_SYSTEM_HAS_FPC_SHORTSTR_COMPARE}{$define FPC_SYSTEM_HAS_FPC_SHORTSTR_COMPARE}function fpc_shortstr_compare(const left,right:shortstring): longint;assembler; [public,alias:'FPC_SHORTSTR_COMPARE']; compilerproc;var  saveesi,saveedi,saveebx : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi        movl    %ebx,saveebx        cld        movl    right,%esi        movl    left,%edi        movzbl  (%esi),%eax        movzbl  (%edi),%ebx        movl    %eax,%edx        incl    %esi        incl    %edi        cmpl    %ebx,%eax        jbe     .LStrCmp1        movl    %ebx,%eax.LStrCmp1:        cmpl    $7,%eax        jl      .LStrCmp2        movl    %edi,%ecx       { Align on 32bits }        negl    %ecx        andl    $3,%ecx        subl    %ecx,%eax        orl     %ecx,%ecx        rep        cmpsb        jne     .LStrCmp3        movl    %eax,%ecx        andl    $3,%eax        shrl    $2,%ecx        orl     %ecx,%ecx        rep        cmpsl        je      .LStrCmp2        movl    $4,%eax        subl    %eax,%esi        subl    %eax,%edi.LStrCmp2:        movl    %eax,%ecx        orl     %eax,%eax        rep        cmpsb        je      .LStrCmp4.LStrCmp3:        movzbl  -1(%esi),%edx      // Compare failing (or equal) position        movzbl  -1(%edi),%ebx.LStrCmp4:        movl    %ebx,%eax          // Compare length or position        subl    %edx,%eax        movl    saveedi,%edi        movl    saveesi,%esi        movl    saveebx,%ebxend;{$endif FPC_SYSTEM_HAS_FPC_SHORTSTR_COMPARE}{$ifndef FPC_SYSTEM_HAS_FPC_PCHAR_TO_SHORTSTR}{$define FPC_SYSTEM_HAS_FPC_PCHAR_TO_SHORTSTR}function fpc_pchar_to_shortstr(p:pchar):shortstring;assembler;[public,alias:'FPC_PCHAR_TO_SHORTSTR']; compilerproc;{$include strpas.inc}{$endif FPC_SYSTEM_HAS_FPC_PCHAR_TO_SHORTSTR}{$ifndef FPC_SYSTEM_HAS_FPC_PCHAR_LENGTH}{$define FPC_SYSTEM_HAS_FPC_PCHAR_LENGTH}function fpc_pchar_length(p:pchar):longint;assembler;[public,alias:'FPC_PCHAR_LENGTH']; compilerproc;{$include strlen.inc}{$endif FPC_SYSTEM_HAS_FPC_PCHAR_LENGTH}{$define FPC_SYSTEM_HAS_GET_FRAME}function get_frame:pointer;assembler;nostackframe;{$ifdef SYSTEMINLINE}inline;{$endif}asm        movl    %ebp,%eaxend ['EAX'];{$define FPC_SYSTEM_HAS_GET_CALLER_ADDR}function get_caller_addr(framebp:pointer):pointer;nostackframe;assembler;{$ifdef SYSTEMINLINE}inline;{$endif}asm{$ifndef REGCALL}        movl    framebp,%eax{$endif}        orl     %eax,%eax        jz      .Lg_a_null        movl    4(%eax),%eax.Lg_a_null:end ['EAX'];{$define FPC_SYSTEM_HAS_GET_CALLER_FRAME}function get_caller_frame(framebp:pointer):pointer;nostackframe;assembler;{$ifdef SYSTEMINLINE}inline;{$endif}asm{$ifndef REGCALL}        movl    framebp,%eax{$endif}        orl     %eax,%eax        jz      .Lgnf_null        movl    (%eax),%eax.Lgnf_null:end ['EAX'];{****************************************************************************                                 Math****************************************************************************}{$define FPC_SYSTEM_HAS_ABS_LONGINT}function abs(l:longint):longint; assembler;nostackframe;{$ifdef SYSTEMINLINE}inline;{$endif}asm{$ifndef REGCALL}        movl    l,%eax{$endif}        cltd        xorl    %edx,%eax        subl    %edx,%eaxend ['EAX','EDX'];{$define FPC_SYSTEM_HAS_ODD_LONGINT}function odd(l:longint):boolean;assembler;nostackframe;{$ifdef SYSTEMINLINE}inline;{$endif}asm{$ifdef SYSTEMINLINE}       movl     l,%eax{$else}{$ifndef REGCALL}       movl     l,%eax{$endif}{$endif}       andl     $1,%eax       setnz    %alend ['EAX'];{$define FPC_SYSTEM_HAS_SQR_LONGINT}function sqr(l:longint):longint;assembler;nostackframe;{$ifdef SYSTEMINLINE}inline;{$endif}asm{$ifdef SYSTEMINLINE}       movl     l,%eax{$else}{$ifndef REGCALL}       movl     l,%eax{$endif}{$endif}        imull   %eax,%eaxend ['EAX'];{$define FPC_SYSTEM_HAS_SPTR}Function Sptr : Pointer;assembler;nostackframe;{$ifdef SYSTEMINLINE}inline;{$endif}asm        movl    %esp,%eaxend;{****************************************************************************                               Bounds Check****************************************************************************}{ do a thread-safe inc/dec }{$define FPC_SYSTEM_HAS_DECLOCKED_LONGINT}function cpudeclocked(var l : longint) : boolean;assembler;nostackframe;  asm{$ifndef REGCALL}     movl       l,%eax{$endif}     { this check should be done because a lock takes a lot }     { of time!                                             }     lock     decl       (%eax)     setzb      %al  end;{$define FPC_SYSTEM_HAS_INCLOCKED_LONGINT}procedure cpuinclocked(var l : longint);assembler;nostackframe;  asm{$ifndef REGCALL}     movl       l,%eax{$endif}     lock     incl       (%eax)  end;// inline SMP check and normal lock.// the locked one is so slow, inlining doesn't matter.function declocked(var l : longint) : boolean; inline;begin  if not ismultithread then    begin     dec(l);     declocked:=l=0;    end   else    declocked:=cpudeclocked(l);end;procedure inclocked(var l : longint); inline;begin  if not ismultithread then    inc(l)   else    cpuinclocked(l);end;{****************************************************************************                                  FPU****************************************************************************}const  fpucw : word = $1332;  { Internal constants for use in system unit }  FPU_Invalid = 1;  FPU_Denormal = 2;  FPU_DivisionByZero = 4;  FPU_Overflow = 8;  FPU_Underflow = $10;  FPU_StackUnderflow = $20;  FPU_StackOverflow = $40;  FPU_ExceptionMask = $ff;{$define FPC_SYSTEM_HAS_SYSRESETFPU}Procedure SysResetFPU;assembler;{$ifdef SYSTEMINLINE}inline;{$endif}asm    fninit    fldcw   fpucw    fwaitend;{$define FPC_SYSTEM_HAS_ANSISTR_DECR_REF}function fpc_freemem_x(p:pointer):ptrint; [external name 'FPC_FREEMEM_X'];Procedure fpc_AnsiStr_Decr_Ref (Var S : Pointer); [Public,Alias:'FPC_ANSISTR_DECR_REF']; compilerproc; nostackframe; assembler;asm        cmpl $0,(%eax)        jne .Ldecr_ref_continue        ret.Ldecr_ref_continue:// Temps allocated between ebp-24 and ebp+0        subl    $4,%esp// Var S located in register// Var l located in register        movl    %eax,(%esp)// [101] l:=@PAnsiRec(S-FirstOff)^.Ref;        movl    (%eax),%edx        subl    $8,%edx// [102] If l^<0 then exit;        cmpl    $0,(%edx)        jl      .Lj3596.Lj3603:// [104] If declocked(l^) then        cmpb    $0,ismultithread        jne     .Lj3610        decl    (%edx)        je      .Lj3620        addl    $4,%esp        ret.Lj3610:        movl    %edx,%eax        call    cpudeclocked        testb   %al,%al        je      .Lj3605.Lj3620:        movl    (%esp),%eax        movl    (%eax),%eax        subl    $8,%eax        call    FPC_FREEMEM_X        movl    (%esp),%eax        movl    $0,(%eax).Lj3618:.Lj3605:.Lj3596:// [107] end;        addl $4,%espend;function fpc_truely_ansistr_unique(Var S : Pointer): Pointer; forward;{$define FPC_SYSTEM_HAS_ANSISTR_UNIQUE}Function fpc_ansistr_Unique(Var S : Pointer): Pointer; [Public,Alias : 'FPC_ANSISTR_UNIQUE']; compilerproc; nostackframe;assembler;asm// Var S located in register// Var $result located in register        movl    %eax,%edx// [437] pointer(result) := pointer(s);        movl    (%eax),%eax// [438] If Pointer(S)=Nil then        testl   %eax,%eax        je      .Lj4031.Lj4036:// [440] if PAnsiRec(Pointer(S)-Firstoff)^.Ref<>1 then        movl    -8(%eax),%ecx        cmpl    $1,%ecx        je      .Lj4038// [441] result:=fpc_truely_ansistr_unique(s);        movl    %edx,%eax        call    fpc_truely_ansistr_unique.Lj4038:.Lj4031:// [442] end;end;
 |