206 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			206 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /* $Id: string.h,v 1.36 2001/12/21 00:54:31 davem Exp $
 | ||
|  |  * string.h: External definitions for optimized assembly string | ||
|  |  *           routines for the Linux Kernel. | ||
|  |  * | ||
|  |  * Copyright (C) 1995,1996 David S. Miller (davem@caip.rutgers.edu) | ||
|  |  * Copyright (C) 1996,1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz) | ||
|  |  */ | ||
|  | 
 | ||
|  | #ifndef __SPARC_STRING_H__
 | ||
|  | #define __SPARC_STRING_H__
 | ||
|  | 
 | ||
|  | #include <asm/page.h>
 | ||
|  | 
 | ||
|  | /* Really, userland/ksyms should not see any of this stuff. */ | ||
|  | 
 | ||
|  | #ifdef __KERNEL__
 | ||
|  | 
 | ||
|  | extern void __memmove(void *,const void *,__kernel_size_t); | ||
|  | extern __kernel_size_t __memcpy(void *,const void *,__kernel_size_t); | ||
|  | extern __kernel_size_t __memset(void *,int,__kernel_size_t); | ||
|  | 
 | ||
|  | #ifndef EXPORT_SYMTAB_STROPS
 | ||
|  | 
 | ||
|  | /* First the mem*() things. */ | ||
|  | #define __HAVE_ARCH_MEMMOVE
 | ||
|  | #undef memmove
 | ||
|  | #define memmove(_to, _from, _n) \
 | ||
|  | ({ \ | ||
|  | 	void *_t = (_to); \ | ||
|  | 	__memmove(_t, (_from), (_n)); \ | ||
|  | 	_t; \ | ||
|  | }) | ||
|  | 
 | ||
|  | #define __HAVE_ARCH_MEMCPY
 | ||
|  | 
 | ||
|  | static inline void *__constant_memcpy(void *to, const void *from, __kernel_size_t n) | ||
|  | { | ||
|  | 	extern void __copy_1page(void *, const void *); | ||
|  | 
 | ||
|  | 	if(n <= 32) { | ||
|  | 		__builtin_memcpy(to, from, n); | ||
|  | 	} else if (((unsigned int) to & 7) != 0) { | ||
|  | 		/* Destination is not aligned on the double-word boundary */ | ||
|  | 		__memcpy(to, from, n); | ||
|  | 	} else { | ||
|  | 		switch(n) { | ||
|  | 		case PAGE_SIZE: | ||
|  | 			__copy_1page(to, from); | ||
|  | 			break; | ||
|  | 		default: | ||
|  | 			__memcpy(to, from, n); | ||
|  | 			break; | ||
|  | 		} | ||
|  | 	} | ||
|  | 	return to; | ||
|  | } | ||
|  | 
 | ||
|  | static inline void *__nonconstant_memcpy(void *to, const void *from, __kernel_size_t n) | ||
|  | { | ||
|  | 	__memcpy(to, from, n); | ||
|  | 	return to; | ||
|  | } | ||
|  | 
 | ||
|  | #undef memcpy
 | ||
|  | #define memcpy(t, f, n) \
 | ||
|  | (__builtin_constant_p(n) ? \ | ||
|  |  __constant_memcpy((t),(f),(n)) : \ | ||
|  |  __nonconstant_memcpy((t),(f),(n))) | ||
|  | 
 | ||
|  | #define __HAVE_ARCH_MEMSET
 | ||
|  | 
 | ||
|  | static inline void *__constant_c_and_count_memset(void *s, char c, __kernel_size_t count) | ||
|  | { | ||
|  | 	extern void bzero_1page(void *); | ||
|  | 	extern __kernel_size_t __bzero(void *, __kernel_size_t); | ||
|  | 
 | ||
|  | 	if(!c) { | ||
|  | 		if(count == PAGE_SIZE) | ||
|  | 			bzero_1page(s); | ||
|  | 		else | ||
|  | 			__bzero(s, count); | ||
|  | 	} else { | ||
|  | 		__memset(s, c, count); | ||
|  | 	} | ||
|  | 	return s; | ||
|  | } | ||
|  | 
 | ||
|  | static inline void *__constant_c_memset(void *s, char c, __kernel_size_t count) | ||
|  | { | ||
|  | 	extern __kernel_size_t __bzero(void *, __kernel_size_t); | ||
|  | 
 | ||
|  | 	if(!c) | ||
|  | 		__bzero(s, count); | ||
|  | 	else | ||
|  | 		__memset(s, c, count); | ||
|  | 	return s; | ||
|  | } | ||
|  | 
 | ||
|  | static inline void *__nonconstant_memset(void *s, char c, __kernel_size_t count) | ||
|  | { | ||
|  | 	__memset(s, c, count); | ||
|  | 	return s; | ||
|  | } | ||
|  | 
 | ||
|  | #undef memset
 | ||
|  | #define memset(s, c, count) \
 | ||
|  | (__builtin_constant_p(c) ? (__builtin_constant_p(count) ? \ | ||
|  |                             __constant_c_and_count_memset((s), (c), (count)) : \ | ||
|  |                             __constant_c_memset((s), (c), (count))) \ | ||
|  |                           : __nonconstant_memset((s), (c), (count))) | ||
|  | 
 | ||
|  | #define __HAVE_ARCH_MEMSCAN
 | ||
|  | 
 | ||
|  | #undef memscan
 | ||
|  | #define memscan(__arg0, __char, __arg2)						\
 | ||
|  | ({										\ | ||
|  | 	extern void *__memscan_zero(void *, size_t);				\ | ||
|  | 	extern void *__memscan_generic(void *, int, size_t);			\ | ||
|  | 	void *__retval, *__addr = (__arg0);					\ | ||
|  | 	size_t __size = (__arg2);						\ | ||
|  | 										\ | ||
|  | 	if(__builtin_constant_p(__char) && !(__char))				\ | ||
|  | 		__retval = __memscan_zero(__addr, __size);			\ | ||
|  | 	else									\ | ||
|  | 		__retval = __memscan_generic(__addr, (__char), __size);		\ | ||
|  | 										\ | ||
|  | 	__retval;								\ | ||
|  | }) | ||
|  | 
 | ||
|  | #define __HAVE_ARCH_MEMCMP
 | ||
|  | extern int memcmp(const void *,const void *,__kernel_size_t); | ||
|  | 
 | ||
|  | /* Now the str*() stuff... */ | ||
|  | #define __HAVE_ARCH_STRLEN
 | ||
|  | extern __kernel_size_t strlen(const char *); | ||
|  | 
 | ||
|  | #define __HAVE_ARCH_STRNCMP
 | ||
|  | 
 | ||
|  | extern int __strncmp(const char *, const char *, __kernel_size_t); | ||
|  | 
 | ||
|  | static inline int __constant_strncmp(const char *src, const char *dest, __kernel_size_t count) | ||
|  | { | ||
|  | 	register int retval; | ||
|  | 	switch(count) { | ||
|  | 	case 0: return 0; | ||
|  | 	case 1: return (src[0] - dest[0]); | ||
|  | 	case 2: retval = (src[0] - dest[0]); | ||
|  | 		if(!retval && src[0]) | ||
|  | 		  retval = (src[1] - dest[1]); | ||
|  | 		return retval; | ||
|  | 	case 3: retval = (src[0] - dest[0]); | ||
|  | 		if(!retval && src[0]) { | ||
|  | 		  retval = (src[1] - dest[1]); | ||
|  | 		  if(!retval && src[1]) | ||
|  | 		    retval = (src[2] - dest[2]); | ||
|  | 		} | ||
|  | 		return retval; | ||
|  | 	case 4: retval = (src[0] - dest[0]); | ||
|  | 		if(!retval && src[0]) { | ||
|  | 		  retval = (src[1] - dest[1]); | ||
|  | 		  if(!retval && src[1]) { | ||
|  | 		    retval = (src[2] - dest[2]); | ||
|  | 		    if (!retval && src[2]) | ||
|  | 		      retval = (src[3] - dest[3]); | ||
|  | 		  } | ||
|  | 		} | ||
|  | 		return retval; | ||
|  | 	case 5: retval = (src[0] - dest[0]); | ||
|  | 		if(!retval && src[0]) { | ||
|  | 		  retval = (src[1] - dest[1]); | ||
|  | 		  if(!retval && src[1]) { | ||
|  | 		    retval = (src[2] - dest[2]); | ||
|  | 		    if (!retval && src[2]) { | ||
|  | 		      retval = (src[3] - dest[3]); | ||
|  | 		      if (!retval && src[3]) | ||
|  | 		        retval = (src[4] - dest[4]); | ||
|  | 		    } | ||
|  | 		  } | ||
|  | 		} | ||
|  | 		return retval; | ||
|  | 	default: | ||
|  | 		retval = (src[0] - dest[0]); | ||
|  | 		if(!retval && src[0]) { | ||
|  | 		  retval = (src[1] - dest[1]); | ||
|  | 		  if(!retval && src[1]) { | ||
|  | 		    retval = (src[2] - dest[2]); | ||
|  | 		    if(!retval && src[2]) | ||
|  | 		      retval = __strncmp(src+3,dest+3,count-3); | ||
|  | 		  } | ||
|  | 		} | ||
|  | 		return retval; | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | #undef strncmp
 | ||
|  | #define strncmp(__arg0, __arg1, __arg2)	\
 | ||
|  | (__builtin_constant_p(__arg2) ?	\ | ||
|  |  __constant_strncmp(__arg0, __arg1, __arg2) : \ | ||
|  |  __strncmp(__arg0, __arg1, __arg2)) | ||
|  |   | ||
|  | #endif /* !EXPORT_SYMTAB_STROPS */
 | ||
|  | 
 | ||
|  | #endif /* __KERNEL__ */
 | ||
|  | 
 | ||
|  | #endif /* !(__SPARC_STRING_H__) */
 |