| // RUN: %clang_dfsan %s -O1 -o %t && %run %t |
| // RUN: %clang_dfsan %s -O0 -DO0 -o %t && %run %t |
| |
| #include <assert.h> |
| #include <sanitizer/dfsan_interface.h> |
| |
| typedef struct Pair { |
| int i; |
| char *ptr; |
| } Pair; |
| |
| __attribute__((noinline)) |
| Pair make_pair(int i, char *ptr) { |
| Pair pair; |
| pair.i = i; |
| pair.ptr = ptr; |
| return pair; |
| } |
| |
| __attribute__((noinline)) |
| Pair copy_pair1(const Pair *pair0) { |
| Pair pair; |
| pair.i = pair0->i; |
| pair.ptr = pair0->ptr; |
| return pair; |
| } |
| |
| __attribute__((noinline)) |
| Pair copy_pair2(const Pair pair0) { |
| Pair pair; |
| pair.i = pair0.i; |
| pair.ptr = pair0.ptr; |
| return pair; |
| } |
| |
| int main(void) { |
| int i = 1; |
| char *ptr = NULL; |
| dfsan_label i_label = 1; |
| dfsan_label ptr_label = 2; |
| dfsan_set_label(i_label, &i, sizeof(i)); |
| dfsan_set_label(ptr_label, &ptr, sizeof(ptr)); |
| |
| Pair pair1 = make_pair(i, ptr); |
| int i1 = pair1.i; |
| char *ptr1 = pair1.ptr; |
| |
| dfsan_label i1_label = dfsan_read_label(&i1, sizeof(i1)); |
| dfsan_label ptr1_label = dfsan_read_label(&ptr1, sizeof(ptr1)); |
| #if defined(O0) |
| assert(i1_label == (i_label | ptr_label)); |
| assert(ptr1_label == (i_label | ptr_label)); |
| #else |
| assert(i1_label == i_label); |
| assert(ptr1_label == ptr_label); |
| #endif |
| |
| Pair pair2 = copy_pair1(&pair1); |
| int i2 = pair2.i; |
| char *ptr2 = pair2.ptr; |
| |
| dfsan_label i2_label = dfsan_read_label(&i2, sizeof(i2)); |
| dfsan_label ptr2_label = dfsan_read_label(&ptr2, sizeof(ptr2)); |
| #if defined(O0) |
| assert(i2_label == (i_label | ptr_label)); |
| assert(ptr2_label == (i_label | ptr_label)); |
| #else |
| assert(i2_label == i_label); |
| assert(ptr2_label == ptr_label); |
| #endif |
| |
| Pair pair3 = copy_pair2(pair1); |
| int i3 = pair3.i; |
| char *ptr3 = pair3.ptr; |
| |
| dfsan_label i3_label = dfsan_read_label(&i3, sizeof(i3)); |
| dfsan_label ptr3_label = dfsan_read_label(&ptr3, sizeof(ptr3)); |
| #if defined(O0) |
| assert(i3_label == (i_label | ptr_label)); |
| assert(ptr3_label == (i_label | ptr_label)); |
| #else |
| assert(i3_label == i_label); |
| assert(ptr3_label == ptr_label); |
| #endif |
| |
| |
| return 0; |
| } |