diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2022-05-19 19:08:57 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2022-05-19 19:08:57 -0700 |
commit | 7aaa45187b87573080e92671336ed623f02428a8 (patch) | |
tree | c3aca4f96bf17c7825004bd562fb6862b0bfa9b1 /ffi.c | |
parent | 312166cc7e31ba6a45babcf49a9e22e892ddfd45 (diff) | |
download | txr-7aaa45187b87573080e92671336ed623f02428a8.tar.gz txr-7aaa45187b87573080e92671336ed623f02428a8.tar.bz2 txr-7aaa45187b87573080e92671336ed623f02428a8.zip |
ffi: bugfix: transition between bitfield cell sizes.
* ffi.c (make_ffi_type_struct): When a bitfield is declared for
a type whose alignment is different from that of the previous
member, then we start the new bitfield on a fresh alignment.
As usual, if the previous member is a bitfield which left a
partially filled byte, we skip to the next byte and then do the
alignment. This seems to match GCC behavior.
Diffstat (limited to 'ffi.c')
-rw-r--r-- | ffi.c | 10 |
1 files changed, 10 insertions, 0 deletions
@@ -3462,6 +3462,7 @@ static val make_ffi_type_struct(val syntax, val lisp_type, cobj(coerce(mem_t *, tft), ffi_type_cls, &ffi_type_struct_ops)); ucnum offs = 0; ucnum most_align = 1; + ucnum prev_align = 1; int need_out_handler = 0; int bit_offs = 0; const unsigned bits_int = 8 * sizeof(int); @@ -3527,6 +3528,13 @@ static val make_ffi_type_struct(val syntax, val lisp_type, setcheck(obj, slot); setcheck(obj, type); + if (mtft->bitfield && align != prev_align) { + if (bit_offs) + offs++; + offs = (offs + almask) & ~almask; + bit_offs = 0; + } + if (mtft->bitfield) { ucnum bits_type = 8 * size; ucnum bits = mtft->nelem; @@ -3593,6 +3601,8 @@ static val make_ffi_type_struct(val syntax, val lisp_type, most_align = align; } + prev_align = align; + need_out_handler = need_out_handler || mtft->out != 0; if (mtft->by_value_in) |