Linux Audio

Check our new training course

Loading...
v6.13.7
  1/* SPDX-License-Identifier: GPL-2.0-only */
  2/*
  3 *  linux/arch/arm/lib/findbit.S
  4 *
  5 *  Copyright (C) 1995-2000 Russell King
  6 *
  7 * 16th March 2001 - John Ripley <jripley@sonicblue.com>
  8 *   Fixed so that "size" is an exclusive not an inclusive quantity.
  9 *   All users of these functions expect exclusive sizes, and may
 10 *   also call with zero size.
 11 * Reworked by rmk.
 12 */
 13#include <linux/linkage.h>
 14#include <asm/assembler.h>
 15#include <asm/unwind.h>
 16                .text
 17
 18#ifdef __ARMEB__
 19#define SWAB_ENDIAN le
 20#else
 21#define SWAB_ENDIAN be
 22#endif
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 23
 24		.macro	find_first, endian, set, name
 25ENTRY(_find_first_\name\()bit_\endian)
 26	UNWIND(	.fnstart)
 
 
 27		teq	r1, #0
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 28		beq	3f
 29		mov	r2, #0
 301:		ldr	r3, [r0], #4
 31		.ifeq \set
 32		mvns	r3, r3			@ invert/test bits
 33		.else
 34		movs	r3, r3			@ test bits
 35		.endif
 36		.ifc \endian, SWAB_ENDIAN
 37		bne	.L_found_swab
 38		.else
 39		bne	.L_found		@ found the bit?
 40		.endif
 41		add	r2, r2, #32		@ next index
 422:		cmp	r2, r1			@ any more?
 43		blo	1b
 443:		mov	r0, r1			@ no more bits
 45		ret	lr
 46	UNWIND(	.fnend)
 47ENDPROC(_find_first_\name\()bit_\endian)
 48		.endm
 49
 50		.macro	find_next, endian, set, name
 51ENTRY(_find_next_\name\()bit_\endian)
 52	UNWIND(	.fnstart)
 53		cmp	r2, r1
 54		bhs	3b
 55		mov	ip, r2, lsr #5		@ word index
 56		add	r0, r0, ip, lsl #2
 57		ands	ip, r2, #31		@ bit position
 58		beq	1b
 59		ldr	r3, [r0], #4
 60		.ifeq \set
 61		mvn	r3, r3			@ invert bits
 62		.endif
 63		.ifc \endian, SWAB_ENDIAN
 64		rev_l	r3, ip
 65		.if	.Lrev_l_uses_tmp
 66		@ we need to recompute ip because rev_l will have overwritten
 67		@ it.
 68		and	ip, r2, #31		@ bit position
 69		.endif
 70		.endif
 71		movs	r3, r3, lsr ip		@ shift off unused bits
 72		bne	.L_found
 73		orr	r2, r2, #31		@ no zero bits
 74		add	r2, r2, #1		@ align bit pointer
 75		b	2b			@ loop for next bit
 76	UNWIND(	.fnend)
 77ENDPROC(_find_next_\name\()bit_\endian)
 78		.endm
 79
 80		.macro	find_bit, endian, set, name
 81		find_first \endian, \set, \name
 82		find_next  \endian, \set, \name
 83		.endm
 84
 85/* _find_first_zero_bit_le and _find_next_zero_bit_le */
 86		find_bit le, 0, zero_
 87
 88/* _find_first_bit_le and _find_next_bit_le */
 89		find_bit le, 1
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 90
 91#ifdef __ARMEB__
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 92
 93/* _find_first_zero_bit_be and _find_next_zero_bit_be */
 94		find_bit be, 0, zero_
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 95
 96/* _find_first_bit_be and _find_next_bit_be */
 97		find_bit be, 1
 
 
 
 
 
 
 
 
 
 
 
 
 
 98
 99#endif
100
101/*
102 * One or more bits in the LSB of r3 are assumed to be set.
103 */
104.L_found_swab:
105	UNWIND(	.fnstart)
106		rev_l	r3, ip
107.L_found:
108#if __LINUX_ARM_ARCH__ >= 7
109		rbit	r3, r3			@ reverse bits
110		clz	r3, r3			@ count high zero bits
111		add	r0, r2, r3		@ add offset of first set bit
112#elif __LINUX_ARM_ARCH__ >= 5
113		rsb	r0, r3, #0
114		and	r3, r3, r0		@ mask out lowest bit set
115		clz	r3, r3			@ count high zero bits
116		rsb	r3, r3, #31		@ offset of first set bit
117		add	r0, r2, r3		@ add offset of first set bit
118#else
119		mov	ip, #~0
120		tst	r3, ip, lsr #16		@ test bits 0-15
121		addeq	r2, r2, #16
122		moveq	r3, r3, lsr #16
123		tst	r3, #0x00ff
124		addeq	r2, r2, #8
125		moveq	r3, r3, lsr #8
126		tst	r3, #0x000f
127		addeq	r2, r2, #4
128		moveq	r3, r3, lsr #4
129		tst	r3, #0x0003
130		addeq	r2, r2, #2
131		moveq	r3, r3, lsr #2
132		tst	r3, #0x0001
133		addeq	r2, r2, #1
134		mov	r0, r2
135#endif
136		cmp	r1, r0			@ Clamp to maxbit
137		movlo	r0, r1
138		ret	lr
139	UNWIND(	.fnend)
v5.9
  1/* SPDX-License-Identifier: GPL-2.0-only */
  2/*
  3 *  linux/arch/arm/lib/findbit.S
  4 *
  5 *  Copyright (C) 1995-2000 Russell King
  6 *
  7 * 16th March 2001 - John Ripley <jripley@sonicblue.com>
  8 *   Fixed so that "size" is an exclusive not an inclusive quantity.
  9 *   All users of these functions expect exclusive sizes, and may
 10 *   also call with zero size.
 11 * Reworked by rmk.
 12 */
 13#include <linux/linkage.h>
 14#include <asm/assembler.h>
 
 15                .text
 16
 17/*
 18 * Purpose  : Find a 'zero' bit
 19 * Prototype: int find_first_zero_bit(void *addr, unsigned int maxbit);
 20 */
 21ENTRY(_find_first_zero_bit_le)
 22		teq	r1, #0	
 23		beq	3f
 24		mov	r2, #0
 251:
 26 ARM(		ldrb	r3, [r0, r2, lsr #3]	)
 27 THUMB(		lsr	r3, r2, #3		)
 28 THUMB(		ldrb	r3, [r0, r3]		)
 29		eors	r3, r3, #0xff		@ invert bits
 30		bne	.L_found		@ any now set - found zero bit
 31		add	r2, r2, #8		@ next bit pointer
 322:		cmp	r2, r1			@ any more?
 33		blo	1b
 343:		mov	r0, r1			@ no free bits
 35		ret	lr
 36ENDPROC(_find_first_zero_bit_le)
 37
 38/*
 39 * Purpose  : Find next 'zero' bit
 40 * Prototype: int find_next_zero_bit(void *addr, unsigned int maxbit, int offset)
 41 */
 42ENTRY(_find_next_zero_bit_le)
 43		teq	r1, #0
 44		beq	3b
 45		ands	ip, r2, #7
 46		beq	1b			@ If new byte, goto old routine
 47 ARM(		ldrb	r3, [r0, r2, lsr #3]	)
 48 THUMB(		lsr	r3, r2, #3		)
 49 THUMB(		ldrb	r3, [r0, r3]		)
 50		eor	r3, r3, #0xff		@ now looking for a 1 bit
 51		movs	r3, r3, lsr ip		@ shift off unused bits
 52		bne	.L_found
 53		orr	r2, r2, #7		@ if zero, then no bits here
 54		add	r2, r2, #1		@ align bit pointer
 55		b	2b			@ loop for next bit
 56ENDPROC(_find_next_zero_bit_le)
 57
 58/*
 59 * Purpose  : Find a 'one' bit
 60 * Prototype: int find_first_bit(const unsigned long *addr, unsigned int maxbit);
 61 */
 62ENTRY(_find_first_bit_le)
 63		teq	r1, #0	
 64		beq	3f
 65		mov	r2, #0
 661:
 67 ARM(		ldrb	r3, [r0, r2, lsr #3]	)
 68 THUMB(		lsr	r3, r2, #3		)
 69 THUMB(		ldrb	r3, [r0, r3]		)
 70		movs	r3, r3
 71		bne	.L_found		@ any now set - found zero bit
 72		add	r2, r2, #8		@ next bit pointer
 
 
 
 
 
 732:		cmp	r2, r1			@ any more?
 74		blo	1b
 753:		mov	r0, r1			@ no free bits
 76		ret	lr
 77ENDPROC(_find_first_bit_le)
 78
 79/*
 80 * Purpose  : Find next 'one' bit
 81 * Prototype: int find_next_zero_bit(void *addr, unsigned int maxbit, int offset)
 82 */
 83ENTRY(_find_next_bit_le)
 84		teq	r1, #0
 85		beq	3b
 86		ands	ip, r2, #7
 87		beq	1b			@ If new byte, goto old routine
 88 ARM(		ldrb	r3, [r0, r2, lsr #3]	)
 89 THUMB(		lsr	r3, r2, #3		)
 90 THUMB(		ldrb	r3, [r0, r3]		)
 
 
 
 
 
 
 
 
 
 
 
 91		movs	r3, r3, lsr ip		@ shift off unused bits
 92		bne	.L_found
 93		orr	r2, r2, #7		@ if zero, then no bits here
 94		add	r2, r2, #1		@ align bit pointer
 95		b	2b			@ loop for next bit
 96ENDPROC(_find_next_bit_le)
 
 
 
 
 
 
 
 97
 98#ifdef __ARMEB__
 
 99
100ENTRY(_find_first_zero_bit_be)
101		teq	r1, #0
102		beq	3f
103		mov	r2, #0
1041:		eor	r3, r2, #0x18		@ big endian byte ordering
105 ARM(		ldrb	r3, [r0, r3, lsr #3]	)
106 THUMB(		lsr	r3, #3			)
107 THUMB(		ldrb	r3, [r0, r3]		)
108		eors	r3, r3, #0xff		@ invert bits
109		bne	.L_found		@ any now set - found zero bit
110		add	r2, r2, #8		@ next bit pointer
1112:		cmp	r2, r1			@ any more?
112		blo	1b
1133:		mov	r0, r1			@ no free bits
114		ret	lr
115ENDPROC(_find_first_zero_bit_be)
116
117ENTRY(_find_next_zero_bit_be)
118		teq	r1, #0
119		beq	3b
120		ands	ip, r2, #7
121		beq	1b			@ If new byte, goto old routine
122		eor	r3, r2, #0x18		@ big endian byte ordering
123 ARM(		ldrb	r3, [r0, r3, lsr #3]	)
124 THUMB(		lsr	r3, #3			)
125 THUMB(		ldrb	r3, [r0, r3]		)
126		eor	r3, r3, #0xff		@ now looking for a 1 bit
127		movs	r3, r3, lsr ip		@ shift off unused bits
128		bne	.L_found
129		orr	r2, r2, #7		@ if zero, then no bits here
130		add	r2, r2, #1		@ align bit pointer
131		b	2b			@ loop for next bit
132ENDPROC(_find_next_zero_bit_be)
133
134ENTRY(_find_first_bit_be)
135		teq	r1, #0
136		beq	3f
137		mov	r2, #0
1381:		eor	r3, r2, #0x18		@ big endian byte ordering
139 ARM(		ldrb	r3, [r0, r3, lsr #3]	)
140 THUMB(		lsr	r3, #3			)
141 THUMB(		ldrb	r3, [r0, r3]		)
142		movs	r3, r3
143		bne	.L_found		@ any now set - found zero bit
144		add	r2, r2, #8		@ next bit pointer
1452:		cmp	r2, r1			@ any more?
146		blo	1b
1473:		mov	r0, r1			@ no free bits
148		ret	lr
149ENDPROC(_find_first_bit_be)
150
151ENTRY(_find_next_bit_be)
152		teq	r1, #0
153		beq	3b
154		ands	ip, r2, #7
155		beq	1b			@ If new byte, goto old routine
156		eor	r3, r2, #0x18		@ big endian byte ordering
157 ARM(		ldrb	r3, [r0, r3, lsr #3]	)
158 THUMB(		lsr	r3, #3			)
159 THUMB(		ldrb	r3, [r0, r3]		)
160		movs	r3, r3, lsr ip		@ shift off unused bits
161		bne	.L_found
162		orr	r2, r2, #7		@ if zero, then no bits here
163		add	r2, r2, #1		@ align bit pointer
164		b	2b			@ loop for next bit
165ENDPROC(_find_next_bit_be)
166
167#endif
168
169/*
170 * One or more bits in the LSB of r3 are assumed to be set.
171 */
 
 
 
172.L_found:
173#if __LINUX_ARM_ARCH__ >= 5
 
 
 
 
174		rsb	r0, r3, #0
175		and	r3, r3, r0
176		clz	r3, r3
177		rsb	r3, r3, #31
178		add	r0, r2, r3
179#else
180		tst	r3, #0x0f
 
 
 
 
 
 
 
181		addeq	r2, r2, #4
182		movne	r3, r3, lsl #4
183		tst	r3, #0x30
184		addeq	r2, r2, #2
185		movne	r3, r3, lsl #2
186		tst	r3, #0x40
187		addeq	r2, r2, #1
188		mov	r0, r2
189#endif
190		cmp	r1, r0			@ Clamp to maxbit
191		movlo	r0, r1
192		ret	lr
193