Blame sysdeps/ia64/strlen.S

Packit Service 82fcde
/* Optimized version of the standard strlen() function.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Copyright (C) 2000-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   Contributed by Dan Pop <Dan.Pop@cern.ch>.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
/* Return: the length of the input string
Packit Service 82fcde
Packit Service 82fcde
   Input:
Packit Service 82fcde
        in0:    str
Packit Service 82fcde
Packit Service 82fcde
   Look for the null character byte by byte, until we reach a word aligned
Packit Service 82fcde
   address, then search word by word, using the czx instruction.  We're
Packit Service 82fcde
   also doing one word of read ahead, which could cause problems if the
Packit Service 82fcde
   null character is on the last word of a page and the next page is not
Packit Service 82fcde
   mapped in the process address space.  Hence the use of the speculative
Packit Service 82fcde
   load.
Packit Service 82fcde
Packit Service 82fcde
   This implementation assumes little endian mode.  For big endian mode,
Packit Service 82fcde
   the instruction czx1.r should be replaced by czx1.l.  */
Packit Service 82fcde
Packit Service 82fcde
#include <sysdep.h>
Packit Service 82fcde
#undef ret
Packit Service 82fcde
Packit Service 82fcde
#define saved_lc	r18
Packit Service 82fcde
#define str		r19
Packit Service 82fcde
#define pos0		r20
Packit Service 82fcde
#define val1		r21
Packit Service 82fcde
#define val2		r22
Packit Service 82fcde
#define origadd		r23
Packit Service 82fcde
#define tmp		r24
Packit Service 82fcde
#define loopcnt		r30
Packit Service 82fcde
#define len		ret0
Packit Service 82fcde
Packit Service 82fcde
ENTRY(strlen)
Packit Service 82fcde
	.prologue
Packit Service 82fcde
	alloc r2 = ar.pfs, 1, 0, 0, 0
Packit Service 82fcde
	.save ar.lc, saved_lc
Packit Service 82fcde
        mov 	saved_lc = ar.lc 	// save the loop counter
Packit Service 82fcde
	.body
Packit Service 82fcde
	mov 	str = in0
Packit Service 82fcde
	mov 	len = r0		// len = 0
Packit Service 82fcde
	and 	tmp = 7, in0		// tmp = str % 8
Packit Service 82fcde
	;;
Packit Service 82fcde
	sub	loopcnt = 8, tmp	// loopcnt = 8 - tmp
Packit Service 82fcde
	cmp.eq	p6, p0 = tmp, r0
Packit Service 82fcde
(p6)	br.cond.sptk	.str_aligned;;
Packit Service 82fcde
	adds	loopcnt = -1, loopcnt;;
Packit Service 82fcde
	mov	ar.lc = loopcnt
Packit Service 82fcde
.l1:
Packit Service 82fcde
	ld1	val2 = [str], 1
Packit Service 82fcde
	;;
Packit Service 82fcde
	cmp.eq	p6, p0 = val2, r0
Packit Service 82fcde
(p6)	br.cond.spnt	.restore_and_exit
Packit Service 82fcde
	adds	len = 1, len
Packit Service 82fcde
	br.cloop.dptk	.l1
Packit Service 82fcde
.str_aligned:
Packit Service 82fcde
	mov	origadd = str		// origadd = orig
Packit Service 82fcde
	ld8	val1 = [str], 8;;
Packit Service 82fcde
	nop.b	0
Packit Service 82fcde
	nop.b 	0
Packit Service 82fcde
.l2:	ld8.s	val2 = [str], 8		// don't bomb out here
Packit Service 82fcde
	czx1.r	pos0 = val1
Packit Service 82fcde
	;;
Packit Service 82fcde
	cmp.ne	p6, p0 = 8, pos0
Packit Service 82fcde
(p6)	br.cond.spnt .foundit
Packit Service 82fcde
	chk.s	val2, .recovery
Packit Service 82fcde
.back:
Packit Service 82fcde
	mov	val1 = val2
Packit Service 82fcde
	br.cond.dptk	.l2
Packit Service 82fcde
.foundit:
Packit Service 82fcde
	sub	tmp = str, origadd	// tmp = crt address - orig
Packit Service 82fcde
	add	len = len, pos0;;
Packit Service 82fcde
	add	len = len, tmp;;
Packit Service 82fcde
	adds	len = -16, len
Packit Service 82fcde
.restore_and_exit:
Packit Service 82fcde
	mov ar.lc = saved_lc		// restore the loop counter
Packit Service 82fcde
	br.ret.sptk.many b0
Packit Service 82fcde
.recovery:
Packit Service 82fcde
	adds	str = -8, str;;
Packit Service 82fcde
	ld8	val2 = [str], 8		// bomb out here
Packit Service 82fcde
	br.cond.sptk	.back
Packit Service 82fcde
END(strlen)
Packit Service 82fcde
libc_hidden_builtin_def (strlen)