From 643fad12cc4a4c70b4ecc00c01625db1781eaf69 Mon Sep 17 00:00:00 2001 From: Edmund Grimley Evans Date: Tue, 7 Jan 2025 13:42:33 +0000 Subject: [PATCH 1/5] i#3699: Change HEAP_ALIGNMENT to 8 on 32-bit ARM. Some C++ library functions require 8-byte alignment. This made the test drcacheoff.simple pass on 32-bit ARM. Issue: #3699 Change-Id: I507833f3367508fd512ee2d1fa272b04203bfff1 --- core/heap.c | 10 ++++++---- core/heap.h | 6 +++++- core/loader_shared.c | 7 ++++++- 3 files changed, 17 insertions(+), 6 deletions(-) diff --git a/core/heap.c b/core/heap.c index 168084a8c72..a38664734c7 100644 --- a/core/heap.c +++ b/core/heap.c @@ -105,7 +105,8 @@ static const uint BLOCK_SIZES[] = { 8, /* for instr bits */ #ifndef X64 /* for x64 future_fragment_t is 24 bytes (could be 20 if we could put flags last) */ - sizeof(future_fragment_t), /* 12 (24 x64) */ + ALIGN_FORWARD(sizeof(future_fragment_t), /* 12 (24 x64) */ + HEAP_ALIGNMENT), #endif /* we have a lot of size 16 requests for IR but they are transient */ 24, /* fcache empties and vm_area_t are now 20, vm area extras still 24 */ @@ -119,8 +120,9 @@ static const uint BLOCK_SIZES[] = { sizeof(instr_t), /* 112 x64 */ # endif #else - sizeof(fragment_t) + sizeof(direct_linkstub_t) + - sizeof(cbr_fallthrough_linkstub_t), /* 60 dbg / 56 rel */ + ALIGN_FORWARD(sizeof(fragment_t) + sizeof(direct_linkstub_t) + + sizeof(cbr_fallthrough_linkstub_t), /* 60 dbg / 56 rel */ + HEAP_ALIGNMENT), # ifndef DEBUG sizeof(instr_t), /* 72 */ # endif @@ -157,7 +159,7 @@ DECLARE_NEVERPROT_VAR(static bool out_of_vmheap_once, false); #endif /* variable-length: we steal one int for the size */ -#define HEADER_SIZE (sizeof(size_t)) +#define HEADER_SIZE HEAP_ALIGNMENT /* VARIABLE_SIZE is assignable */ #define VARIABLE_SIZE(p) (*(size_t *)((p)-HEADER_SIZE)) #define MEMSET_HEADER(p, value) VARIABLE_SIZE(p) = HEAP_TO_PTR_UINT(value) diff --git a/core/heap.h b/core/heap.h index f46287e01ce..fe65190caab 100644 --- a/core/heap.h +++ b/core/heap.h @@ -93,7 +93,11 @@ typedef enum { } map_flags_t; typedef byte *heap_pc; -#define HEAP_ALIGNMENT sizeof(heap_pc *) +#ifdef ARM +# define HEAP_ALIGNMENT 8 /* Some C++ functions require 8-byte alignment on ARM. */ +#else +# define HEAP_ALIGNMENT sizeof(heap_pc *) +#endif extern vm_area_vector_t *landing_pad_areas; #ifdef X64 diff --git a/core/loader_shared.c b/core/loader_shared.c index 8e7d68fb00a..27a1acace00 100644 --- a/core/loader_shared.c +++ b/core/loader_shared.c @@ -940,7 +940,13 @@ redirect_malloc(size_t size) /* We need extra space to store the size and alignment bit and ensure the returned * pointer is aligned. */ +#ifdef ARM + size_t alloc_size = size + STANDARD_HEAP_ALIGNMENT; + ASSERT(HEAP_ALIGNMENT == STANDARD_HEAP_ALIGNMENT); +#else size_t alloc_size = size + sizeof(size_t) + STANDARD_HEAP_ALIGNMENT - HEAP_ALIGNMENT; + ASSERT(HEAP_ALIGNMENT * 2 == STANDARD_HEAP_ALIGNMENT); +#endif /* Our header is the size itself, with the top bit stolen to indicate alignment. */ if (TEST(REDIRECT_HEADER_SHIFTED, alloc_size)) { /* We do not support the top bit being set as that conflicts with the bit in @@ -958,7 +964,6 @@ redirect_malloc(size_t size) ptr_uint_t res = ALIGN_FORWARD((ptr_uint_t)mem + sizeof(size_t), STANDARD_HEAP_ALIGNMENT); size_t header = alloc_size; - ASSERT(HEAP_ALIGNMENT * 2 == STANDARD_HEAP_ALIGNMENT); ASSERT(!TEST(REDIRECT_HEADER_SHIFTED, header)); if (res == (ptr_uint_t)mem + sizeof(size_t)) { /* Already aligned. */ From 44a881cea125ddbe3b586840e5eb5ab6922199ee Mon Sep 17 00:00:00 2001 From: Edmund Grimley Evans Date: Tue, 7 Jan 2025 14:06:11 +0000 Subject: [PATCH 2/5] format Change-Id: Id4d045560f056bb905916be41e45e49bc1eedf35 --- core/heap.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/heap.c b/core/heap.c index a38664734c7..8d7b231b60b 100644 --- a/core/heap.c +++ b/core/heap.c @@ -121,7 +121,7 @@ static const uint BLOCK_SIZES[] = { # endif #else ALIGN_FORWARD(sizeof(fragment_t) + sizeof(direct_linkstub_t) + - sizeof(cbr_fallthrough_linkstub_t), /* 60 dbg / 56 rel */ + sizeof(cbr_fallthrough_linkstub_t), /* 60 dbg / 56 rel */ HEAP_ALIGNMENT), # ifndef DEBUG sizeof(instr_t), /* 72 */ From 1cb85187dbc4da01b9a90410838dae643b6c8409 Mon Sep 17 00:00:00 2001 From: Edmund Grimley Evans Date: Tue, 7 Jan 2025 18:47:07 +0000 Subject: [PATCH 3/5] Comment and improve MEMSET_HEADER. Change-Id: I797d3ddccc69d862c0c2b543a12aafdfc6ae8cdd --- core/heap.c | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/core/heap.c b/core/heap.c index 8d7b231b60b..020ce71ecc2 100644 --- a/core/heap.c +++ b/core/heap.c @@ -158,11 +158,21 @@ DECLARE_NEVERPROT_VAR(static int block_peak_align_pad[BLOCK_TYPES], { 0 }); DECLARE_NEVERPROT_VAR(static bool out_of_vmheap_once, false); #endif -/* variable-length: we steal one int for the size */ +/* The size of a variable-size allocation is stored as a size_t in the header. + * On 32-bit ARM, HEAP_ALIGNMENT is twice the size of a size_t but the wasted + * space for DR's own use is not expected to be significant as only allocs + * that do not fit in the buckets use headers. Client heap allocations + * probably bear the biggest hit. + */ #define HEADER_SIZE HEAP_ALIGNMENT /* VARIABLE_SIZE is assignable */ #define VARIABLE_SIZE(p) (*(size_t *)((p)-HEADER_SIZE)) -#define MEMSET_HEADER(p, value) VARIABLE_SIZE(p) = HEAP_TO_PTR_UINT(value) +#define MEMSET_HEADER(p, value) \ + do { \ + ASSERT(HEADER_SIZE % sizeof(VARIABLE_SIZE(p)) == 0); \ + for (size_t i = 0; i < HEADER_SIZE / sizeof(VARIABLE_SIZE(p)); i++) \ + (&VARIABLE_SIZE(p))[i] = HEAP_TO_PTR_UINT(value); \ + } while(0) #define GET_VARIABLE_ALLOCATION_SIZE(p) (VARIABLE_SIZE(p) + HEADER_SIZE) /* The heap is allocated in units. From 312c2212f93f5bc0a0f0563270047ee42ce3fa11 Mon Sep 17 00:00:00 2001 From: Edmund Grimley Evans Date: Tue, 7 Jan 2025 19:15:19 +0000 Subject: [PATCH 4/5] format Change-Id: I99c5f59061314267281d3c6dd3504da9a71a2a43 --- core/heap.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/heap.c b/core/heap.c index 020ce71ecc2..a828841d201 100644 --- a/core/heap.c +++ b/core/heap.c @@ -172,7 +172,7 @@ DECLARE_NEVERPROT_VAR(static bool out_of_vmheap_once, false); ASSERT(HEADER_SIZE % sizeof(VARIABLE_SIZE(p)) == 0); \ for (size_t i = 0; i < HEADER_SIZE / sizeof(VARIABLE_SIZE(p)); i++) \ (&VARIABLE_SIZE(p))[i] = HEAP_TO_PTR_UINT(value); \ - } while(0) + } while (0) #define GET_VARIABLE_ALLOCATION_SIZE(p) (VARIABLE_SIZE(p) + HEADER_SIZE) /* The heap is allocated in units. From 96b2d7114c9c22251d220e6062208660757d045b Mon Sep 17 00:00:00 2001 From: Edmund Grimley Evans Date: Tue, 7 Jan 2025 19:22:47 +0000 Subject: [PATCH 5/5] Avoid stupid warning on Windows. Change-Id: Ie60b0797d1180369ea06d768f067039cd5d523bb --- core/heap.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/core/heap.c b/core/heap.c index a828841d201..7ee3298c763 100644 --- a/core/heap.c +++ b/core/heap.c @@ -170,8 +170,8 @@ DECLARE_NEVERPROT_VAR(static bool out_of_vmheap_once, false); #define MEMSET_HEADER(p, value) \ do { \ ASSERT(HEADER_SIZE % sizeof(VARIABLE_SIZE(p)) == 0); \ - for (size_t i = 0; i < HEADER_SIZE / sizeof(VARIABLE_SIZE(p)); i++) \ - (&VARIABLE_SIZE(p))[i] = HEAP_TO_PTR_UINT(value); \ + for (size_t k = 0; k < HEADER_SIZE / sizeof(VARIABLE_SIZE(p)); k++) \ + (&VARIABLE_SIZE(p))[k] = HEAP_TO_PTR_UINT(value); \ } while (0) #define GET_VARIABLE_ALLOCATION_SIZE(p) (VARIABLE_SIZE(p) + HEADER_SIZE)