aboutsummaryrefslogtreecommitdiff
path: root/newlib/libc/string/local.h
blob: 8cb43f8e3418cdaa33b057d86442975f3d65be75 (plain)
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
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
#include <_ansi.h>
#include <../ctype/local.h>

/* internal function to compute width of wide char. */
int __wcwidth (wint_t);

/*
   Taken from glibc:
   Add the compiler optimization to inhibit loop transformation to library
   calls.  This is used to avoid recursive calls in memset and memmove
   default implementations.
*/
#ifdef _HAVE_CC_INHIBIT_LOOP_TO_LIBCALL
# define __inhibit_loop_to_libcall \
  __attribute__ ((__optimize__ ("-fno-tree-loop-distribute-patterns")))
#else
# define __inhibit_loop_to_libcall
#endif

/* Nonzero if X is not aligned on a "long" boundary.
 * This macro is used to skip a few bytes to find an aligned pointer.
 * It's better to keep it as is even if _HAVE_HW_MISALIGNED_ACCESS is enabled,
 * to avoid small performance penalties (if they are not zero).  */
#define UNALIGNED_X(X) ((long)X & (sizeof (long) - 1))

#ifdef _HAVE_HW_MISALIGNED_ACCESS
/* Hardware performs unaligned operations with little
 * to no penalty compared to byte-to-byte copy.  */
#define UNALIGNED_X_Y(X, Y) (0)
#else /* _HAVE_HW_MISALIGNED_ACCESS */
/* Nonzero if either X or Y is not aligned on a "long" boundary.  */
#define UNALIGNED_X_Y(X, Y) \
  (((long)X & (sizeof (long) - 1)) | ((long)Y & (sizeof (long) - 1)))
#endif /* _HAVE_HW_MISALIGNED_ACCESS */

/* How many bytes are copied each iteration of the word copy loop.  */
#define LITTLE_BLOCK_SIZE (sizeof (long))

/* How many bytes are copied each iteration of the 4X unrolled loop.  */
#define BIG_BLOCK_SIZE (sizeof (long) << 2)

/* Threshhold for punting to the little block byte copier.  */
#define TOO_SMALL_LITTLE_BLOCK(LEN)  ((LEN) < LITTLE_BLOCK_SIZE)

/* Threshhold for punting to the big block byte copier.  */
#define TOO_SMALL_BIG_BLOCK(LEN)  ((LEN) < BIG_BLOCK_SIZE)

/* Macros for detecting endchar.  */
#if LONG_MAX == 2147483647L
#define DETECT_NULL(X) (((X) - 0x01010101) & ~(X) & 0x80808080)
#else
#if LONG_MAX == 9223372036854775807L
/* Nonzero if X (a long int) contains a NULL byte.  */
#define DETECT_NULL(X) (((X) - 0x0101010101010101) & ~(X) & 0x8080808080808080)
#else
#error long int is not a 32bit or 64bit type.
#endif
#endif

/* Returns nonzero if (long)X contains the byte used to fill (long)MASK.  */
#define DETECT_CHAR(X, MASK) (DETECT_NULL(X ^ MASK))