On 23 Jan 2022, at 20:17, Junio C Hamano <gitster@xxxxxxxxx> wrote: > > Jessica Clarke <jrtc27@xxxxxxxxxx> writes: > >> Currently mem_pool_alloc uses sizeof(uintmax_t) as a proxy for what >> should be _Alignof(max_align_t) in C11. On most architectures this is > > Lose "Currently", as the present tense describes the status quo, the > shape of the problematic code we have today that wants improvement > by the proposed patch. Do you want a v3 or is that something you'll amend on git-am? >> sufficient (though on m68k it is in fact overly strict, since the >> de-facto ABI, which differs from the specified System V ABI, has the >> maximum alignment of all types as 2 bytes), but on CHERI, and thus Arm's >> Morello prototype, it is insufficient for any type that stores a >> pointer, which must be aligned to 128 bits (on 64-bit architectures >> extended with CHERI), whilst uintmax_t is a 64-bit integer. > > OK. > >> Fix this by introducing our own approximation for max_align_t and a >> means to compute _Alignof it without relying on C11. Currently this >> union only contains uintmax_t and void *, but more types can be added as >> needed. > > Nicely described. > >> +/* >> + * The inner union is an approximation for C11's max_align_t, and the >> + * struct + offsetof computes _Alignof. This can all just be replaced >> + * with _Alignof(max_align_t) if/when C11 is part of the baseline. >> + * >> + * Add more types to the union if the current set is insufficient. >> + */ >> +struct git_max_alignment { >> + char unalign; >> + union { >> + uintmax_t max_align_uintmax; >> + void *max_align_pointer; >> + } aligned; >> +}; >> +#define GIT_MAX_ALIGNMENT offsetof(struct git_max_alignment, aligned) >> + > > The original computed the alignment requirement for uintmax_t as > sizeof(uintmax_t), not as > > offsetof(struct { > char unalign; > union { uintmax_t i; } aligned; > }, aligned) > > because if you have an array of a type, each element of it must be > aligned appropriately already for that type, without the unalignment > the outer struct enforces. I wonder if your complex offsetof is > equivalent to sizeof(union { uintmax_t u; void *p; })? > > IOW, in this struct: > > struct max_alignment_helper { > char unalign; > union { > uintmax_t uintmax_t_unused; > void *pointer_unused; > } u[2]; > } s; > > both s.u[0] and s.u[1] must be properly aligned, so wouldn't the > alignment requirement for the union type, which can be used to hold > a single value of either uintmax_t or a poinhter, be the distance > between these two array elements, i.e. sizeof(s.u[0])? > > To put it differently in yet another way, wouldn't it simplify down > to this? > > union max_alignment_helper { > uintmax_t uintmax_t_unused; > void *pointer_unused; > }; > #define GIT_MAX_ALIGNMENT sizeof(union max_alignment_helper); > > I am not saying that the "a forcibly unaligned union in a struct" is > a bad/wrong way to express what you want to achieve. I just do not > know if there is a reason to choose it over a seemingly simpler > sizeof(that union) without the outer struct and unalign member. So, sizeof(X) does not always equal _Alignof(X), even for primitive types, _Alignof need only be a factor of sizeof. The two are the same on most architectures, and is a sensible ABI, but the exception is the m68k case I was referring to above. On m68k, sizeof(long long) == 8, but _Alignof(long long) == 2 (yes this is a real pain point of its ABI; in particular int is only 2-byte aligned, but futex(2) explicitly requires 4-byte alignment). So using sizeof definitely gets you something sufficiently aligned, but can waste space. This doesn’t affect CHERI/Morello, all our implementations keep sizeof == _Alignof, but as I was changing this code I felt I should use the more precise construct. Jess > Other than that, looks OK to me. Especially the parts that use the > macro look correctly converted. > > Thanks. > >> @@ -69,9 +85,9 @@ void *mem_pool_alloc(struct mem_pool *pool, size_t len) >> struct mp_block *p = NULL; >> void *r; >> >> - /* round up to a 'uintmax_t' alignment */ >> - if (len & (sizeof(uintmax_t) - 1)) >> - len += sizeof(uintmax_t) - (len & (sizeof(uintmax_t) - 1)); >> + /* round up to a 'GIT_MAX_ALIGNMENT' alignment */ >> + if (len & (GIT_MAX_ALIGNMENT - 1)) >> + len += GIT_MAX_ALIGNMENT - (len & (GIT_MAX_ALIGNMENT - 1)); >> >> if (pool->mp_block && >> pool->mp_block->end - pool->mp_block->next_free >= len) > > >> /* >> * Allocate a new mp_block and insert it after the block specified in >> * `insert_after`. If `insert_after` is NULL, then insert block at the >> @@ -69,9 +85,9 @@ void *mem_pool_alloc(struct mem_pool *pool, size_t len) >> struct mp_block *p = NULL; >> void *r; >> >> - /* round up to a 'uintmax_t' alignment */ >> - if (len & (sizeof(uintmax_t) - 1)) >> - len += sizeof(uintmax_t) - (len & (sizeof(uintmax_t) - 1)); >> + /* round up to a 'GIT_MAX_ALIGNMENT' alignment */ >> + if (len & (GIT_MAX_ALIGNMENT - 1)) >> + len += GIT_MAX_ALIGNMENT - (len & (GIT_MAX_ALIGNMENT - 1)); >> >> if (pool->mp_block && >> pool->mp_block->end - pool->mp_block->next_free >= len)