xref: /minix3/common/lib/libc/arch/sparc/string/ffs.S (revision 84d9c625bfea59e274550651111ae9edfdc40fbd)
1*84d9c625SLionel Sambuc/*	$NetBSD: ffs.S,v 1.4 2013/09/12 15:36:14 joerg Exp $	*/
2b6cbf720SGianluca Guida
3b6cbf720SGianluca Guida/*
4b6cbf720SGianluca Guida * Copyright (c) 1992, 1993
5b6cbf720SGianluca Guida *	The Regents of the University of California.  All rights reserved.
6b6cbf720SGianluca Guida *
7b6cbf720SGianluca Guida * This software was developed by the Computer Systems Engineering group
8b6cbf720SGianluca Guida * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
9b6cbf720SGianluca Guida * contributed to Berkeley.
10b6cbf720SGianluca Guida *
11b6cbf720SGianluca Guida * Redistribution and use in source and binary forms, with or without
12b6cbf720SGianluca Guida * modification, are permitted provided that the following conditions
13b6cbf720SGianluca Guida * are met:
14b6cbf720SGianluca Guida * 1. Redistributions of source code must retain the above copyright
15b6cbf720SGianluca Guida *    notice, this list of conditions and the following disclaimer.
16b6cbf720SGianluca Guida * 2. Redistributions in binary form must reproduce the above copyright
17b6cbf720SGianluca Guida *    notice, this list of conditions and the following disclaimer in the
18b6cbf720SGianluca Guida *    documentation and/or other materials provided with the distribution.
19b6cbf720SGianluca Guida * 3. Neither the name of the University nor the names of its contributors
20b6cbf720SGianluca Guida *    may be used to endorse or promote products derived from this software
21b6cbf720SGianluca Guida *    without specific prior written permission.
22b6cbf720SGianluca Guida *
23b6cbf720SGianluca Guida * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
24b6cbf720SGianluca Guida * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25b6cbf720SGianluca Guida * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26b6cbf720SGianluca Guida * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
27b6cbf720SGianluca Guida * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28b6cbf720SGianluca Guida * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29b6cbf720SGianluca Guida * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30b6cbf720SGianluca Guida * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31b6cbf720SGianluca Guida * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32b6cbf720SGianluca Guida * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33b6cbf720SGianluca Guida * SUCH DAMAGE.
34b6cbf720SGianluca Guida *
35b6cbf720SGianluca Guida * from: Header: ffs.s,v 1.3 92/07/07 00:23:57 torek Exp
36b6cbf720SGianluca Guida */
37b6cbf720SGianluca Guida
38b6cbf720SGianluca Guida#include <machine/asm.h>
39b6cbf720SGianluca Guida#if defined(LIBC_SCCS) && !defined(lint)
40b6cbf720SGianluca Guida#if 0
41b6cbf720SGianluca Guida	.asciz "@(#)ffs.s	8.1 (Berkeley) 6/4/93"
42b6cbf720SGianluca Guida#else
43*84d9c625SLionel Sambuc	RCSID("$NetBSD: ffs.S,v 1.4 2013/09/12 15:36:14 joerg Exp $")
44b6cbf720SGianluca Guida#endif
45b6cbf720SGianluca Guida#endif  /* LIBC_SCCS and not lint */
46b6cbf720SGianluca Guida
47b6cbf720SGianluca Guida/*
48b6cbf720SGianluca Guida * ffs returns the number of the rightmost bit set in its argument,
49b6cbf720SGianluca Guida * i.e., the lowest value such that (x & (ffs(x) - 1)) is nonzero.
50b6cbf720SGianluca Guida * If no bits are set, ffs returns 0.
51b6cbf720SGianluca Guida *
52b6cbf720SGianluca Guida * We use a table lookup on each byte.
53b6cbf720SGianluca Guida *
54b6cbf720SGianluca Guida * In each section below, %o1 is the current byte (0, 1, 2, or 3).
55b6cbf720SGianluca Guida * The last byte is handled specially: for the first three,
56b6cbf720SGianluca Guida * if that byte is nonzero, we return the table value
57b6cbf720SGianluca Guida * (plus 0, 8, or 16 for the byte number), but for the last
58b6cbf720SGianluca Guida * one, we just return the table value plus 24.  This means
59b6cbf720SGianluca Guida * that ffstab[0] must be -24 so that ffs(0) will return 0.
60b6cbf720SGianluca Guida */
61f14fb602SLionel SambucWEAK_ALIAS(__ffssi2,ffs)
62b6cbf720SGianluca GuidaENTRY(ffs)
63*84d9c625SLionel Sambuc#ifdef __PIC__
64b6cbf720SGianluca Guida	PICCY_SET(ffstab, %o2, %o3)
65b6cbf720SGianluca Guida#else
66b6cbf720SGianluca Guida	set	ffstab, %o2
67b6cbf720SGianluca Guida#endif
68b6cbf720SGianluca Guida	andcc	%o0, 0xff, %o1	! get low byte
69b6cbf720SGianluca Guida	be,a	1f		! try again if 0
70b6cbf720SGianluca Guida	 srl	%o0, 8, %o0	! delay slot, get ready for next byte
71b6cbf720SGianluca Guida
72b6cbf720SGianluca Guida	retl			! return ffstab[%o1]
73b6cbf720SGianluca Guida	 ldsb	[%o2 + %o1], %o0
74b6cbf720SGianluca Guida
75b6cbf720SGianluca Guida1:
76b6cbf720SGianluca Guida	andcc	%o0, 0xff, %o1	! byte 1 like byte 0...
77b6cbf720SGianluca Guida	be,a	2f
78b6cbf720SGianluca Guida	 srl	%o0, 8, %o0	! (use delay to prepare for byte 2)
79b6cbf720SGianluca Guida
80b6cbf720SGianluca Guida	ldsb	[%o2 + %o1], %o0
81b6cbf720SGianluca Guida	retl			! return ffstab[%o1] + 8
82b6cbf720SGianluca Guida	 add	%o0, 8, %o0
83b6cbf720SGianluca Guida
84b6cbf720SGianluca Guida2:
85b6cbf720SGianluca Guida	andcc	%o0, 0xff, %o1
86b6cbf720SGianluca Guida	be,a	3f
87b6cbf720SGianluca Guida	 srl	%o0, 8, %o0	! (prepare for byte 3)
88b6cbf720SGianluca Guida
89b6cbf720SGianluca Guida	ldsb	[%o2 + %o1], %o0
90b6cbf720SGianluca Guida	retl			! return ffstab[%o1] + 16
91b6cbf720SGianluca Guida	 add	%o0, 16, %o0
92b6cbf720SGianluca Guida
93b6cbf720SGianluca Guida3:				! just return ffstab[%o0] + 24
94b6cbf720SGianluca Guida	ldsb	[%o2 + %o0], %o0
95b6cbf720SGianluca Guida	retl
96b6cbf720SGianluca Guida	 add	%o0, 24, %o0
97b6cbf720SGianluca Guida
98b6cbf720SGianluca Guidaffstab:
99b6cbf720SGianluca Guida	.byte	-24,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 00-0f */
100b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 10-1f */
101b6cbf720SGianluca Guida	.byte	6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 20-2f */
102b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 30-3f */
103b6cbf720SGianluca Guida	.byte	7,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 40-4f */
104b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 50-5f */
105b6cbf720SGianluca Guida	.byte	6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 60-6f */
106b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 70-7f */
107b6cbf720SGianluca Guida	.byte	8,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 80-8f */
108b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* 10-9f */
109b6cbf720SGianluca Guida	.byte	6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* a0-af */
110b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* b0-bf */
111b6cbf720SGianluca Guida	.byte	7,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* c0-cf */
112b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* d0-df */
113b6cbf720SGianluca Guida	.byte	6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* e0-ef */
114b6cbf720SGianluca Guida	.byte	5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1	/* f0-ff */
115