mariadb/strings/strings-x86.s

406 lines
9 KiB
ArmAsm
Raw Normal View History

# Copyright (C) 2000 MySQL AB
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
# Optimized string functions Intel 80x86 (gcc/gas syntax)
.file "strings.s"
2000-07-31 21:29:14 +02:00
.version "1.00"
.text
# Move a alligned, not overlapped, by (long) divided memory area
# Args: to,from,length
2000-07-31 21:29:14 +02:00
.globl bmove_allign
.type bmove_allign,@function
bmove_allign:
movl %edi,%edx
movl %esi,%eax
movl 4(%esp),%edi # to
movl 8(%esp),%esi # from
movl 12(%esp),%ecx # length
addw $3,%cx # fix if not divisible with long
2000-07-31 21:29:14 +02:00
shrw $2,%cx
rep
movsl
movl %eax,%esi
movl %edx,%edi
ret
.end:
.size bmove_allign,.end-bmove_allign
# Move a string from higher to lower
# Arg from+1,to+1,length
2000-07-31 21:29:14 +02:00
.globl bmove_upp
.type bmove_upp,@function
bmove_upp:
std # Work downward
2000-07-31 21:29:14 +02:00
movl %edi,%edx
movl %esi,%eax
movl 4(%esp),%edi # p1
movl 8(%esp),%esi # p2
movl 12(%esp),%ecx # length
decl %edi # Don't move last arg
2000-07-31 21:29:14 +02:00
decl %esi
rep
movsb # One byte a time because overlap
cld # C library wants cld
2000-07-31 21:29:14 +02:00
movl %eax,%esi
movl %edx,%edi
ret
.bmove_upp_end:
.size bmove_upp,.bmove_upp_end-bmove_upp
# Append fillchars to string
# Args: dest,len,fill
2000-07-31 21:29:14 +02:00
.globl strappend
.type strappend,@function
strappend:
pushl %edi
movl 8(%esp),%edi # Memory pointer
movl 12(%esp),%ecx # Length
clrl %eax # Find end of string
2000-07-31 21:29:14 +02:00
repne
scasb
jnz sa_99 # String to long, shorten it
movzb 16(%esp),%eax # Fillchar
decl %edi # Point at end null
incl %ecx # rep made one dec for null-char
movb %al,%ah # (2) Set up a 32 bit pattern.
movw %ax,%dx # (2)
shll $16,%eax # (3)
movw %dx,%ax # (2) %eax has the 32 bit pattern.
movl %ecx,%edx # (2) Save the count of bytes.
shrl $2,%ecx # (2) Number of dwords.
2000-07-31 21:29:14 +02:00
rep
stosl # (5 + 5n)
movb $3,%cl # (2)
and %edx,%ecx # (2) Fill in the odd bytes
2000-07-31 21:29:14 +02:00
rep
stosb # Move last bytes if any
2000-07-31 21:29:14 +02:00
sa_99: movb $0,(%edi) # End of string
2000-07-31 21:29:14 +02:00
popl %edi
ret
.strappend_end:
.size strappend,.strappend_end-strappend
# Find if string contains any char in another string
# Arg: str,set
# Ret: Pointer to first found char in str
2000-07-31 21:29:14 +02:00
.globl strcont
.type strcont,@function
strcont:
movl %edi,%edx
pushl %esi
movl 8(%esp),%esi # str
movl 12(%esp),%ecx # set
clrb %ah # For endtest
2000-07-31 21:29:14 +02:00
jmp sc_60
sc_10: scasb
jz sc_fo # Found char
sc_20: cmp (%edi),%ah # Test if null
jnz sc_10 # Not end of set yet
incl %esi # Next char in str
sc_60: movl %ecx,%edi # %edi = Set
movb (%esi),%al # Test if this char exist
2000-07-31 21:29:14 +02:00
andb %al,%al
jnz sc_20 # Not end of string
clrl %esi # Return Null
sc_fo: movl %esi,%eax # Char found here
movl %edx,%edi # Restore
2000-07-31 21:29:14 +02:00
popl %esi
ret
.strcont_end:
.size strcont,.strcont_end-strcont
# Find end of string
# Arg: str
# ret: Pointer to end null
2000-07-31 21:29:14 +02:00
.globl strend
.type strend,@function
strend:
movl %edi,%edx # Save
movl 4(%esp),%edi # str
clrl %eax # Find end of string
2000-07-31 21:29:14 +02:00
movl %eax,%ecx
decl %ecx # ECX = -1
2000-07-31 21:29:14 +02:00
repne
scasb
movl %edi,%eax
decl %eax # End of string
movl %edx,%edi # Restore
2000-07-31 21:29:14 +02:00
ret
.strend_end:
.size strend,.strend_end-strend
# Make a string with len fill-chars and endnull
# Args: dest,len,fill
# Ret: dest+len
2000-07-31 21:29:14 +02:00
.globl strfill
.type strfill,@function
strfill:
pushl %edi
movl 8(%esp),%edi # Memory pointer
movl 12(%esp),%ecx # Length
movzb 16(%esp),%eax # Fill
2000-07-31 21:29:14 +02:00
movb %al,%ah # (2) Set up a 32 bit pattern
movw %ax,%dx # (2)
shll $16,%eax # (3)
movw %dx,%ax # (2) %eax has the 32 bit pattern.
2000-07-31 21:29:14 +02:00
movl %ecx,%edx # (2) Save the count of bytes.
shrl $2,%ecx # (2) Number of dwords.
2000-07-31 21:29:14 +02:00
rep
stosl # (5 + 5n)
movb $3,%cl # (2)
and %edx,%ecx # (2) Fill in the odd bytes
2000-07-31 21:29:14 +02:00
rep
stosb # Move last bytes if any
2000-07-31 21:29:14 +02:00
movb %cl,(%edi) # End NULL
movl %edi,%eax # End i %eax
2000-07-31 21:29:14 +02:00
popl %edi
ret
.strfill_end:
.size strfill,.strfill_end-strfill
# Find a char in or end of a string
# Arg: str,char
# Ret: pointer to found char or NullS
2000-07-31 21:29:14 +02:00
.globl strcend
.type strcend,@function
strcend:
movl %edi,%edx
movl 4(%esp),%edi # str
movb 8(%esp),%ah # search
clrb %al # for scasb to find end
2000-07-31 21:29:14 +02:00
se_10: cmpb (%edi),%ah
jz se_20 # Found char
2000-07-31 21:29:14 +02:00
scasb
jnz se_10 # Not end
dec %edi # Not found, point at end of string
2000-07-31 21:29:14 +02:00
se_20: movl %edi,%eax
movl %edx,%edi # Restore
2000-07-31 21:29:14 +02:00
ret
.strcend_end:
.size strcend,.strcend_end-strcend
# Test if string has a given suffix
2000-07-31 21:29:14 +02:00
.globl is_prefix
.type is_prefix,@function
is_prefix:
movl %edi,%edx # Save %edi
pushl %esi # and %esi
movl 12(%esp),%esi # get suffix
movl 8(%esp),%edi # s1
movl $1,%eax # Ok and zero-test
2000-07-31 21:29:14 +02:00
ip_10: cmpb (%esi),%ah
jz suf_ok # End of string/ found suffix
cmpsb # Compare strings
jz ip_10 # Same, possible prefix
xor %eax,%eax # Not suffix
2000-07-31 21:29:14 +02:00
suf_ok: popl %esi
movl %edx,%edi
ret
.is_prefix_end:
.size is_prefix,.is_prefix_end-is_prefix
# Find a substring in string
# Arg: str,search
2000-07-31 21:29:14 +02:00
.globl strstr
.type strstr,@function
strstr:
pushl %edi
pushl %esi
movl 12(%esp),%esi # str
movl 16(%esp),%edi # search
2000-07-31 21:29:14 +02:00
movl %edi,%ecx
incl %ecx # %ecx = search+1
movb (%edi),%ah # %ah = First char in search
2000-07-31 21:29:14 +02:00
jmp sf_10
sf_00: movl %edx,%esi # si = Current str-pos
sf_10: movb (%esi),%al # Test if this char exist
2000-07-31 21:29:14 +02:00
andb %al,%al
jz sf_90 # End of string, didn't find search
2000-07-31 21:29:14 +02:00
incl %esi
cmpb %al,%ah
jnz sf_10 # Didn't find first char, continue
movl %esi,%edx # Save str-pos in %edx
2000-07-31 21:29:14 +02:00
movl %ecx,%edi
sf_20: cmpb $0,(%edi)
jz sf_fo # Found substring
2000-07-31 21:29:14 +02:00
cmpsb
jz sf_20 # Char ok
jmp sf_00 # Next str-pos
2000-07-31 21:29:14 +02:00
sf_90: movl $1,%edx # Return Null
sf_fo: movl %edx,%eax # Char found here
decl %eax # Pointed one after
2000-07-31 21:29:14 +02:00
popl %esi
popl %edi
ret
.strstr_end:
.size strstr,.strstr_end-strstr
# Find a substring in string, return index
# Arg: str,search
2000-07-31 21:29:14 +02:00
.globl strinstr
.type strinstr,@function
strinstr:
pushl %ebp
movl %esp,%ebp
pushl 12(%ebp) # search
pushl 8(%ebp) # str
2000-07-31 21:29:14 +02:00
call strstr
add $8,%esp
or %eax,%eax
jz si_99 # Not found, return NULL
sub 8(%ebp),%eax # Pos from start
inc %eax # And first pos = 1
2000-07-31 21:29:14 +02:00
si_99: popl %ebp
ret
.strinstr_end:
.size strinstr,.strinstr_end-strinstr
# Make a string of len length from another string
# Arg: dst,src,length
# ret: end of dst
2000-07-31 21:29:14 +02:00
.globl strmake
.type strmake,@function
strmake:
pushl %edi
pushl %esi
movl 12(%esp),%edi # dst
movl 16(%esp),%esi # src
movl 20(%esp),%ecx # Length of memory-area
clrb %al # For test of end-null
jecxz sm_90 # Nothing to move, put zero at end.
sm_10: cmpb (%esi),%al # Next char to move
movsb # move arg
jz sm_99 # last char, we are ready
loop sm_10 # Continue moving
sm_90: movb %al,(%edi) # Set end pos
incl %edi # Fix that di points at end null
sm_99: decl %edi # di points now at end null
movl %edi,%eax # Ret value.p $
2000-07-31 21:29:14 +02:00
popl %esi
popl %edi
ret
.strmake_end:
.size strmake,.strmake_end-strmake
# Move a string with max len chars
# arg: dst,src,len
# ret: pos to first null or dst+len
2000-07-31 21:29:14 +02:00
.globl strnmov
.type strnmov,@function
strnmov:
pushl %edi
pushl %esi
movl 12(%esp),%edi # dst
movl 16(%esp),%esi # src
movl 20(%esp),%ecx # Length of memory-area
jecxz snm_99 # Nothing to do
clrb %al # For test of end-null
snm_10: cmpb (%esi),%al # Next char to move
movsb # move arg
jz snm_20 # last char, fill with null
loop snm_10 # Continue moving
incl %edi # Point two after last
snm_20: decl %edi # Point at first null (or last+1)
snm_99: movl %edi,%eax # Pointer at last char
2000-07-31 21:29:14 +02:00
popl %esi
popl %edi
ret
.strnmov_end:
.size strnmov,.strnmov_end-strnmov
.globl strmov
.type strmov,@function
strmov:
movl %esi,%ecx # Save old %esi and %edi
2000-07-31 21:29:14 +02:00
movl %edi,%edx
movl 8(%esp),%esi # get source pointer (s2)
movl 4(%esp),%edi # %edi -> s1
2000-07-31 21:29:14 +02:00
smo_10: movb (%esi),%al
movsb # move arg
2000-07-31 21:29:14 +02:00
andb %al,%al
jnz smo_10 # Not last
2000-07-31 21:29:14 +02:00
movl %edi,%eax
dec %eax
movl %ecx,%esi # Restore
2000-07-31 21:29:14 +02:00
movl %edx,%edi
ret
.strmov_end:
.size strmov,.strmov_end-strmov
.globl strxmov
.type strxmov,@function
strxmov:
movl %ebx,%edx # Save %ebx, %esi and %edi
2000-07-31 21:29:14 +02:00
mov %esi,%ecx
push %edi
leal 8(%esp),%ebx # Get destination
2000-07-31 21:29:14 +02:00
movl (%ebx),%edi
xorb %al,%al
jmp next_str # Handle source ebx+4
2000-07-31 21:29:14 +02:00
start_str:
movsb
cmpb -1(%edi),%al
jne start_str
decl %edi # Don't copy last null
2000-07-31 21:29:14 +02:00
next_str:
addl $4,%ebx
movl (%ebx),%esi
orl %esi,%esi
jne start_str
movb %al,0(%edi) # Force last to ASCII 0
2000-07-31 21:29:14 +02:00
movl %edi,%eax # Return ptr to ASCII 0
pop %edi # Restore registers
2000-07-31 21:29:14 +02:00
movl %ecx,%esi
movl %edx,%ebx
ret
.strxmov_end:
.size strxmov,strxmov_end-strxmov