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 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 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
|
/* Tests for data model handling of unknown fns. */
#include <stddef.h>
#include "analyzer-decls.h"
void unknown_fn (void *);
void test_1 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
unknown_fn (NULL);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
unknown_fn (&i);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Even though we're not passing &i to unknown_fn, it escaped
above, so unknown_fn could write to it. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
/* As test_1, but with an unknown fn_ptr. */
void test_1a (void (*fn_ptr) (void *))
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
fn_ptr (NULL);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
fn_ptr (&i);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Even though we're not passing &i to unknown_fn, it escaped
above, so fn_ptr (NULL) could write to it. */
fn_ptr (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
int *global_for_test_2;
void test_2 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
global_for_test_2 = &i;
unknown_fn (NULL);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
global_for_test_2 = NULL;
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Even though the global no longer points to i, it escaped
above, so unknown_fn could write to it. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
struct used_by_test_3
{
int *int_ptr;
};
void test_3 (void)
{
int i;
struct used_by_test_3 s;
s.int_ptr = &i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
unknown_fn (NULL);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
__analyzer_eval (s.int_ptr == &i); /* { dg-warning "TRUE" } */
/* i should escape here. */
unknown_fn (&s);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
__analyzer_eval (s.int_ptr == &i); /* { dg-warning "UNKNOWN" } */
s.int_ptr = NULL;
__analyzer_eval (s.int_ptr == NULL); /* { dg-warning "TRUE" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Even though nothing we know about points to i, it escaped
above, so unknown_fn could write to it. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
struct used_by_test_4
{
int *int_ptr;
};
void test_4 (struct used_by_test_4 *st4_ptr)
{
/* Something unknown called "test_4", and hence *st4_ptr has
effectively already escaped. */
int i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
unknown_fn (NULL);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
/* Given that *st4_ptr has effectively already escaped, calling
an unknown fn should invalidate our knowledge of i". */
st4_ptr->int_ptr = &i;
unknown_fn (NULL);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
/* ...and "&i" should now be treated as having escaped. */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
st4_ptr->int_ptr = NULL;
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
static void __attribute__((noinline))
known_fn (void *ptr)
{
/* Empty. */
}
void test_5 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
known_fn (&i);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Ensure that we don't consider &i to have escaped. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
}
extern int __attribute__ ((__pure__))
unknown_pure_fn (void *);
void test_6 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
unknown_pure_fn (&i);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Ensure that we don't consider &i to have escaped. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
}
extern void unknown_const_fn (const void *);
void test_7 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
/* &i is passed as a const void *, so i shouldn't be clobbered by
the call. */
unknown_const_fn (&i);
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* Ensure that we don't consider &i to have escaped. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
}
struct used_by_test_8
{
int *int_ptr;
};
void test_8 (void)
{
int i;
i = 42;
__analyzer_eval (i == 42); /* { dg-warning "TRUE" } */
struct used_by_test_8 st8;
st8.int_ptr = &i;
/* Although unknown_const_fn takes a const void *, the
int_ptr is a non-const int *, and so &i should be considered
writable. */
unknown_const_fn (&st8);
__analyzer_eval (i == 42); /* { dg-warning "UNKNOWN" } */
i = 17;
__analyzer_eval (i == 17); /* { dg-warning "TRUE" } */
/* &i should be considered to have escaped. */
unknown_fn (NULL);
__analyzer_eval (i == 17); /* { dg-warning "UNKNOWN" } */
}
|