123 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
| /***
 | |
| *bsearch.c - do a binary search
 | |
| *
 | |
| *   Copyright (c) Microsoft Corporation. All rights reserved.
 | |
| *
 | |
| *Purpose:
 | |
| *   defines bsearch() - do a binary search an an array
 | |
| *
 | |
| *******************************************************************************/
 | |
| 
 | |
| #include <cruntime.h>
 | |
| #include <stdlib.h>
 | |
| #include <search.h>
 | |
| #include <internal.h>
 | |
| 
 | |
| #if defined (_M_CEE)
 | |
| #define __fileDECL  __clrcall
 | |
| #else  /* defined (_M_CEE) */
 | |
| #define __fileDECL  __cdecl
 | |
| #endif  /* defined (_M_CEE) */
 | |
| /***
 | |
| *char *bsearch() - do a binary search on an array
 | |
| *
 | |
| *Purpose:
 | |
| *   Does a binary search of a sorted array for a key.
 | |
| *
 | |
| *Entry:
 | |
| *   const char *key    - key to search for
 | |
| *   const char *base   - base of sorted array to search
 | |
| *   unsigned int num   - number of elements in array
 | |
| *   unsigned int width - number of bytes per element
 | |
| *   int (*compare)()   - pointer to function that compares two array
 | |
| *           elements, returning neg when #1 < #2, pos when #1 > #2, and
 | |
| *           0 when they are equal. Function is passed pointers to two
 | |
| *           array elements.
 | |
| *
 | |
| *Exit:
 | |
| *   if key is found:
 | |
| *           returns pointer to occurrence of key in array
 | |
| *   if key is not found:
 | |
| *           returns NULL
 | |
| *
 | |
| *Exceptions:
 | |
| *   Input parameters are validated. Refer to the validation section of the function.
 | |
| *
 | |
| *******************************************************************************/
 | |
| 
 | |
| #ifdef __USE_CONTEXT
 | |
| #define __COMPARE(context, p1, p2) (*compare)(context, p1, p2)
 | |
| #else  /* __USE_CONTEXT */
 | |
| #define __COMPARE(context, p1, p2) (*compare)(p1, p2)
 | |
| #endif  /* __USE_CONTEXT */
 | |
| 
 | |
| #if !defined (_M_CEE)
 | |
| _CRTIMP
 | |
| #endif  /* !defined (_M_CEE) */
 | |
| 
 | |
| SECURITYSAFECRITICAL_ATTRIBUTE
 | |
| #ifdef __USE_CONTEXT
 | |
| void * __fileDECL bsearch_s (
 | |
|     const void *key,
 | |
|     const void *base,
 | |
|     size_t num,
 | |
|     size_t width,
 | |
|     int (__fileDECL *compare)(void *, const void *, const void *),
 | |
|     void *context
 | |
|     )
 | |
| #else  /* __USE_CONTEXT */
 | |
| void * __fileDECL bsearch (
 | |
|     const void *key,
 | |
|     const void *base,
 | |
|     size_t num,
 | |
|     size_t width,
 | |
|     int (__fileDECL *compare)(const void *, const void *)
 | |
|     )
 | |
| #endif  /* __USE_CONTEXT */
 | |
| {
 | |
|     char *lo = (char *)base;
 | |
|     char *hi = (char *)base + (num - 1) * width;
 | |
|     char *mid;
 | |
|     size_t half;
 | |
|     int result;
 | |
| 
 | |
|     /* validation section */
 | |
|     _VALIDATE_RETURN(base != NULL || num == 0, EINVAL, NULL);
 | |
|     _VALIDATE_RETURN(width > 0, EINVAL, NULL);
 | |
|     _VALIDATE_RETURN(compare != NULL, EINVAL, NULL);
 | |
| 
 | |
|         /*
 | |
|         We allow a NULL key here because it breaks some older code and because we do not dereference
 | |
|         this ourselves so we can't be sure that it's a problem for the comparison function
 | |
|         */
 | |
| 
 | |
|     while (lo <= hi)
 | |
|     {
 | |
|         if ((half = num / 2) != 0)
 | |
|         {
 | |
|             mid = lo + (num & 1 ? half : (half - 1)) * width;
 | |
|             if (!(result = __COMPARE(context, key, mid)))
 | |
|                 return(mid);
 | |
|             else if (result < 0)
 | |
|             {
 | |
|                 hi = mid - width;
 | |
|                 num = num & 1 ? half : half-1;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 lo = mid + width;
 | |
|                 num = half;
 | |
|             }
 | |
|         }
 | |
|         else if (num)
 | |
|             return (__COMPARE(context, key, lo) ? NULL : lo);
 | |
|         else
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| #undef __fileDECL
 | |
| #undef __COMPARE
 |