xref: /llvm-project/compiler-rt/test/scudo/realloc.cpp (revision f7c5c0d87b8ae5e55006fd3a31994cd68d64f102)
1*f7c5c0d8SMitch Phillips // RUN: %clangxx_scudo %s -lstdc++ -o %t
2*f7c5c0d8SMitch Phillips // RUN: %run %t pointers   2>&1
3*f7c5c0d8SMitch Phillips // RUN: %run %t contents   2>&1
4*f7c5c0d8SMitch Phillips // RUN: %run %t usablesize 2>&1
5*f7c5c0d8SMitch Phillips 
6*f7c5c0d8SMitch Phillips // Tests that our reallocation function returns the same pointer when the
7*f7c5c0d8SMitch Phillips // requested size can fit into the previously allocated chunk. Also tests that
8*f7c5c0d8SMitch Phillips // a new chunk is returned if the size is greater, and that the contents of the
9*f7c5c0d8SMitch Phillips // chunk are left unchanged. Finally, checks that realloc copies the usable
10*f7c5c0d8SMitch Phillips // size of the old chunk to the new one (as opposed to the requested size).
11*f7c5c0d8SMitch Phillips 
12*f7c5c0d8SMitch Phillips #include <assert.h>
13*f7c5c0d8SMitch Phillips #include <malloc.h>
14*f7c5c0d8SMitch Phillips #include <string.h>
15*f7c5c0d8SMitch Phillips 
16*f7c5c0d8SMitch Phillips #include <vector>
17*f7c5c0d8SMitch Phillips 
18*f7c5c0d8SMitch Phillips #include <sanitizer/allocator_interface.h>
19*f7c5c0d8SMitch Phillips 
main(int argc,char ** argv)20*f7c5c0d8SMitch Phillips int main(int argc, char **argv) {
21*f7c5c0d8SMitch Phillips   void *p, *old_p;
22*f7c5c0d8SMitch Phillips   // Those sizes will exercise both allocators (Primary & Secondary).
23*f7c5c0d8SMitch Phillips   std::vector<size_t> sizes{1, 16, 1024, 32768, 1 << 16, 1 << 17, 1 << 20};
24*f7c5c0d8SMitch Phillips 
25*f7c5c0d8SMitch Phillips   assert(argc == 2);
26*f7c5c0d8SMitch Phillips 
27*f7c5c0d8SMitch Phillips   if (!strcmp(argv[1], "usablesize")) {
28*f7c5c0d8SMitch Phillips     // This tests a sketchy behavior inherited from poorly written libraries
29*f7c5c0d8SMitch Phillips     // that have become somewhat standard. When realloc'ing a chunk, the
30*f7c5c0d8SMitch Phillips     // copied contents should span the usable size of the chunk, not the
31*f7c5c0d8SMitch Phillips     // requested size.
32*f7c5c0d8SMitch Phillips     size_t size = 496, usable_size;
33*f7c5c0d8SMitch Phillips     p = nullptr;
34*f7c5c0d8SMitch Phillips     // Make sure we get a chunk with a usable size actually larger than size.
35*f7c5c0d8SMitch Phillips     do {
36*f7c5c0d8SMitch Phillips       if (p)
37*f7c5c0d8SMitch Phillips         free(p);
38*f7c5c0d8SMitch Phillips       size += 16;
39*f7c5c0d8SMitch Phillips       p = malloc(size);
40*f7c5c0d8SMitch Phillips       usable_size = __sanitizer_get_allocated_size(p);
41*f7c5c0d8SMitch Phillips       assert(usable_size >= size);
42*f7c5c0d8SMitch Phillips     } while (usable_size == size);
43*f7c5c0d8SMitch Phillips     for (int i = 0; i < usable_size; i++)
44*f7c5c0d8SMitch Phillips       reinterpret_cast<char *>(p)[i] = 'A';
45*f7c5c0d8SMitch Phillips     old_p = p;
46*f7c5c0d8SMitch Phillips     // Make sure we get a different chunk so that the data is actually copied.
47*f7c5c0d8SMitch Phillips     do {
48*f7c5c0d8SMitch Phillips       size *= 2;
49*f7c5c0d8SMitch Phillips       p = realloc(p, size);
50*f7c5c0d8SMitch Phillips       assert(p);
51*f7c5c0d8SMitch Phillips     } while (p == old_p);
52*f7c5c0d8SMitch Phillips     // The contents of the new chunk must match the old one up to usable_size.
53*f7c5c0d8SMitch Phillips     for (int i = 0; i < usable_size; i++)
54*f7c5c0d8SMitch Phillips       assert(reinterpret_cast<char *>(p)[i] == 'A');
55*f7c5c0d8SMitch Phillips     free(p);
56*f7c5c0d8SMitch Phillips   } else {
57*f7c5c0d8SMitch Phillips     for (size_t size : sizes) {
58*f7c5c0d8SMitch Phillips       if (!strcmp(argv[1], "pointers")) {
59*f7c5c0d8SMitch Phillips         old_p = p = realloc(nullptr, size);
60*f7c5c0d8SMitch Phillips         assert(p);
61*f7c5c0d8SMitch Phillips         size = __sanitizer_get_allocated_size(p);
62*f7c5c0d8SMitch Phillips         // Our realloc implementation will return the same pointer if the size
63*f7c5c0d8SMitch Phillips         // requested is lower than or equal to the usable size of the associated
64*f7c5c0d8SMitch Phillips         // chunk.
65*f7c5c0d8SMitch Phillips         p = realloc(p, size - 1);
66*f7c5c0d8SMitch Phillips         assert(p == old_p);
67*f7c5c0d8SMitch Phillips         p = realloc(p, size);
68*f7c5c0d8SMitch Phillips         assert(p == old_p);
69*f7c5c0d8SMitch Phillips         // And a new one if the size is greater.
70*f7c5c0d8SMitch Phillips         p = realloc(p, size + 1);
71*f7c5c0d8SMitch Phillips         assert(p != old_p);
72*f7c5c0d8SMitch Phillips         // A size of 0 will free the chunk and return nullptr.
73*f7c5c0d8SMitch Phillips         p = realloc(p, 0);
74*f7c5c0d8SMitch Phillips         assert(!p);
75*f7c5c0d8SMitch Phillips         old_p = nullptr;
76*f7c5c0d8SMitch Phillips       }
77*f7c5c0d8SMitch Phillips       if (!strcmp(argv[1], "contents")) {
78*f7c5c0d8SMitch Phillips         p = realloc(nullptr, size);
79*f7c5c0d8SMitch Phillips         assert(p);
80*f7c5c0d8SMitch Phillips         for (int i = 0; i < size; i++)
81*f7c5c0d8SMitch Phillips           reinterpret_cast<char *>(p)[i] = 'A';
82*f7c5c0d8SMitch Phillips         p = realloc(p, size + 1);
83*f7c5c0d8SMitch Phillips         // The contents of the reallocated chunk must match the original one.
84*f7c5c0d8SMitch Phillips         for (int i = 0; i < size; i++)
85*f7c5c0d8SMitch Phillips           assert(reinterpret_cast<char *>(p)[i] == 'A');
86*f7c5c0d8SMitch Phillips       }
87*f7c5c0d8SMitch Phillips     }
88*f7c5c0d8SMitch Phillips   }
89*f7c5c0d8SMitch Phillips   return 0;
90*f7c5c0d8SMitch Phillips }
91*f7c5c0d8SMitch Phillips 
92*f7c5c0d8SMitch Phillips // CHECK: ERROR: invalid chunk type when reallocating address
93