#ifndef _BITS_STRINGS_H #define _BITS_STRINGS_H /** @file * * String functions * */ FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL ); /** * Find first (i.e. least significant) set bit * * @v value Value * @ret lsb Least significant bit set in value (LSB=1), or zero */ static inline __attribute__ (( always_inline )) int __ffsl ( long value ) { unsigned long bits = value; unsigned long lsb; unsigned int lz; /* Extract least significant set bit */ lsb = ( bits & -bits ); /* Count number of leading zeroes before LSB */ __asm__ ( "clz %0, %1" : "=r" ( lz ) : "r" ( lsb ) ); return ( 32 - lz ); } /** * Find first (i.e. least significant) set bit * * @v value Value * @ret lsb Least significant bit set in value (LSB=1), or zero */ static inline __attribute__ (( always_inline )) int __ffsll ( long long value ){ unsigned long high = ( value >> 32 ); unsigned long low = ( value >> 0 ); if ( low ) { return ( __ffsl ( low ) ); } else if ( high ) { return ( 32 + __ffsl ( high ) ); } else { return 0; } } /** * Find last (i.e. most significant) set bit * * @v value Value * @ret msb Most significant bit set in value (LSB=1), or zero */ static inline __attribute__ (( always_inline )) int __flsl ( long value ) { unsigned int lz; /* Count number of leading zeroes */ __asm__ ( "clz %0, %1" : "=r" ( lz ) : "r" ( value ) ); return ( 32 - lz ); } /** * Find last (i.e. most significant) set bit * * @v value Value * @ret msb Most significant bit set in value (LSB=1), or zero */ static inline __attribute__ (( always_inline )) int __flsll ( long long value ){ unsigned long high = ( value >> 32 ); unsigned long low = ( value >> 0 ); if ( high ) { return ( 32 + __flsl ( high ) ); } else if ( low ) { return ( __flsl ( low ) ); } else { return 0; } } #endif /* _BITS_STRINGS_H */