aboutsummaryrefslogtreecommitdiffstats
path: root/gcc-4.9/gcc/testsuite/gcc.target/arc/cond-set-use.c
blob: aee27251a5cdcc4073db6aee15b8d17151874841 (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
125
126
127
128
/* { dg-do run } */
/* { dg-options "-Os" } */

/* Based on gethostbyname_r,
 * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org>
 *
 * Licensed under the LGPL v2.1, see the file COPYING.LIB
 *
 * Extraction / wrapping as test by
 * Joern Rennecke  <joern.rennecke@embecosm.com>
 * Copyright (C) 2013 Free Software Foundation, Inc.
 */

typedef unsigned size_t;
typedef int ssize_t;
typedef unsigned uint32_t;
struct resolv_answer {
 char *dotted;
 int atype;
 int aclass;
 int ttl;
 int rdlength;
 const unsigned char *rdata;
 int rdoffset;
 char* buf;
 size_t buflen;
 size_t add_count;
};
struct hostent
{
  char *h_name;
  char **h_aliases;
  int h_addrtype;
  int h_length;
  char **h_addr_list;
};

int *__attribute__  ((noinline,weak)) nop (void * p) { return p; };
void __attribute__  ((noinline,weak)) seta (struct resolv_answer * p)
{ p->atype = 1;}

int ghostbyname_r(
  struct hostent *result_buf,
  char *buf,
  size_t buflen,
  struct hostent **result,
  int *h_errnop)
{
 char **addr_list;
 char **alias;
 char *alias0;
 int i0;
 struct resolv_answer a;
 int i;

 *result = ((void *)0);

         *h_errnop = -1;

         if ((ssize_t)buflen <= 5)
  return 34;

 alias = (char **)buf;
 addr_list = (char **)buf;

  /* This got turned into branch with conditional move in delay slot.  */
 if ((ssize_t)buflen < 256)
  return 34;


 {
  if (!nop(&i0)) {
   result_buf->h_aliases = alias;
   result_buf->h_addrtype = 2;
   result_buf->h_length = 4;
   result_buf->h_addr_list = addr_list;
   *result = result_buf;
   *h_errnop = 0;
   return 0;
  }
 }


 seta (&a);

 if (a.atype == 1) {

  int need_bytes = sizeof(addr_list[0]) * (a.add_count + 1 + 1);

  int ips_len = a.add_count * a.rdlength;

  buflen -= (need_bytes + ips_len);
  if ((ssize_t)buflen < 0) {
   i = 34;
   goto free_and_ret;
  }

  result_buf->h_addrtype = 2;
  *result = result_buf;
  *h_errnop = 0;
  i = 0;
  goto free_and_ret;
 }

 /* For cse, the 1 was is loaded into a call-saved register;
    the load was hoisted into a delay slot before the conditional load,
    clobbering result_buf, which (conditionally) lived in the same
    call-saved register, because mark_referenced_resources considered the
    destination of the COND_EXEC only clobbered, but not used.  */
 *h_errnop = 1;
 *nop(&i0) = 1;
 i = 2;

 free_and_ret:
 nop (&i0);
 return i;
}

int
main ()
{
  struct hostent buf, *res;
  int i;
  char c;
  ghostbyname_r (&buf,  &c, 1024, &res, &i);
  ghostbyname_r (&buf,  0, 1024, &res, &i);
  return 0;
}