summaryrefslogtreecommitdiff
path: root/gcc/testsuite/gcc.dg/strlenopt-91.c
blob: 2381d03e44a24038a80910482547cb9f571c7666 (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
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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/* PR tree-optimization/92412 - excessive errno aliasing assumption defeats
   optimization
   { dg-do compile }
   { dg-options "-O2 -Wall -fdump-tree-optimized" } */

typedef __SIZE_TYPE__ size_t;

extern void* alloca (size_t);
extern void* calloc (size_t, size_t);
extern void* malloc (size_t);

extern const char exta[4];
static char stata[] = "123";

void sink (const void*, ...);

#define T(ptr, alloc) do {						\
    const char *p = ptr;						\
    if (p[0] != '1' || p[1] != '2' || p[2] != '3' || p[3] != '\0'	\
	|| __builtin_strlen (p) != 3)					\
      return;								\
									\
    void *q = alloc;							\
    __builtin_strcpy (q, p);						\
									\
    if (p[0] != '1' || p[1] != '2' || p[2] != '3' || p[3] != '\0'	\
	|| __builtin_strlen (p) != 3					\
	|| __builtin_strlen (q) != 3)					\
      __builtin_abort ();						\
									\
    sink (p, q);							\
  } while (0)


void alloca_test_local (unsigned n)
{
  char loca[] = "123";
  T (loca, alloca (n));
}

void alloca_test_extern_const (unsigned n)
{
  T (exta, alloca (n));
}

void alloca_test_static (unsigned n)
{
  T (stata, alloca (n));
}


// Verify fix for PR tree-optimization/92412.
void calloc_test_local (unsigned m, unsigned n)
{
  char loca[] = "123";
  T (loca, calloc (m, n));
}

void calloc_test_extern_const (unsigned m, unsigned n)
{
  T (exta, calloc (m, n));
}

void calloc_test_static (unsigned m, unsigned n)
{
  T (stata, calloc (m, n));
}


// Verify fix for PR tree-optimization/92412.
void malloc_test_local (unsigned n)
{
  char loca[] = "123";
  T (loca, malloc (n));
}

void malloc_test_extern_const (unsigned n)
{
  T (exta, malloc (n));
}

void malloc_test_static (unsigned n)
{
  T (stata, malloc (n));
}


#undef T
#define T(ptr, n) do {							\
    const char *p = ptr;						\
    if (p[0] != '1' || p[1] != '2' || p[2] != '3' || p[3] != '\0'	\
	|| __builtin_strlen (p) != 3)					\
      return;								\
									\
    char vla[n];							\
    char *q = vla;							\
    __builtin_strcpy (q, p);						\
									\
    if (p[0] != '1' || p[1] != '2' || p[2] != '3' || p[3] != '\0'	\
	|| __builtin_strlen (p) != 3					\
	|| __builtin_strlen (q) != 3)					\
      __builtin_abort ();						\
									\
    sink (p, vla);							\
  } while (0)


void vla_test_local (unsigned n)
{
  char loca[] = "123";
  T (loca, n);
}

void vla_test_extern_const (unsigned n)
{
  T (exta, n);
}

void vla_test_static (unsigned n)
{
  T (stata, n);
}

/* { dg-final { scan-tree-dump-not "abort" "optimized" } } */