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