206 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			206 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * 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__) */
							 |