Blame view

lib/string.c 13.6 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
  /*
   *  linux/lib/string.c
   *
   *  Copyright (C) 1991, 1992  Linus Torvalds
   */
  
  /*
   * stupid library routines.. The optimized versions should generally be found
   * as inline code in <asm-xx/string.h>
   *
   * These are buggy as well..
   *
   * * Fri Jun 25 1999, Ingo Oeser <ioe@informatik.tu-chemnitz.de>
   * -  Added strsep() which will replace strtok() soon (because strsep() is
   *    reentrant and should be faster). Use only strsep() in new code, please.
   *
   * * Sat Feb 09 2002, Jason Thomas <jason@topic.com.au>,
   *                    Matthew Hawkins <matt@mh.dropbear.id.au>
   * -  Kissed strtok() goodbye
   */
  
  #include <linux/types.h>
  #include <linux/string.h>
  #include <linux/ctype.h>
  #include <linux/module.h>
  
  #ifndef __HAVE_ARCH_STRNICMP
  /**
   * strnicmp - Case insensitive, length-limited string comparison
   * @s1: One string
   * @s2: The other string
   * @len: the maximum number of characters to compare
   */
  int strnicmp(const char *s1, const char *s2, size_t len)
  {
  	/* Yes, Virginia, it had better be unsigned */
  	unsigned char c1, c2;
850b92479   Jesper Juhl   [PATCH] lib/strin...
38
  	c1 = c2 = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
39
40
  	if (len) {
  		do {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
41
42
43
44
  			c1 = *s1;
  			c2 = *s2;
  			s1++;
  			s2++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
45
46
47
48
49
50
51
52
53
54
55
56
57
58
  			if (!c1)
  				break;
  			if (!c2)
  				break;
  			if (c1 == c2)
  				continue;
  			c1 = tolower(c1);
  			c2 = tolower(c2);
  			if (c1 != c2)
  				break;
  		} while (--len);
  	}
  	return (int)c1 - (int)c2;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
59
60
  EXPORT_SYMBOL(strnicmp);
  #endif
ded220bd8   David S. Miller   [STRING]: Move st...
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
  #ifndef __HAVE_ARCH_STRCASECMP
  int strcasecmp(const char *s1, const char *s2)
  {
  	int c1, c2;
  
  	do {
  		c1 = tolower(*s1++);
  		c2 = tolower(*s2++);
  	} while (c1 == c2 && c1 != 0);
  	return c1 - c2;
  }
  EXPORT_SYMBOL(strcasecmp);
  #endif
  
  #ifndef __HAVE_ARCH_STRNCASECMP
  int strncasecmp(const char *s1, const char *s2, size_t n)
  {
  	int c1, c2;
  
  	do {
  		c1 = tolower(*s1++);
  		c2 = tolower(*s2++);
  	} while ((--n > 0) && c1 == c2 && c1 != 0);
  	return c1 - c2;
  }
  EXPORT_SYMBOL(strncasecmp);
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
88
89
90
91
92
93
  #ifndef __HAVE_ARCH_STRCPY
  /**
   * strcpy - Copy a %NUL terminated string
   * @dest: Where to copy the string to
   * @src: Where to copy the string from
   */
0c28130b5   Paolo 'Blaisorblade' Giarrusso   [PATCH] x86_64: m...
94
  #undef strcpy
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
95
  char *strcpy(char *dest, const char *src)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
  {
  	char *tmp = dest;
  
  	while ((*dest++ = *src++) != '\0')
  		/* nothing */;
  	return tmp;
  }
  EXPORT_SYMBOL(strcpy);
  #endif
  
  #ifndef __HAVE_ARCH_STRNCPY
  /**
   * strncpy - Copy a length-limited, %NUL-terminated string
   * @dest: Where to copy the string to
   * @src: Where to copy the string from
   * @count: The maximum number of bytes to copy
   *
   * The result is not %NUL-terminated if the source exceeds
   * @count bytes.
252795264   walter harms   [PATCH] documenta...
115
116
117
118
   *
   * In the case where the length of @src is less than  that  of
   * count, the remainder of @dest will be padded with %NUL.
   *
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
119
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
120
  char *strncpy(char *dest, const char *src, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
121
122
123
124
  {
  	char *tmp = dest;
  
  	while (count) {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
125
126
  		if ((*tmp = *src) != 0)
  			src++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
  		tmp++;
  		count--;
  	}
  	return dest;
  }
  EXPORT_SYMBOL(strncpy);
  #endif
  
  #ifndef __HAVE_ARCH_STRLCPY
  /**
   * strlcpy - Copy a %NUL terminated string into a sized buffer
   * @dest: Where to copy the string to
   * @src: Where to copy the string from
   * @size: size of destination buffer
   *
   * Compatible with *BSD: the result is always a valid
   * NUL-terminated string that fits in the buffer (unless,
   * of course, the buffer size is zero). It does not pad
   * out the result like strncpy() does.
   */
  size_t strlcpy(char *dest, const char *src, size_t size)
  {
  	size_t ret = strlen(src);
  
  	if (size) {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
152
  		size_t len = (ret >= size) ? size - 1 : ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
154
155
156
157
158
159
160
161
162
163
164
165
166
  		memcpy(dest, src, len);
  		dest[len] = '\0';
  	}
  	return ret;
  }
  EXPORT_SYMBOL(strlcpy);
  #endif
  
  #ifndef __HAVE_ARCH_STRCAT
  /**
   * strcat - Append one %NUL-terminated string to another
   * @dest: The string to be appended to
   * @src: The string to append to it
   */
0c28130b5   Paolo 'Blaisorblade' Giarrusso   [PATCH] x86_64: m...
167
  #undef strcat
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
168
  char *strcat(char *dest, const char *src)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
169
170
171
172
173
174
175
  {
  	char *tmp = dest;
  
  	while (*dest)
  		dest++;
  	while ((*dest++ = *src++) != '\0')
  		;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
176
177
178
179
180
181
182
183
184
185
186
187
  	return tmp;
  }
  EXPORT_SYMBOL(strcat);
  #endif
  
  #ifndef __HAVE_ARCH_STRNCAT
  /**
   * strncat - Append a length-limited, %NUL-terminated string to another
   * @dest: The string to be appended to
   * @src: The string to append to it
   * @count: The maximum numbers of bytes to copy
   *
72fd4a35a   Robert P. J. Day   [PATCH] Numerous ...
188
   * Note that in contrast to strncpy(), strncat() ensures the result is
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
189
190
   * terminated.
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
191
  char *strncat(char *dest, const char *src, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
192
193
194
195
196
197
198
199
200
201
202
203
204
  {
  	char *tmp = dest;
  
  	if (count) {
  		while (*dest)
  			dest++;
  		while ((*dest++ = *src++) != 0) {
  			if (--count == 0) {
  				*dest = '\0';
  				break;
  			}
  		}
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
  	return tmp;
  }
  EXPORT_SYMBOL(strncat);
  #endif
  
  #ifndef __HAVE_ARCH_STRLCAT
  /**
   * strlcat - Append a length-limited, %NUL-terminated string to another
   * @dest: The string to be appended to
   * @src: The string to append to it
   * @count: The size of the destination buffer.
   */
  size_t strlcat(char *dest, const char *src, size_t count)
  {
  	size_t dsize = strlen(dest);
  	size_t len = strlen(src);
  	size_t res = dsize + len;
  
  	/* This would be a bug */
  	BUG_ON(dsize >= count);
  
  	dest += dsize;
  	count -= dsize;
  	if (len >= count)
  		len = count-1;
  	memcpy(dest, src, len);
  	dest[len] = 0;
  	return res;
  }
  EXPORT_SYMBOL(strlcat);
  #endif
  
  #ifndef __HAVE_ARCH_STRCMP
  /**
   * strcmp - Compare two strings
   * @cs: One string
   * @ct: Another string
   */
0c28130b5   Paolo 'Blaisorblade' Giarrusso   [PATCH] x86_64: m...
243
  #undef strcmp
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
244
  int strcmp(const char *cs, const char *ct)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
245
  {
cc75fb71c   Jesper Juhl   [PATCH] lib/strin...
246
  	signed char __res;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
247
248
249
250
251
  
  	while (1) {
  		if ((__res = *cs - *ct++) != 0 || !*cs++)
  			break;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
253
254
255
256
257
258
259
260
261
262
263
  	return __res;
  }
  EXPORT_SYMBOL(strcmp);
  #endif
  
  #ifndef __HAVE_ARCH_STRNCMP
  /**
   * strncmp - Compare two length-limited strings
   * @cs: One string
   * @ct: Another string
   * @count: The maximum number of bytes to compare
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
264
  int strncmp(const char *cs, const char *ct, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
265
  {
cc75fb71c   Jesper Juhl   [PATCH] lib/strin...
266
  	signed char __res = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
267
268
269
270
271
272
  
  	while (count) {
  		if ((__res = *cs - *ct++) != 0 || !*cs++)
  			break;
  		count--;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
273
274
275
276
277
278
279
280
281
282
283
  	return __res;
  }
  EXPORT_SYMBOL(strncmp);
  #endif
  
  #ifndef __HAVE_ARCH_STRCHR
  /**
   * strchr - Find the first occurrence of a character in a string
   * @s: The string to be searched
   * @c: The character to search for
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
284
  char *strchr(const char *s, int c)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
285
  {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
286
  	for (; *s != (char)c; ++s)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
287
288
  		if (*s == '\0')
  			return NULL;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
289
  	return (char *)s;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
290
291
292
293
294
295
296
297
298
299
  }
  EXPORT_SYMBOL(strchr);
  #endif
  
  #ifndef __HAVE_ARCH_STRRCHR
  /**
   * strrchr - Find the last occurrence of a character in a string
   * @s: The string to be searched
   * @c: The character to search for
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
300
  char *strrchr(const char *s, int c)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
  {
         const char *p = s + strlen(s);
         do {
             if (*p == (char)c)
                 return (char *)p;
         } while (--p >= s);
         return NULL;
  }
  EXPORT_SYMBOL(strrchr);
  #endif
  
  #ifndef __HAVE_ARCH_STRNCHR
  /**
   * strnchr - Find a character in a length limited string
   * @s: The string to be searched
   * @count: The number of characters to be searched
   * @c: The character to search for
   */
  char *strnchr(const char *s, size_t count, int c)
  {
  	for (; count-- && *s != '\0'; ++s)
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
322
323
  		if (*s == (char)c)
  			return (char *)s;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
324
325
326
327
  	return NULL;
  }
  EXPORT_SYMBOL(strnchr);
  #endif
481fad483   Pekka Enberg   [PATCH] strstrip(...
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
  /**
   * strstrip - Removes leading and trailing whitespace from @s.
   * @s: The string to be stripped.
   *
   * Note that the first trailing whitespace is replaced with a %NUL-terminator
   * in the given string @s. Returns a pointer to the first non-whitespace
   * character in @s.
   */
  char *strstrip(char *s)
  {
  	size_t size;
  	char *end;
  
  	size = strlen(s);
  
  	if (!size)
  		return s;
  
  	end = s + size - 1;
6e6d9fa6f   Michael Holzheu   [PATCH] strstrip ...
347
  	while (end >= s && isspace(*end))
481fad483   Pekka Enberg   [PATCH] strstrip(...
348
349
350
351
352
353
354
355
356
  		end--;
  	*(end + 1) = '\0';
  
  	while (*s && isspace(*s))
  		s++;
  
  	return s;
  }
  EXPORT_SYMBOL(strstrip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
357
358
359
360
361
  #ifndef __HAVE_ARCH_STRLEN
  /**
   * strlen - Find the length of a string
   * @s: The string to be sized
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
362
  size_t strlen(const char *s)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
  {
  	const char *sc;
  
  	for (sc = s; *sc != '\0'; ++sc)
  		/* nothing */;
  	return sc - s;
  }
  EXPORT_SYMBOL(strlen);
  #endif
  
  #ifndef __HAVE_ARCH_STRNLEN
  /**
   * strnlen - Find the length of a length-limited string
   * @s: The string to be sized
   * @count: The maximum number of bytes to search
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
379
  size_t strnlen(const char *s, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
380
381
382
383
384
385
386
387
388
389
390
391
  {
  	const char *sc;
  
  	for (sc = s; count-- && *sc != '\0'; ++sc)
  		/* nothing */;
  	return sc - s;
  }
  EXPORT_SYMBOL(strnlen);
  #endif
  
  #ifndef __HAVE_ARCH_STRSPN
  /**
72fd4a35a   Robert P. J. Day   [PATCH] Numerous ...
392
   * strspn - Calculate the length of the initial substring of @s which only contain letters in @accept
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
   * @s: The string to be searched
   * @accept: The string to search for
   */
  size_t strspn(const char *s, const char *accept)
  {
  	const char *p;
  	const char *a;
  	size_t count = 0;
  
  	for (p = s; *p != '\0'; ++p) {
  		for (a = accept; *a != '\0'; ++a) {
  			if (*p == *a)
  				break;
  		}
  		if (*a == '\0')
  			return count;
  		++count;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
411
412
413
414
415
  	return count;
  }
  
  EXPORT_SYMBOL(strspn);
  #endif
8833d328c   Kyle McMartin   [PATCH] Clean up ...
416
  #ifndef __HAVE_ARCH_STRCSPN
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
417
  /**
72fd4a35a   Robert P. J. Day   [PATCH] Numerous ...
418
   * strcspn - Calculate the length of the initial substring of @s which does not contain letters in @reject
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
   * @s: The string to be searched
   * @reject: The string to avoid
   */
  size_t strcspn(const char *s, const char *reject)
  {
  	const char *p;
  	const char *r;
  	size_t count = 0;
  
  	for (p = s; *p != '\0'; ++p) {
  		for (r = reject; *r != '\0'; ++r) {
  			if (*p == *r)
  				return count;
  		}
  		++count;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
435
  	return count;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
436
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
437
  EXPORT_SYMBOL(strcspn);
8833d328c   Kyle McMartin   [PATCH] Clean up ...
438
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
439
440
441
442
443
444
445
  
  #ifndef __HAVE_ARCH_STRPBRK
  /**
   * strpbrk - Find the first occurrence of a set of characters
   * @cs: The string to be searched
   * @ct: The characters to search for
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
446
  char *strpbrk(const char *cs, const char *ct)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
447
  {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
448
  	const char *sc1, *sc2;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
449

51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
450
451
  	for (sc1 = cs; *sc1 != '\0'; ++sc1) {
  		for (sc2 = ct; *sc2 != '\0'; ++sc2) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
452
  			if (*sc1 == *sc2)
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
453
  				return (char *)sc1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
454
455
456
457
  		}
  	}
  	return NULL;
  }
894b5779c   Kyle McMartin   [PATCH] No arch-s...
458
  EXPORT_SYMBOL(strpbrk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
459
460
461
462
463
464
465
466
467
468
469
470
471
472
  #endif
  
  #ifndef __HAVE_ARCH_STRSEP
  /**
   * strsep - Split a string into tokens
   * @s: The string to be searched
   * @ct: The characters to search for
   *
   * strsep() updates @s to point after the token, ready for the next call.
   *
   * It returns empty tokens, too, behaving exactly like the libc function
   * of that name. In fact, it was stolen from glibc2 and de-fancy-fied.
   * Same semantics, slimmer shape. ;)
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
473
  char *strsep(char **s, const char *ct)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
474
  {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
475
476
  	char *sbegin = *s;
  	char *end;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
477
478
479
480
481
482
483
484
  
  	if (sbegin == NULL)
  		return NULL;
  
  	end = strpbrk(sbegin, ct);
  	if (end)
  		*end++ = '\0';
  	*s = end;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
485
486
  	return sbegin;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
487
488
  EXPORT_SYMBOL(strsep);
  #endif
34990cf70   David Brownell   Add a new sysfs_s...
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
  /**
   * sysfs_streq - return true if strings are equal, modulo trailing newline
   * @s1: one string
   * @s2: another string
   *
   * This routine returns true iff two strings are equal, treating both
   * NUL and newline-then-NUL as equivalent string terminations.  It's
   * geared for use with sysfs input strings, which generally terminate
   * with newlines but are compared against values without newlines.
   */
  bool sysfs_streq(const char *s1, const char *s2)
  {
  	while (*s1 && *s1 == *s2) {
  		s1++;
  		s2++;
  	}
  
  	if (*s1 == *s2)
  		return true;
  	if (!*s1 && *s2 == '
  ' && !s2[1])
  		return true;
  	if (*s1 == '
  ' && !s1[1] && !*s2)
  		return true;
  	return false;
  }
  EXPORT_SYMBOL(sysfs_streq);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
517
518
519
520
521
522
523
524
525
  #ifndef __HAVE_ARCH_MEMSET
  /**
   * memset - Fill a region of memory with the given value
   * @s: Pointer to the start of the area.
   * @c: The byte to fill the area with
   * @count: The size of the area.
   *
   * Do not use memset() to access IO space, use memset_io() instead.
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
526
  void *memset(void *s, int c, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
  {
850b92479   Jesper Juhl   [PATCH] lib/strin...
528
  	char *xs = s;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
529
530
531
  
  	while (count--)
  		*xs++ = c;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
  	return s;
  }
  EXPORT_SYMBOL(memset);
  #endif
  
  #ifndef __HAVE_ARCH_MEMCPY
  /**
   * memcpy - Copy one area of memory to another
   * @dest: Where to copy to
   * @src: Where to copy from
   * @count: The size of the area.
   *
   * You should not use this function to access IO space, use memcpy_toio()
   * or memcpy_fromio() instead.
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
547
  void *memcpy(void *dest, const void *src, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
548
  {
850b92479   Jesper Juhl   [PATCH] lib/strin...
549
  	char *tmp = dest;
4c416ab71   Jan-Benedict Glaw   [PATCH] Silence a...
550
  	const char *s = src;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
551
552
553
  
  	while (count--)
  		*tmp++ = *s++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
554
555
556
557
558
559
560
561
562
563
564
565
566
567
  	return dest;
  }
  EXPORT_SYMBOL(memcpy);
  #endif
  
  #ifndef __HAVE_ARCH_MEMMOVE
  /**
   * memmove - Copy one area of memory to another
   * @dest: Where to copy to
   * @src: Where to copy from
   * @count: The size of the area.
   *
   * Unlike memcpy(), memmove() copes with overlapping areas.
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
568
  void *memmove(void *dest, const void *src, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
569
  {
82da2c372   Paul Jackson   [PATCH] lib/strin...
570
571
  	char *tmp;
  	const char *s;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
572
573
  
  	if (dest <= src) {
850b92479   Jesper Juhl   [PATCH] lib/strin...
574
575
  		tmp = dest;
  		s = src;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
576
577
  		while (count--)
  			*tmp++ = *s++;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
578
  	} else {
850b92479   Jesper Juhl   [PATCH] lib/strin...
579
580
581
582
  		tmp = dest;
  		tmp += count;
  		s = src;
  		s += count;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
583
584
  		while (count--)
  			*--tmp = *--s;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
585
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
586
587
588
589
590
591
592
593
594
595
596
597
  	return dest;
  }
  EXPORT_SYMBOL(memmove);
  #endif
  
  #ifndef __HAVE_ARCH_MEMCMP
  /**
   * memcmp - Compare two areas of memory
   * @cs: One area of memory
   * @ct: Another area of memory
   * @count: The size of the area.
   */
0c28130b5   Paolo 'Blaisorblade' Giarrusso   [PATCH] x86_64: m...
598
  #undef memcmp
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
599
  int memcmp(const void *cs, const void *ct, size_t count)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
600
601
602
  {
  	const unsigned char *su1, *su2;
  	int res = 0;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
603
  	for (su1 = cs, su2 = ct; 0 < count; ++su1, ++su2, count--)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
  		if ((res = *su1 - *su2) != 0)
  			break;
  	return res;
  }
  EXPORT_SYMBOL(memcmp);
  #endif
  
  #ifndef __HAVE_ARCH_MEMSCAN
  /**
   * memscan - Find a character in an area of memory.
   * @addr: The memory area
   * @c: The byte to search for
   * @size: The size of the area.
   *
   * returns the address of the first occurrence of @c, or 1 byte past
   * the area if @c is not found
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
621
  void *memscan(void *addr, int c, size_t size)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
622
  {
850b92479   Jesper Juhl   [PATCH] lib/strin...
623
  	unsigned char *p = addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
624
625
626
  
  	while (size) {
  		if (*p == c)
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
627
  			return (void *)p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
628
629
630
  		p++;
  		size--;
  	}
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
631
    	return (void *)p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
632
633
634
635
636
637
638
639
640
641
  }
  EXPORT_SYMBOL(memscan);
  #endif
  
  #ifndef __HAVE_ARCH_STRSTR
  /**
   * strstr - Find the first substring in a %NUL terminated string
   * @s1: The string to be searched
   * @s2: The string to search for
   */
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
642
  char *strstr(const char *s1, const char *s2)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
643
644
645
646
647
  {
  	int l1, l2;
  
  	l2 = strlen(s2);
  	if (!l2)
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
648
  		return (char *)s1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
649
650
651
  	l1 = strlen(s1);
  	while (l1 >= l2) {
  		l1--;
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
652
653
  		if (!memcmp(s1, s2, l2))
  			return (char *)s1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
  		s1++;
  	}
  	return NULL;
  }
  EXPORT_SYMBOL(strstr);
  #endif
  
  #ifndef __HAVE_ARCH_MEMCHR
  /**
   * memchr - Find a character in an area of memory.
   * @s: The memory area
   * @c: The byte to search for
   * @n: The size of the area.
   *
   * returns the address of the first occurrence of @c, or %NULL
   * if @c is not found
   */
  void *memchr(const void *s, int c, size_t n)
  {
  	const unsigned char *p = s;
  	while (n-- != 0) {
          	if ((unsigned char)c == *p++) {
51a0f0f65   Jesper Juhl   [PATCH] lib/strin...
676
  			return (void *)(p - 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
677
678
679
680
681
682
  		}
  	}
  	return NULL;
  }
  EXPORT_SYMBOL(memchr);
  #endif