cfallin opened issue #3205:
In #3202 and a number of past issues as well, we've repeatedly encountered some uncertainty as to (i) how boolean values of varying widths are stored in registers, (ii) whether they can be loaded/stored or not and what representation they take in memory (see e.g. #3102 where we addressed this for trampoline args), and (iii) how this interacts with boolean<->integer conversion ops like
bint
andbmask
.The understanding we seem to generally agree on is that booleans of width > 1 bit have all bits set (true) or all bits cleared (false). And a
b1
is stored in a register with the same invariant as other integer values: the upper bits (in this case everything above the LSB) are undefined.However, this is far from being confusion-free and unambiguous; thus, it seems reasonable to (i) agree definitively on how bools are represented in registers and memory, and (ii) audit our compliance to whatever invariants we decide on.
cfallin labeled issue #3205:
In #3202 and a number of past issues as well, we've repeatedly encountered some uncertainty as to (i) how boolean values of varying widths are stored in registers, (ii) whether they can be loaded/stored or not and what representation they take in memory (see e.g. #3102 where we addressed this for trampoline args), and (iii) how this interacts with boolean<->integer conversion ops like
bint
andbmask
.The understanding we seem to generally agree on is that booleans of width > 1 bit have all bits set (true) or all bits cleared (false). And a
b1
is stored in a register with the same invariant as other integer values: the upper bits (in this case everything above the LSB) are undefined.However, this is far from being confusion-free and unambiguous; thus, it seems reasonable to (i) agree definitively on how bools are represented in registers and memory, and (ii) audit our compliance to whatever invariants we decide on.
cfallin commented on issue #3205:
cc @abrown @afonso360 @bjorn3 @alexcrichton @akirilov-arm for recent folks who've discussed this topic.
cfallin commented on issue #3205:
I'd like to propose the following for loads and stores:
- Define loads and stores of boolean values. We haven't done this in the past, but it is a somewhat incongruous hole in our semantics, IMHO, and we do load and store booleans sometimes behind the scenes (see: trampoline args). I think it would be better if we formally say what it means to do so.
- Let's define the size of a
b1
to be one byte, and others (b8
and up) as the size of the equivalent integer type.- A store of a
b1
stores 0 or 1 (i.e., it is zero-extended to 8 bits); a store of ab8
..b128
stores all zeroes or all ones.- A load of a boolean type has undefined behavior if the value loaded from memory is not canonical as per the above bullet point.
The last is probably the most controversial; disallowing loads and stores of bools avoids the semantic question, which may have been why this was done (@sunfishcode comments?). But we already allow bitcasts, so we already have some notion of the bits underlying a bool, so IMHO it's better to allow loading/storing those bits.
An alternative is to canonicalize a boolean when loading. For a
b1
this is a mask (and
op); for ab8
and up we could just compare to zero and if the loaded value is nonzero (any bit is set), canonicalize totrue
(all bits set). I actually prefer this option, after a bit of thought: it gives well-defined semantics and avoids illegal bool values (e.g.b128
with half the bits set) that might cause other undefined behavior.Then for bitcasts:
- A raw_bitcast or bitcast from a bool to an int produces the same value that a store would produce.
- A raw_bitcast or bitcast from an int to a bool canonicalizes just as a load does.
All of the above should fully pin down how we store the values in registers as well: we (i) have a well-defined integer interpretation of the bool, (ii) always store it in canonicalized form (all zeroes or all ones in defined bits), and (iii) keep to the same "upper bits undefined" invariant as for integers. With one exception: for a
b1
type I think that we should maintain the lowest 8 bits defined, i.e. lowest byte of0x00
or0x01
only, so that a store can be done without a mask.Thoughts?
cfallin edited a comment on issue #3205:
I'd like to propose the following for loads and stores:
- Define loads and stores of boolean values. We haven't done this in the past, but it is a somewhat incongruous hole in our semantics, IMHO, and we do load and store booleans sometimes behind the scenes (see: trampoline args). I think it would be better if we formally say what it means to do so.
- Let's define the size of a
b1
to be one byte, and others (b8
and up) as the size of the equivalent integer type.- A store of a
b1
stores 0 or 1 (i.e., it is zero-extended to 8 bits); a store of ab8
..b128
stores all zeroes or all ones.- <strike>A load of a boolean type has undefined behavior if the value loaded from memory is not canonical as per the above bullet point.</strike> better option, see below
The last is probably the most controversial; disallowing loads and stores of bools avoids the semantic question, which may have been why this was done (@sunfishcode comments?). But we already allow bitcasts, so we already have some notion of the bits underlying a bool, so IMHO it's better to allow loading/storing those bits.
An alternative is to canonicalize a boolean when loading. For a
b1
this is a mask (and
op); for ab8
and up we could just compare to zero and if the loaded value is nonzero (any bit is set), canonicalize totrue
(all bits set). I actually prefer this option, after a bit of thought: it gives well-defined semantics and avoids illegal bool values (e.g.b128
with half the bits set) that might cause other undefined behavior.Then for bitcasts:
- A raw_bitcast or bitcast from a bool to an int produces the same value that a store would produce.
- A raw_bitcast or bitcast from an int to a bool canonicalizes just as a load does.
All of the above should fully pin down how we store the values in registers as well: we (i) have a well-defined integer interpretation of the bool, (ii) always store it in canonicalized form (all zeroes or all ones in defined bits), and (iii) keep to the same "upper bits undefined" invariant as for integers. With one exception: for a
b1
type I think that we should maintain the lowest 8 bits defined, i.e. lowest byte of0x00
or0x01
only, so that a store can be done without a mask.Thoughts?
cfallin commented on issue #3205:
I've just added this to the agenda for Monday's public Cranelift biweekly meeting as well.
abrown commented on issue #3205:
I think a side effect of going doing the path above would be that
raw_bitcast
is no longer necessary.bitcast
says: "A bitcast is equivalent to storing one type and loading the other type from the same address." If we can load and store booleans I think that eliminates the reason forraw_bitcast
's existence (?).
cfallin commented on issue #3205:
That's a good point; the distinction between the two is pretty confusing, so if we could remove
raw_bitcast
as well and have only one kind of bitcast, all the better.
fitzgen commented on issue #3205:
A store of a
b1
stores 0 or 1 (i.e., it is zero-extended to 8 bits); a store of ab8
..b128
stores all zeroes or all ones.Why not have
b1
store all ones? The inconsistency betweenb1
andb8..b128
is a bit of a surprise.
cfallin commented on issue #3205:
Why not have
b1
store all ones?So my thought process is:
b1
is, semantically, one bit wide; so "all ones" is just0x01
. But because memory is byte-oriented, we have to store a whole byte, so we zero-extend. I definitely recognize this is kind of subjective and "all ones in all bits that are written" is just as much a valid perspective though.Pragmatic reasons:
b1
will be most common, and generating a zero or one is slightly easier than zero or-1
(all ones), e.g. the one instructioncset
on aarch64; andbint
(bool to int) from ab1
with a zero-or-one value is a no-op (just a copy) while it's a masking operation with zero or-1
.
cfallin commented on issue #3205:
I should probably add another possibility too: we could define all of
b1
..b128
to use0
for false and1
for true.The main reason for the all-ones representation, to my understanding, was to be able to use e.g. a
b128
as a SIMD mask after araw_bitcast
(or at least, that's the way it has been explained to me -- many issues back I think someone linked some code that had assumed this). But this is sort of getting a desired higher-level semantic behavior implicitly by setting up the low-level definitions in a certain way; it would be better to require the "turn this bool into an all-ones mask" use-case to usebmask
, which already exists for this purpose. Then we get to use the (IMHO) slightly more intuitive0
/1
values which work well for most cases, and we have a fully consistent behavior. We would just need to look for use-cases that assume all-ones currently and ensure they usebmask
instead.This probably needs a sanity check from a SIMD perspective though -- @abrown / @jlb6740 / @akirilov-arm, thoughts?
cfallin commented on issue #3205:
Ah, here is one example of a dependence on all-ones in cg_clif: link (linked by @afonso360 in #3003)
abrown commented on issue #3205:
I would mention that the all-ones/all-zeroes representation for SIMD is not just because using the output of a comparison as a mask is convenient, it's also that the Wasm SIMD spec specifies that representation (but obviously without the boolean type) and that machines (x64, aarch64) represent things this way as well.
cfallin commented on issue #3205:
@abrown this has me thinking further: the existing Wasm-SIMD implementation (lowering in
cranelift-wasm
specifically) does not useb128
(or any other boolean type), right?So one fundamental question I have is: what is the use-case for boolean types wider than 1 bit? If for an actual SIMD use-case we represent the mask as an
I8X16
(that's what I gather from here and the implementation ofoptionally_bitcast_vector
), then it seems that the question of what Wasm-SIMD masks require is separate from the question of whatb128
means.In other words, what we decide here doesn't impact Wasm-SIMD, because Wasm-SIMD doesn't depend on CLIF-level bool types; so it's better to base on decision on whatever makes the most semantic sense and leads to the least confusion, IMHO.
If
b128
is just "a 1-bit-narrow value stored in a much wider space", then IMHO it makes as much sense to zero-extend it (as one would zero-extend ani8
stored in ani128
worth of space) as sign-extend it.Argued from another perspective: If we say that booleans (abstractly) have two possible values, then storing such a value in anything wider than 1 bit grants us the freedom to choose the bit representation for each of those two possible values. In other words,
b128
is still just semanticallytrue
orfalse
; the bit-patterns that represent those are invisible at the CLIF-semantics level.So I'm starting to convince myself a bit more than the
0
/1
approach is cleanest, and then we need to look for uses ofraw_bitcast
(as in cg_clif) to peek under the hood and see the actual bits, and convert those uses intobmask
orbint
as appropriate. But I'm (still) very curious how that sits with others...
bjorn3 commented on issue #3205:
Why not have b1 store all ones? The inconsistency between b1 and b8..b128 is a bit of a surprise.
Rust stores bools as 0 or 1. Storing b1 as all zeros or all ones would require an extra instruction in the common case to convert it to 0 or 1 when storing it. The larger boolean types are pretty much only useful for vector types where it is common for masks to have all zeros or all ones as value.
akirilov-arm commented on issue #3205:
@cfallin I am a bit confused about your comments on
B128
and its relation to SIMD - isn't that a scalar type? I.e. in the SIMD case wouldn't one use something likeB16X8
?Pragmatic reasons:
b1
will be most common, and generating a zero or one is slightly easier than zero or-1
(all ones), e.g. the one instructioncset
on aarch64...The all ones case is actually a single AArch64 instruction as well,
CSETM
(assuming that the upper bits for types likeB16
are undefined, so that we can set them to all ones).
cfallin commented on issue #3205:
@akirilov-arm indeed, that was more or less my point as well -- in the past it has been described as all-ones "because SIMD" and e.g. see the
raw_bitcast
linked above that uses it as a mask, but really it's atrue
/false
value still, and so the fundamental question is "what does a multi-bit scalar boolean mean".
cfallin labeled issue #3205:
In #3202 and a number of past issues as well, we've repeatedly encountered some uncertainty as to (i) how boolean values of varying widths are stored in registers, (ii) whether they can be loaded/stored or not and what representation they take in memory (see e.g. #3102 where we addressed this for trampoline args), and (iii) how this interacts with boolean<->integer conversion ops like
bint
andbmask
.The understanding we seem to generally agree on is that booleans of width > 1 bit have all bits set (true) or all bits cleared (false). And a
b1
is stored in a register with the same invariant as other integer values: the upper bits (in this case everything above the LSB) are undefined.However, this is far from being confusion-free and unambiguous; thus, it seems reasonable to (i) agree definitively on how bools are represented in registers and memory, and (ii) audit our compliance to whatever invariants we decide on.
cfallin labeled issue #3205:
In #3202 and a number of past issues as well, we've repeatedly encountered some uncertainty as to (i) how boolean values of varying widths are stored in registers, (ii) whether they can be loaded/stored or not and what representation they take in memory (see e.g. #3102 where we addressed this for trampoline args), and (iii) how this interacts with boolean<->integer conversion ops like
bint
andbmask
.The understanding we seem to generally agree on is that booleans of width > 1 bit have all bits set (true) or all bits cleared (false). And a
b1
is stored in a register with the same invariant as other integer values: the upper bits (in this case everything above the LSB) are undefined.However, this is far from being confusion-free and unambiguous; thus, it seems reasonable to (i) agree definitively on how bools are represented in registers and memory, and (ii) audit our compliance to whatever invariants we decide on.
elliottt commented on issue #3205:
Here's a link to the 2021-08-23 meeting notes where this was discussed originally.
elliottt commented on issue #3205:
There has been discussion recently about whether or not we need the boolean types at all, and we seem to be converging on the opinion that we should remove them rather than handle the issues surrounding load/store with
b1
values. Effectively, this would mean that we will remove all boolean types, scalar and vector, replacing them with their bit-width integer equivalent (except forb1
that would be replaced withi8
).I'd like to propose the following semantics for instructions that return boolean values now:
- Instructions that currently return
b1
values will switch to returningi8
values in the range [0,1]. This doesn't immediately address the current issues surrounding load/store withb1
values, and I'm interested to hear how we should address cases that look likex = load.i8 ...; brz x ...
.- All other boolean types would return integer values of the same bit-width, where true would be represented as all
1
s and false would be represented as all0
s. I believe this would be effectively no value-level change from the current behavior.
cfallin commented on issue #3205:
Thanks @elliottt -- I think the above is largely a good plan and will result in nice simplifications and removal of awkward special cases!
x = load.i8 ...; brz x ...
For this case specifically I think the semantics would be for
brz
to branch ifx != 0
; in fact I think it already works this way (accepted types defined here and the mnemonics are "branch if zero" and "branch if not zero"). So the 0/1 encodings of a bool work as always, and other nonzero values are also truthy.
abrown commented on issue #3205:
I agree. IIRC, there may be an optimization or two that rely on the boolean type but given the move towards e-graphs those optimizations will need to be re-worked anyway.
elliottt closed issue #3205:
In #3202 and a number of past issues as well, we've repeatedly encountered some uncertainty as to (i) how boolean values of varying widths are stored in registers, (ii) whether they can be loaded/stored or not and what representation they take in memory (see e.g. #3102 where we addressed this for trampoline args), and (iii) how this interacts with boolean<->integer conversion ops like
bint
andbmask
.The understanding we seem to generally agree on is that booleans of width > 1 bit have all bits set (true) or all bits cleared (false). And a
b1
is stored in a register with the same invariant as other integer values: the upper bits (in this case everything above the LSB) are undefined.However, this is far from being confusion-free and unambiguous; thus, it seems reasonable to (i) agree definitively on how bools are represented in registers and memory, and (ii) audit our compliance to whatever invariants we decide on.
Last updated: Jan 24 2025 at 00:11 UTC