1*84d9c625SLionel Sambuc/* $NetBSD: ffs.S,v 1.3 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.3 2013/09/12 15:36:14 joerg Exp $") 44b6cbf720SGianluca Guida#endif 45b6cbf720SGianluca Guida#endif /* LIBC_SCCS and not lint */ 46b6cbf720SGianluca Guida 47b6cbf720SGianluca Guida#if 0 48b6cbf720SGianluca Guida /* 49b6cbf720SGianluca Guida * We have a popcount instruction -- use it. 50b6cbf720SGianluca Guida * only uses %o0, %o1, %o2 51b6cbf720SGianluca Guida * 52b6cbf720SGianluca Guida * Here's the pseudo-code from the v9 spec: 53b6cbf720SGianluca Guida * 54b6cbf720SGianluca Guida * int ffs(unsigned zz) { 55b6cbf720SGianluca Guida * return popc( zz ^ ( ~ (-zz))); 56b6cbf720SGianluca Guida * } 57b6cbf720SGianluca Guida * 58b6cbf720SGianluca Guida * XXXX sptifires and blackbirds don't implement popc, 59b6cbf720SGianluca Guida * so we won't use this nice clean code 8^(. 60b6cbf720SGianluca Guida */ 61b6cbf720SGianluca GuidaENTRY(ffs) 62b6cbf720SGianluca Guida neg %o0, %o1 ! %o1 = -zz 63b6cbf720SGianluca Guida xnor %o0, %o1, %o2 ! %o2 = zz ^ ~ -zz 64b6cbf720SGianluca Guida popc %o2, %o1 65b6cbf720SGianluca Guida movrz %o0, %g0, %o1 ! result of ffs(0) should be zero 66b6cbf720SGianluca Guida retl 67b6cbf720SGianluca Guida mov %o1, %o0 68b6cbf720SGianluca Guida#endif 69b6cbf720SGianluca Guida/* 70b6cbf720SGianluca Guida * ffs returns the number of the rightmost bit set in its argument, 71b6cbf720SGianluca Guida * i.e., the lowest value such that (x & (ffs(x) - 1)) is nonzero. 72b6cbf720SGianluca Guida * If no bits are set, ffs returns 0. 73b6cbf720SGianluca Guida * 74b6cbf720SGianluca Guida * We use a table lookup on each byte. 75b6cbf720SGianluca Guida * 76b6cbf720SGianluca Guida * In each section below, %o1 is the current byte (0, 1, 2, or 3). 77b6cbf720SGianluca Guida * The last byte is handled specially: for the first three, 78b6cbf720SGianluca Guida * if that byte is nonzero, we return the table value 79b6cbf720SGianluca Guida * (plus 0, 8, or 16 for the byte number), but for the last 80b6cbf720SGianluca Guida * one, we just return the table value plus 24. This means 81b6cbf720SGianluca Guida * that ffstab[0] must be -24 so that ffs(0) will return 0. 82b6cbf720SGianluca Guida */ 83b6cbf720SGianluca GuidaENTRY(ffs) 84*84d9c625SLionel Sambuc#ifdef __PIC__ 85b6cbf720SGianluca Guida PICCY_SET(ffstab, %o2, %o3) 86b6cbf720SGianluca Guida#else 87b6cbf720SGianluca Guida set ffstab, %o2 88b6cbf720SGianluca Guida#endif 89b6cbf720SGianluca Guida andcc %o0, 0xff, %o1 ! get low byte 90b6cbf720SGianluca Guida be,a 1f ! try again if 0 91b6cbf720SGianluca Guida srl %o0, 8, %o0 ! delay slot, get ready for next byte 92b6cbf720SGianluca Guida 93b6cbf720SGianluca Guida retl ! return ffstab[%o1] 94b6cbf720SGianluca Guida ldsb [%o2 + %o1], %o0 95b6cbf720SGianluca Guida 96b6cbf720SGianluca Guida1: 97b6cbf720SGianluca Guida andcc %o0, 0xff, %o1 ! byte 1 like byte 0... 98b6cbf720SGianluca Guida be,a 2f 99b6cbf720SGianluca Guida srl %o0, 8, %o0 ! (use delay to prepare for byte 2) 100b6cbf720SGianluca Guida 101b6cbf720SGianluca Guida ldsb [%o2 + %o1], %o0 102b6cbf720SGianluca Guida retl ! return ffstab[%o1] + 8 103b6cbf720SGianluca Guida add %o0, 8, %o0 104b6cbf720SGianluca Guida 105b6cbf720SGianluca Guida2: 106b6cbf720SGianluca Guida andcc %o0, 0xff, %o1 107b6cbf720SGianluca Guida be,a 3f 108b6cbf720SGianluca Guida srl %o0, 8, %o0 ! (prepare for byte 3) 109b6cbf720SGianluca Guida 110b6cbf720SGianluca Guida ldsb [%o2 + %o1], %o0 111b6cbf720SGianluca Guida retl ! return ffstab[%o1] + 16 112b6cbf720SGianluca Guida add %o0, 16, %o0 113b6cbf720SGianluca Guida 114b6cbf720SGianluca Guida3: ! just return ffstab[%o0] + 24 115b6cbf720SGianluca Guida ldsb [%o2 + %o0], %o0 116b6cbf720SGianluca Guida retl 117b6cbf720SGianluca Guida add %o0, 24, %o0 118b6cbf720SGianluca Guida 119b6cbf720SGianluca Guidaffstab: 120b6cbf720SGianluca Guida .byte -24,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 00-0f */ 121b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 10-1f */ 122b6cbf720SGianluca Guida .byte 6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 20-2f */ 123b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 30-3f */ 124b6cbf720SGianluca Guida .byte 7,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 40-4f */ 125b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 50-5f */ 126b6cbf720SGianluca Guida .byte 6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 60-6f */ 127b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 70-7f */ 128b6cbf720SGianluca Guida .byte 8,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 80-8f */ 129b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* 10-9f */ 130b6cbf720SGianluca Guida .byte 6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* a0-af */ 131b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* b0-bf */ 132b6cbf720SGianluca Guida .byte 7,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* c0-cf */ 133b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* d0-df */ 134b6cbf720SGianluca Guida .byte 6,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* e0-ef */ 135b6cbf720SGianluca Guida .byte 5,1,2,1,3,1,2,1,4,1,2,1,3,1,2,1 /* f0-ff */ 136