|  | /* strchrnul (str, ch) -- Return pointer to first occurrence of CH in STR | 
|  | or the final NUL byte. | 
|  | For Motorola 68000. | 
|  | Copyright (C) 1999-2016 Free Software Foundation, Inc. | 
|  | This file is part of the GNU C Library. | 
|  | Contributed by Andreas Schwab <schwab@gnu.org>. | 
|  |  | 
|  | The GNU C Library is free software; you can redistribute it and/or | 
|  | modify it under the terms of the GNU Lesser General Public | 
|  | License as published by the Free Software Foundation; either | 
|  | version 2.1 of the License, or (at your option) any later version. | 
|  |  | 
|  | The GNU C Library 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 | 
|  | Lesser General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU Lesser General Public | 
|  | License along with the GNU C Library.  If not, see | 
|  | <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | #include <sysdep.h> | 
|  | #include "asm-syntax.h" | 
|  |  | 
|  | TEXT | 
|  | ENTRY(__strchrnul) | 
|  | /* Save the callee-saved registers we use.  */ | 
|  | movel	R(d2),MEM_PREDEC(sp) | 
|  | cfi_adjust_cfa_offset (4) | 
|  | movel	R(d3),MEM_PREDEC(sp) | 
|  | cfi_adjust_cfa_offset (4) | 
|  | cfi_rel_offset (R(d2), 4) | 
|  | cfi_rel_offset (R(d3), 0) | 
|  |  | 
|  | /* Get string pointer and character.  */ | 
|  | movel	MEM_DISP(sp,12),R(a0) | 
|  | moveb	MEM_DISP(sp,19),R(d0) | 
|  |  | 
|  | /* Distribute the character to all bytes of a longword.  */ | 
|  | movel	R(d0),R(d1) | 
|  | lsll	#8,R(d1) | 
|  | moveb	R(d0),R(d1) | 
|  | movel	R(d1),R(d0) | 
|  | swap	R(d0) | 
|  | movew	R(d1),R(d0) | 
|  |  | 
|  | /* First search for the character one byte at a time until the | 
|  | pointer is aligned to a longword boundary.  */ | 
|  | movel	R(a0),R(d1) | 
|  | #ifdef __mcoldfire__ | 
|  | andl	#3,R(d1) | 
|  | #else | 
|  | andw	#3,R(d1) | 
|  | #endif | 
|  | beq	L(L1) | 
|  | moveb	MEM(a0),R(d2) | 
|  | cmpb	R(d0),R(d2) | 
|  | beq	L(L9) | 
|  | tstb	R(d2) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | #ifdef __mcoldfire__ | 
|  | subql	#3,R(d1) | 
|  | #else | 
|  | subqw	#3,R(d1) | 
|  | #endif | 
|  | beq	L(L1) | 
|  | moveb	MEM(a0),R(d2) | 
|  | cmpb	R(d0),R(d2) | 
|  | beq	L(L9) | 
|  | tstb	R(d2) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | #ifdef __mcoldfire__ | 
|  | addql	#1,R(d1) | 
|  | #else | 
|  | addqw	#1,R(d1) | 
|  | #endif | 
|  | beq	L(L1) | 
|  | moveb	MEM(a0),R(d2) | 
|  | cmpb	R(d0),R(d2) | 
|  | beq	L(L9) | 
|  | tstb	R(d2) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | L(L1:) | 
|  | /* Load the magic bits.  Unlike the generic implementation we can | 
|  | use the carry bit as the fourth hole.  */ | 
|  | movel	#0xfefefeff,R(d3) | 
|  |  | 
|  | /* We exit the loop if adding MAGIC_BITS to LONGWORD fails to | 
|  | change any of the hole bits of LONGWORD. | 
|  |  | 
|  | 1) Is this safe?  Will it catch all the zero bytes? | 
|  | Suppose there is a byte with all zeros.  Any carry bits | 
|  | propagating from its left will fall into the hole at its | 
|  | least significant bit and stop.  Since there will be no | 
|  | carry from its most significant bit, the LSB of the | 
|  | byte to the left will be unchanged, and the zero will be | 
|  | detected. | 
|  |  | 
|  | 2) Is this worthwhile?  Will it ignore everything except | 
|  | zero bytes?  Suppose every byte of LONGWORD has a bit set | 
|  | somewhere.  There will be a carry into bit 8.	If bit 8 | 
|  | is set, this will carry into bit 16.  If bit 8 is clear, | 
|  | one of bits 9-15 must be set, so there will be a carry | 
|  | into bit 16.  Similarly, there will be a carry into bit | 
|  | 24.  If one of bits 24-31 is set, there will be a carry | 
|  | into bit 32 (=carry flag), so all of the hole bits will | 
|  | be changed. | 
|  |  | 
|  | 3) But wait!  Aren't we looking for C, not zero? | 
|  | Good point.  So what we do is XOR LONGWORD with a longword, | 
|  | each of whose bytes is C.  This turns each byte that is C | 
|  | into a zero.  */ | 
|  |  | 
|  | L(L2:) | 
|  | /* Get the longword in question.  */ | 
|  | movel	MEM_POSTINC(a0),R(d1) | 
|  | /* XOR with the byte we search for.  */ | 
|  | eorl	R(d0),R(d1) | 
|  |  | 
|  | /* Add the magic value.  We get carry bits reported for each byte | 
|  | which is not C.  */ | 
|  | movel	R(d3),R(d2) | 
|  | addl	R(d1),R(d2) | 
|  |  | 
|  | /* Check the fourth carry bit before it is clobbered by the next | 
|  | XOR.  If it is not set we have a hit.  */ | 
|  | bcc	L(L8) | 
|  |  | 
|  | /* We are only interested in carry bits that change due to the | 
|  | previous add, so remove original bits.  */ | 
|  | eorl	R(d1),R(d2) | 
|  |  | 
|  | /* Now test for the other three overflow bits. | 
|  | Set all non-carry bits.  */ | 
|  | orl	R(d3),R(d2) | 
|  | /* Add 1 to get zero if all carry bits were set.  */ | 
|  | addql	#1,R(d2) | 
|  |  | 
|  | /* If we don't get zero then at least one byte of the word equals | 
|  | C.  */ | 
|  | bne	L(L8) | 
|  |  | 
|  | /* Next look for a NUL byte. | 
|  | Restore original longword without reload.  */ | 
|  | eorl	R(d0),R(d1) | 
|  | /* Add the magic value.  We get carry bits reported for each byte | 
|  | which is not NUL.  */ | 
|  | movel	R(d3),R(d2) | 
|  | addl	R(d1),R(d2) | 
|  |  | 
|  | /* Check the fourth carry bit before it is clobbered by the next | 
|  | XOR.  If it is not set we have a hit.  */ | 
|  | bcc	L(L8) | 
|  |  | 
|  | /* We are only interested in carry bits that change due to the | 
|  | previous add, so remove original bits.  */ | 
|  | eorl	R(d1),R(d2) | 
|  |  | 
|  | /* Now test for the other three overflow bits. | 
|  | Set all non-carry bits.  */ | 
|  | orl	R(d3),R(d2) | 
|  | /* Add 1 to get zero if all carry bits were set.  */ | 
|  | addql	#1,R(d2) | 
|  |  | 
|  | /* If we don't get zero then at least one byte of the word was | 
|  | NUL.  Otherwise continue with the next longword.  */ | 
|  | bne	L(L8) | 
|  |  | 
|  | /* Get the longword in question.  */ | 
|  | movel	MEM_POSTINC(a0),R(d1) | 
|  | /* XOR with the byte we search for.  */ | 
|  | eorl	R(d0),R(d1) | 
|  |  | 
|  | /* Add the magic value.  We get carry bits reported for each byte | 
|  | which is not C.  */ | 
|  | movel	R(d3),R(d2) | 
|  | addl	R(d1),R(d2) | 
|  |  | 
|  | /* Check the fourth carry bit before it is clobbered by the next | 
|  | XOR.  If it is not set we have a hit.  */ | 
|  | bcc	L(L8) | 
|  |  | 
|  | /* We are only interested in carry bits that change due to the | 
|  | previous add, so remove original bits */ | 
|  | eorl	R(d1),R(d2) | 
|  |  | 
|  | /* Now test for the other three overflow bits. | 
|  | Set all non-carry bits.  */ | 
|  | orl	R(d3),R(d2) | 
|  | /* Add 1 to get zero if all carry bits were set.  */ | 
|  | addql	#1,R(d2) | 
|  |  | 
|  | /* If we don't get zero then at least one byte of the word equals | 
|  | C.  */ | 
|  | bne	L(L8) | 
|  |  | 
|  | /* Next look for a NUL byte. | 
|  | Restore original longword without reload.  */ | 
|  | eorl	R(d0),R(d1) | 
|  | /* Add the magic value.  We get carry bits reported for each byte | 
|  | which is not NUL.  */ | 
|  | movel	R(d3),R(d2) | 
|  | addl	R(d1),R(d2) | 
|  |  | 
|  | /* Check the fourth carry bit before it is clobbered by the next | 
|  | XOR.  If it is not set we have a hit.  */ | 
|  | bcc	L(L8) | 
|  |  | 
|  | /* We are only interested in carry bits that change due to the | 
|  | previous add, so remove original bits */ | 
|  | eorl	R(d1),R(d2) | 
|  |  | 
|  | /* Now test for the other three overflow bits. | 
|  | Set all non-carry bits.  */ | 
|  | orl	R(d3),R(d2) | 
|  | /* Add 1 to get zero if all carry bits were set.  */ | 
|  | addql	#1,R(d2) | 
|  |  | 
|  | /* If we don't get zero then at least one byte of the word was | 
|  | NUL.  Otherwise continue with the next longword.  */ | 
|  | beq	L(L2) | 
|  |  | 
|  | L(L8:) | 
|  | /* We have a hit.  Check to see which byte it was.  First | 
|  | compensate for the autoincrement in the loop.  */ | 
|  | subql	#4,R(a0) | 
|  |  | 
|  | moveb	MEM(a0),R(d1) | 
|  | cmpb	R(d0),R(d1) | 
|  | beq	L(L9) | 
|  | tstb	R(d1) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | moveb	MEM(a0),R(d1) | 
|  | cmpb	R(d0),R(d1) | 
|  | beq	L(L9) | 
|  | tstb	R(d1) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | moveb	MEM(a0),R(d1) | 
|  | cmpb	R(d0),R(d1) | 
|  | beq	L(L9) | 
|  | tstb	R(d1) | 
|  | beq	L(L9) | 
|  | addql	#1,R(a0) | 
|  |  | 
|  | /* Otherwise the fourth byte must equal C or be NUL.  */ | 
|  | L(L9:) | 
|  | movel	R(a0),R(d0) | 
|  | movel	MEM_POSTINC(sp),R(d3) | 
|  | cfi_adjust_cfa_offset (-4) | 
|  | cfi_restore (R(d3)) | 
|  | movel	MEM_POSTINC(sp),R(d2) | 
|  | cfi_adjust_cfa_offset (-4) | 
|  | cfi_restore (R(d2)) | 
|  | rts | 
|  | END(__strchrnul) | 
|  |  | 
|  | weak_alias (__strchrnul, strchrnul) |