|  | // RUN: %clangxx_scudo %s -lstdc++ -o %t | 
|  | // RUN: %run %t pointers   2>&1 | 
|  | // RUN: %run %t contents   2>&1 | 
|  | // RUN: %run %t usablesize 2>&1 | 
|  |  | 
|  | // Tests that our reallocation function returns the same pointer when the | 
|  | // requested size can fit into the previously allocated chunk. Also tests that | 
|  | // a new chunk is returned if the size is greater, and that the contents of the | 
|  | // chunk are left unchanged. Finally, checks that realloc copies the usable | 
|  | // size of the old chunk to the new one (as opposed to the requested size). | 
|  |  | 
|  | #include <assert.h> | 
|  | #include <malloc.h> | 
|  | #include <string.h> | 
|  |  | 
|  | #include <vector> | 
|  |  | 
|  | #include <sanitizer/allocator_interface.h> | 
|  |  | 
|  | int main(int argc, char **argv) | 
|  | { | 
|  | void *p, *old_p; | 
|  | // Those sizes will exercise both allocators (Primary & Secondary). | 
|  | std::vector<size_t> sizes{1, 16, 1024, 32768, 1 << 16, 1 << 17, 1 << 20}; | 
|  |  | 
|  | assert(argc == 2); | 
|  |  | 
|  | if (!strcmp(argv[1], "usablesize")) { | 
|  | // This tests a sketchy behavior inherited from poorly written libraries | 
|  | // that have become somewhat standard. When realloc'ing a chunk, the | 
|  | // copied contents should span the usable size of the chunk, not the | 
|  | // requested size. | 
|  | size_t size = 496, usable_size; | 
|  | p = nullptr; | 
|  | // Make sure we get a chunk with a usable size actually larger than size. | 
|  | do { | 
|  | if (p) free(p); | 
|  | size += 16; | 
|  | p = malloc(size); | 
|  | usable_size = __sanitizer_get_allocated_size(p); | 
|  | assert(usable_size >= size); | 
|  | } while (usable_size == size); | 
|  | for (int i = 0; i < usable_size; i++) | 
|  | reinterpret_cast<char *>(p)[i] = 'A'; | 
|  | old_p = p; | 
|  | // Make sure we get a different chunk so that the data is actually copied. | 
|  | do { | 
|  | size *= 2; | 
|  | p = realloc(p, size); | 
|  | assert(p); | 
|  | } while (p == old_p); | 
|  | // The contents of the new chunk must match the old one up to usable_size. | 
|  | for (int i = 0; i < usable_size; i++) | 
|  | assert(reinterpret_cast<char *>(p)[i] == 'A'); | 
|  | free(p); | 
|  | } else { | 
|  | for (size_t size : sizes) { | 
|  | if (!strcmp(argv[1], "pointers")) { | 
|  | old_p = p = realloc(nullptr, size); | 
|  | assert(p); | 
|  | size = __sanitizer_get_allocated_size(p); | 
|  | // Our realloc implementation will return the same pointer if the size | 
|  | // requested is lower than or equal to the usable size of the associated | 
|  | // chunk. | 
|  | p = realloc(p, size - 1); | 
|  | assert(p == old_p); | 
|  | p = realloc(p, size); | 
|  | assert(p == old_p); | 
|  | // And a new one if the size is greater. | 
|  | p = realloc(p, size + 1); | 
|  | assert(p != old_p); | 
|  | // A size of 0 will free the chunk and return nullptr. | 
|  | p = realloc(p, 0); | 
|  | assert(!p); | 
|  | old_p = nullptr; | 
|  | } | 
|  | if (!strcmp(argv[1], "contents")) { | 
|  | p = realloc(nullptr, size); | 
|  | assert(p); | 
|  | for (int i = 0; i < size; i++) | 
|  | reinterpret_cast<char *>(p)[i] = 'A'; | 
|  | p = realloc(p, size + 1); | 
|  | // The contents of the reallocated chunk must match the original one. | 
|  | for (int i = 0; i < size; i++) | 
|  | assert(reinterpret_cast<char *>(p)[i] == 'A'); | 
|  | } | 
|  | } | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // CHECK: ERROR: invalid chunk type when reallocating address |