| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (C) 2004 Red Hat, Inc. All Rights Reserved. | 
					
						
							|  |  |  |  * Written by David Howells (dhowells@redhat.com) | 
					
						
							|  |  |  |  * Copyright (C) 2008 IBM Corporation | 
					
						
							|  |  |  |  * Written by Rusty Russell <rusty@rustcorp.com.au> | 
					
						
							|  |  |  |  * (Inspired by David Howell's find_next_bit implementation) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version | 
					
						
							|  |  |  |  * 2 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-29 17:49:55 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/bitops.h"
 | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Find the next set bit in a memory region. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | unsigned long find_next_bit(const unsigned long *addr, unsigned long size, | 
					
						
							| 
									
										
										
										
											2018-12-13 23:37:37 +01:00
										 |  |  |                             unsigned long offset) | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-30 16:31:58 +08:00
										 |  |  |     const unsigned long *p = addr + BIT_WORD(offset); | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  |     unsigned long result = offset & ~(BITS_PER_LONG-1); | 
					
						
							|  |  |  |     unsigned long tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (offset >= size) { | 
					
						
							|  |  |  |         return size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     size -= result; | 
					
						
							|  |  |  |     offset %= BITS_PER_LONG; | 
					
						
							|  |  |  |     if (offset) { | 
					
						
							|  |  |  |         tmp = *(p++); | 
					
						
							|  |  |  |         tmp &= (~0UL << offset); | 
					
						
							|  |  |  |         if (size < BITS_PER_LONG) { | 
					
						
							|  |  |  |             goto found_first; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp) { | 
					
						
							|  |  |  |             goto found_middle; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         size -= BITS_PER_LONG; | 
					
						
							|  |  |  |         result += BITS_PER_LONG; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-03-26 10:58:34 +01:00
										 |  |  |     while (size >= 4*BITS_PER_LONG) { | 
					
						
							|  |  |  |         unsigned long d1, d2, d3; | 
					
						
							|  |  |  |         tmp = *p; | 
					
						
							|  |  |  |         d1 = *(p+1); | 
					
						
							|  |  |  |         d2 = *(p+2); | 
					
						
							|  |  |  |         d3 = *(p+3); | 
					
						
							|  |  |  |         if (tmp) { | 
					
						
							|  |  |  |             goto found_middle; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (d1 | d2 | d3) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         p += 4; | 
					
						
							|  |  |  |         result += 4*BITS_PER_LONG; | 
					
						
							|  |  |  |         size -= 4*BITS_PER_LONG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     while (size >= BITS_PER_LONG) { | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  |         if ((tmp = *(p++))) { | 
					
						
							|  |  |  |             goto found_middle; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         result += BITS_PER_LONG; | 
					
						
							|  |  |  |         size -= BITS_PER_LONG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!size) { | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     tmp = *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | found_first: | 
					
						
							|  |  |  |     tmp &= (~0UL >> (BITS_PER_LONG - size)); | 
					
						
							|  |  |  |     if (tmp == 0UL) {		/* Are any bits set? */ | 
					
						
							|  |  |  |         return result + size;	/* Nope. */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | found_middle: | 
					
						
							| 
									
										
										
										
											2013-02-13 17:47:37 -08:00
										 |  |  |     return result + ctzl(tmp); | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * This implementation of find_{first,next}_zero_bit was stolen from | 
					
						
							|  |  |  |  * Linus' asm-alpha/bitops.h. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | unsigned long find_next_zero_bit(const unsigned long *addr, unsigned long size, | 
					
						
							| 
									
										
										
										
											2018-12-13 23:37:37 +01:00
										 |  |  |                                  unsigned long offset) | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-30 16:31:58 +08:00
										 |  |  |     const unsigned long *p = addr + BIT_WORD(offset); | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  |     unsigned long result = offset & ~(BITS_PER_LONG-1); | 
					
						
							|  |  |  |     unsigned long tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (offset >= size) { | 
					
						
							|  |  |  |         return size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     size -= result; | 
					
						
							|  |  |  |     offset %= BITS_PER_LONG; | 
					
						
							|  |  |  |     if (offset) { | 
					
						
							|  |  |  |         tmp = *(p++); | 
					
						
							|  |  |  |         tmp |= ~0UL >> (BITS_PER_LONG - offset); | 
					
						
							|  |  |  |         if (size < BITS_PER_LONG) { | 
					
						
							|  |  |  |             goto found_first; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (~tmp) { | 
					
						
							|  |  |  |             goto found_middle; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         size -= BITS_PER_LONG; | 
					
						
							|  |  |  |         result += BITS_PER_LONG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     while (size & ~(BITS_PER_LONG-1)) { | 
					
						
							|  |  |  |         if (~(tmp = *(p++))) { | 
					
						
							|  |  |  |             goto found_middle; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         result += BITS_PER_LONG; | 
					
						
							|  |  |  |         size -= BITS_PER_LONG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!size) { | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     tmp = *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | found_first: | 
					
						
							|  |  |  |     tmp |= ~0UL << size; | 
					
						
							|  |  |  |     if (tmp == ~0UL) {	/* Are any bits zero? */ | 
					
						
							|  |  |  |         return result + size;	/* Nope. */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | found_middle: | 
					
						
							| 
									
										
										
										
											2013-02-13 17:47:42 -08:00
										 |  |  |     return result + ctzl(~tmp); | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned long find_last_bit(const unsigned long *addr, unsigned long size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned long words; | 
					
						
							|  |  |  |     unsigned long tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Start at final word. */ | 
					
						
							|  |  |  |     words = size / BITS_PER_LONG; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Partial final word? */ | 
					
						
							|  |  |  |     if (size & (BITS_PER_LONG-1)) { | 
					
						
							|  |  |  |         tmp = (addr[words] & (~0UL >> (BITS_PER_LONG | 
					
						
							|  |  |  |                                        - (size & (BITS_PER_LONG-1))))); | 
					
						
							|  |  |  |         if (tmp) { | 
					
						
							|  |  |  |             goto found; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (words) { | 
					
						
							|  |  |  |         tmp = addr[--words]; | 
					
						
							|  |  |  |         if (tmp) { | 
					
						
							|  |  |  |         found: | 
					
						
							| 
									
										
										
										
											2013-02-13 17:47:41 -08:00
										 |  |  |             return words * BITS_PER_LONG + BITS_PER_LONG - 1 - clzl(tmp); | 
					
						
							| 
									
										
										
										
											2011-02-04 09:06:04 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Not found */ | 
					
						
							|  |  |  |     return size; | 
					
						
							|  |  |  | } |