diff options
author | Thomas Voss <mail@thomasvoss.com> | 2024-05-14 17:47:30 +0200 |
---|---|---|
committer | Thomas Voss <mail@thomasvoss.com> | 2024-05-14 17:47:30 +0200 |
commit | 702ca783218c22e223a64d20baf81de79a489df8 (patch) | |
tree | 6542a6c5ebfa3585d36485422e6c67a5a580d648 | |
parent | a77f15c48e1ed69b3d766eff10fcfd3d2252b17a (diff) |
Workaround for alignof() stupidity
-rw-r--r-- | include/dynarr.h | 27 |
1 files changed, 21 insertions, 6 deletions
diff --git a/include/dynarr.h b/include/dynarr.h index 09dd41b..78871cb 100644 --- a/include/dynarr.h +++ b/include/dynarr.h @@ -7,6 +7,18 @@ #include "_alloc_fn.h" #include "alloc.h" +/* This is required because alignof() only portably works with types and not + expressions… */ +/* clang-format off */ +#ifdef __GNUC__ +# define _mlib_da_alignof(expr) __extension__ alignof(expr) +#else +# include <stddef.h> +# define _mlib_da_alignof(expr) \ + ((size_t) &((struct {char x; typeof(expr) y;} *)0)->y) +#endif +/* clang-format on */ + #define dynarr(T) \ struct { \ T *buf; \ @@ -19,8 +31,9 @@ do { \ if (++(da)->len > (da)->cap) { \ size_t ncap = stdc_bit_ceil((da)->len); \ - (da)->buf = (da)->alloc((da)->ctx, (da)->buf, (da)->cap, ncap, \ - sizeof(*(da)->buf), alignof(*(da)->buf)); \ + (da)->buf = \ + (da)->alloc((da)->ctx, (da)->buf, (da)->cap, ncap, \ + sizeof(*(da)->buf), _mlib_da_alignof(*(da)->buf)); \ (da)->cap = ncap; \ } \ (da)->buf[(da)->len - 1] = (x); \ @@ -30,8 +43,9 @@ do { \ if (((da)->len += (n)) > (da)->cap) { \ size_t ncap = stdc_bit_ceil((da)->len); \ - (da)->buf = (da)->alloc((da)->ctx, (da)->buf, (da)->cap, ncap, \ - sizeof(*(da)->buf), alignof(*(da)->buf)); \ + (da)->buf = \ + (da)->alloc((da)->ctx, (da)->buf, (da)->cap, ncap, \ + sizeof(*(da)->buf), _mlib_da_alignof(*(da)->buf)); \ (da)->cap = ncap; \ } \ memcpy((da)->buf + (da)->len - (n), (xs), (n) * sizeof(*(da)->buf)); \ @@ -40,8 +54,9 @@ #define DAGROW(da, n) \ do { \ if ((n) > (da)->cap) { \ - (da)->buf = (da)->alloc((da)->ctx, (da)->buf, (da)->cap, (n), \ - sizeof(*(da)->buf), alignof(*(da)->buf)); \ + (da)->buf = \ + (da)->alloc((da)->ctx, (da)->buf, (da)->cap, (n), \ + sizeof(*(da)->buf), _mlib_da_alignof(*(da)->buf)); \ (da)->cap = (n); \ } \ } while (false) |