| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495 | {    $Id$    This file is part of the Free Pascal run time library.    Copyright (c) 2000 by Jonas Maebe, member of the    Free Pascal development team    Processor dependent part of strings.pp, that can be shared with    sysutils unit.    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. **********************************************************************}{ Note: the implementation of these routines is for BIG ENDIAN only!! (JM) }function strcopy(dest,source : pchar) : pchar;assembler;{ in: dest in r3, source in r4 }{ out: result (dest) in r3     }asm{  in: dest in r3, source in r4  }{  out: result (dest) in r3      }        {  load the begin of the source string in the data cache }        dcbt    0,r4        {  get # of misaligned bytes  }        rlwinm. r10,r4,0,31-2+1,31        subfic  r10,r10,4        mtctr   r10        {  since we have to return dest intact, use another register for  }        {  dest in the copy loop                                          }        subi    r9,r3,1        subi    r4,r4,1        beq     LStrCopyAlignedLStrCopyAlignLoop:        {  load next byte  }        lbzu    r0,1(r4)        {  end of string?  }        cmpli   cr0,r0,0        {  store byte  }        stbu    r0,1(r9)        {  loop if misaligned bytes left and not end of string found }        bdnzf   eq,LStrCopyAlignLoop        beq     LStrCopyDoneLStrCopyAligned:        subi    r4,r4,3        subi    r9,r9,3        { setup magic constants }        li      r8,0x0feff        addis   r8,r8,0x0feff        li      r7,0x08080        addis    r7,r7,0x08081        { load first 4 bytes  }        lwzu    r0,4(r4)LStrCopyAlignedLoop:        { test for zero byte }        add     r10,r0,r8        andc    r10,r10,r0        and.    r10,r10,r7        bne     LStrCopyEndFound        stwu    r0,4(r9)        { load next 4 bytes (do it here so the load can begin while the }        { the branch is processed)                                      }        lwzu    r0,4(r4)        b       LStrCopyAlignedLoopLStrCopyEndFound:        { result is either 0, 8, 16 or 24 depending on which byte is zero }        cntlzw  r10,r10        addi    r9,r9,3LStrCopyWrapUpLoop:        subi    r10,r10,8        rlwinm  r0,r0,8,0,31        stbu    r0,1(r9)        bge     LStrCopyWrapUpLoopLStrCopyDone:        {  r3 still contains dest here  }end ['r4','r7','r8','r0','r9','r10','cr0','ctr'];function strecopy(dest,source : pchar) : pchar;assembler;{ in: dest in r3, source in r4        }{ out: result (end of new dest) in r3 }asm        {  load the begin of the source string in the data cache }        dcbt    0,r4        {  get # of misaligned bytes  }        rlwinm. r10,r4,0,31-2+1,31        subfic  r10,r10,4        mtctr   r10        subi    r3,r3,1        subi    r4,r4,1        beq     LStrCopyAlignedLStrCopyAlignLoop:        {  load next byte  }        lbzu    r0,1(r4)        {  end of string?  }        cmpli   cr0,r0,0        {  store byte  }        stbu    r0,1(r3)        {  loop if misaligned bytes left and not end of string found }        bdnzf   eq,LStrCopyAlignLoop        beq     LStrCopyDoneLStrCopyAligned:        subi    r4,r4,3        subi    r3,r3,3        { setup magic constants }        li      r8,0x0feff        addis   r8,r8,0x0feff        li      r9,0x08080        addis    r9,r9,0x08081LStrCopyAlignedLoop:        {  load next 4 bytes  }        lwzu    r0,4(r4)        { test for zero byte }        add     r10,r0,r8        andc    r10,r10,r0        and.    r10,r10,r9        bne     LStrCopyEndFound        stwu    r0,4(r3)        b       LStrCopyAlignedLoopLStrCopyEndFound:        { result is either 0, 8, 16 or 24 depending on which byte is zero }        cntlzw  r10,r10        addi    r3,r3,3LStrCopyWrapUpLoop:        subic.  r10,r10,8        rlwinm  r0,r0,8,0,31        stbu    r0,1(r3)        bge     LStrCopyWrapUpLoopLStrCopyDone:        {  r3 contains new dest here  }end ['r3','r4','r8','r0','r3','r10','cr0','ctr'];function strlcopy(dest,source : pchar;maxlen : longint) : pchar;assembler;{ in: dest in r3, source in r4, maxlen in r5 }{ out: result (dest) in r3                   }asm        {  load the begin of the source string in the data cache }        dcbt    0,r4        mtctr   r5        subi    r4,r4,1        subi    r0,r3,1LStrlCopyLoop:        lbzu    r10,1(r4)        cmpli   r10,0        stbu    r10,1(r0)        bdnzf   cr0*4+eq, LStrlCopyLoop        { if we stopped because we copied a #0, we're done }        beq     LStrlCopyDone        { otherwise add the #0 }        li      r10,0        stb     r10,1(r0)LStrlCopyDone:end ['r0','r4','r30','cr0'];function strlen(p : pchar) : longint;assembler;{$i strlen.inc}function strend(p : pchar) : pchar;assembler;{ in: p in r3                  }{ out: result (end of p) in r3 }asm        {  load the begin of the string in the data cache }        dcbt    0,r3        { empty/invalid string? }        cmpli   r3,0        { if yes, do nothing }        beq     LStrEndDone        subi    r3,r3,1LStrEndLoop:        lbzu    r0,1(r3)        cmpli   r0,0        bne     LStrEndLoopLStrEndDone:end ['r0','r3','r4','cr0'];function strcomp(str1,str2 : pchar) : longint;assembler;{ in: str1 in r3, str2 in r4                                                }{ out: result (= 0 if strings equal, < 0 if str1 < str2, > 0 if str1 > str2 }{      in r3                                                                }asm        { use r0 instead of r3 for str1 since r3 contains result }        subi    r0,r3,1        subi    r4,r4,1LStrCompLoop:        { load next chars }        lbzu    r9,1(r0)        { check if one is zero }        cmpli   cr1,r9,0        lbzu    r10,1(r4)        { calculate difference }        sub.    r3,r9,r10        { if chars not equal, we're ready }        bne     LStrCompDone        { if they are equal and one is zero, then the other one is zero too }        { and we're done as well (r3 also contains 0 then)                  }        { otherwise loop                                                    }        bne     cr1,LStrCompLoopLStrCompDone:end ['r0','r3','r4','r9','r10','cr0','cr1'];function strlcomp(str1,str2 : pchar;l : longint) : longint;assembler;{ (same as strcomp, but maximally compare until l'th character)             }{ in: str1 in r3, str2 in r4, l in r5                                       }{ out: result (= 0 if strings equal, < 0 if str1 < str2, > 0 if str1 > str2 }{      in r3                                                                }asm        { load the begin of one of the strings in the data cache }        dcbt    0,r3        { use r0 instead of r3 for str1 since r3 contains result }        cmpl    r5,0        subi    r0,r3,1        li      r3,0        beq     LStrlCompDone        mtctr   r5        subi    r4,r4,1LStrlCompLoop:        { load next chars }        lbzu    r9,1(r0)        { check if one is zero }        cmpli   cr1,r9,0        lbzu    r10,1(r4)        { calculate difference }        sub.    r3,r9,r10        { if chars not equal, we're ready }        bne     LStrlCompDone        { if they are equal and one is zero, then the other one is zero too }        { and we're done as well (r3 also contains 0 then)                  }        { otherwise loop (if ctr <> 0)                                      }        bdnzf  cr1*4+eq,LStrlCompLoopLStrlCompDone:end ['r0','r3','r4','r9','r10','cr0','cr1','ctr'];function stricomp(str1,str2 : pchar) : longint;assembler;{ in: str1 in r3, str2 in r4                                 }{ out: result of case insensitive comparison (< 0, = 0, > 0) }asm        { use r28 instead of r3 for str1 since r3 contains result }        subi    r28,r3,1        subi    r4,r4,1LStriCompLoop:        { load next chars }        lbzu    r29,1(r28)        { check if one is zero }        cmpli   cr1,r29,0        lbzu    r30,1(r4)        { calculate difference }        sub.    r3,r29,r30        { if chars are equal, no further test is necessary }        beq+    LStriCompEqual        { make both lowercase, no branches }        li       r27,0        li       r25,0        { r3 := r29 - 'A' }        subic    r3,r29,'A'        { if r29 < 'A' then r27 := 0 else r27 := $ffffffff }        addme    r27,r27        { same for r30 }        subic    r3,r30,'A'        addme    r25,r25        { r3 := 'Z' - r29 }        subfic   r3,r29,'Z'        { if r29 < 'A' then r27 := 0 else r27 := $20 }        andi     r27,r27,0x020        { if r29 > Z then r26 := 0 else r26 := $ffffffff }        subfe    r26,r26,r26        { same for r30 }        subfic   r3,r30,'Z'        andi     r25,r25,0x020        subfe    r24,r24,r24        { if (r29 in ['A'..'Z'] then r27 := $20 else r27 := 0 }        and      r27,r27,r26        { same for r30 }        and      r25,r25,r24        { make lowercase }        add      r29,r29,r27        { same for r30 }        add      r30,r30,r25        { compare again }        sub.     r3,r29,r30        bne      LStrCompDoneLStriCompEqual:        { if they are equal and one is zero, then the other one is zero too }        { and we're done as well (r3 also contains 0 then)                  }        { otherwise loop                                                    }        bne     cr1,LStriCompLoopLStriCompDone:end ['r3','r4','r26','r27','r28','r29','r30','cr0','cr1'];function strlicomp(str1,str2 : pchar;l : longint) : longint;assembler;{ (same as stricomp, but maximally compare until l'th character) }{ in: str1 in r3, str2 in r4, l in r5                            }{ out: result of case insensitive comparison (< 0, = 0, > 0)     }asm        {  load the begin of one of the string in the data cache }        dcbt    0,r3        { use r0 instead of r3 for str1 since r3 contains result }        cmpl    r5,0        subi    r0,r3,1        li      r3,0        beq-    LStrlCompDone        mtctr   r5        subi    r4,r4,1LStriCompLoop:        { load next chars }        lbzu    r9,1(r0)        { check if one is zero }        cmpli   cr1,r9,0        lbzu    r10,1(r4)        { calculate difference }        sub.    r3,r9,r10        { if chars are equal, no further test is necessary }        beq+    LStriCompEqual        { see stricomp for explanation }        li       r8,0        li       r5,0        subic    r3,r9,'A'        addme    r8,r8        subic    r3,r10,'A'        addme    r5,r5        subfic   r3,r9,'Z'        andi     r8,r8,0x020        subfe    r7,r7,r7        subfic   r3,r10,'Z'        andi     r5,r5,0x020        subfe    r24,r24,r24        and      r8,r8,r7        and      r5,r5,r24        add      r9,r9,r8        add      r10,r10,r5        { compare again }        sub.     r3,r9,r10        bne      LStrCompDoneLStriCompEqual:        { if they are equal and one is zero, then the other one is zero too }        { and we're done as well (r3 also contains 0 then)                  }        { otherwise loop (if ctr <> 0)                                      }        bdnzf    cr1*4+eq,LStriCompLoopLStriCompDone:end ['r0','r3','r4','r5','r7','r8','r9','r10','cr0','cr1','ctr'];function strscan(p : pchar;c : char) : pchar;assembler;asm        { empty/invalid string? }        cmpli   r3,0        { if yes, do nothing }        beq     LStrScanDone        subi    r3,r3,1LStrScanLoop:        lbzu    r0,1(r3)        cmpl    cr1,r0,r4        cmpli   r0,0        beq     cr1,LStrScanDone        bne     LStrScanLoopLStrScanDone:end ['r0','r3','r4','cr0','cr1'];function strrscan(p : pchar;c : char) : pchar;assembler;asm        { empty/invalid string? }        cmpli   r3,0        { if yes, do nothing }        beq     LStrrScanDone        { make r0 $ffffffff, later on we take min(r0,r3) }        li      r0,0x0ffff        subi    r3,r3,1LStrrScanLoop:        lbzu    r10,1(r3)        cmpl    cr1,r10,r4        cmpli   cr0,r10,0        bne+    cr1,LStrrScanNotFound        { store address of found position }        mr      r0,r3LStrrScanNotFound:        bne     LStrrScanLoop        { Select min of r3 and r0 -> end of string or found position    }        { From the PPC compiler writer's guide, not sure if I could ever }        { come up with something like this :)                            }        subfc   r10,r3,r0   { r10 = r0 - r3, CA = (r0 >= r3) ? 1 : 0 }        subfe   r0,r0,r0    { r0' = (r0 >= r3) ? 0 : -1              }        and     r10,r10,r0  { r10 = (r0 >= r3) ? 0 : r0 - r3         }        add     r3,r10,r3   { r3  = (r0 >= r3) ?  r3 : r0            }LStrrScanDone:end ['r0','r3','r4','r10','cr0','cr1'];function strupper(p : pchar) : pchar;assembler;asm        cmpli   r3,0        beq     LStrUpperNil        subi    r9,r3,1LStrUpperLoop:        lbzu    r10,1(r9)        { a <= x <= b <=> cardinal(x-a) <= cardinal(b-a) }        subi    r0,r10,97        cmpli   r0,122-97        cmpli   cr1,r10,0        subi    r10,r10,0x20        bgt     LStrUpper1        stb     r10,0(r9)LStrUpper1:        bne     cr1,LStrUpperLoopLStrUpperNil:end ['r0','r9','r10','cr0','cr1'];function strlower(p : pchar) : pchar;assembler;asm        cmpli   r3,0        beq     LStrLowerNil        subi    r9,r3,1LStrLowerLoop:        lbzu    r10,1(r9)        { a <= x <= b <=> cardinal(x-a) <= cardinal(b-a) }        subi    r0,r10,65        cmpli   r0,90-65        cmpli   cr1,r10,0        addi    r10,r10,0x20        bgt     LStrLower1        stb     r10,0(r9)LStrLower1:        bne     cr1,LStrLowerLoopLStrLowerNil:end ['r0','r9','r10','cr0','cr1'];{  $Log$  Revision 1.10  2001-09-28 13:25:04  jonas    * fixed wrong alignment code (sometimes we aligned to multiple of 8      instead of the desired multiple of 4)  Revision 1.9  2001/09/27 15:30:29  jonas    * conversion to compilerproc and to structure used by i386 rtl    * some bugfixes    * powerpc.inc is almost complete (only fillchar/word/dword, get_frame etc      and the class helpers are still needed    - removed unnecessary register saving in set.inc (thanks to compilerproc)    * use registers reserved for parameters as much as possible instead of      those reserved for local vars (since those have to be saved by the      called anyway, while the ones for local vars have to be saved by the      callee)  Revision 1.8  2001/07/21 15:51:50  jonas    * fixed small bug in stricomp  Revision 1.7  2001/07/07 12:46:12  jonas    * some small bugfixes and cache optimizations  Revision 1.6  2001/02/23 14:05:33  jonas    * optimized strcopy/strecopy  Revision 1.5  2001/02/11 17:59:14  jonas    * fixed bug in strscan  Revision 1.4  2001/02/11 12:15:03  jonas    * some small optimizations and bugfixes  Revision 1.3  2001/02/10 16:09:43  jonas   + implemented all missing routines and changed reg allocation to follow ABI  Revision 1.2  2001/02/10 12:28:22  jonas    * fixed some bugs, simplified/optimized already implemented routines and code some more  Revision 1.1  2000/11/05 17:17:08  jonas    + first implementation, not yet finished}
 |