What are the semantics of uextend? How does ISLE know how much to extend the input value?
On a related note, we're currently trying to verify uextend and are having trouble because we don't know the size of Inst
. Is there a way to enforce a bit width on Inst
for some terms? Does it even make sense to think about Inst
having a size?
How does ISLE know how much to extend the input value?
The bit width of the result value and of the input value should determine the extend that is happening. For example, see the usage of the has_type
and value_type
extractors here: https://github.com/bytecodealliance/wasmtime/blob/25a588c35f52a7005d7f455facc15e9e35273497/cranelift/codegen/src/isa/x64/lower.isle#L2150-L2152
Does it even make sense to think about
Inst
having a size?
Not really. Instead it makes sense to talk about the size of values, and an Inst
can produce zero or more values, each of which have an associated size.
How does ISLE know how much to extend the input value?
for a rule like
(rule (lower (has_type (fits_in_64 out) (uextend x @ (value_type in))))
(extend x $false (ty_bits in) (ty_bits out)))
are we saying that in
could have any width, and we extend it to 8, 16, 32, or 64 bytes?
Does it even make sense to think about Inst having a size?
Ah, yes, that makes more sense. So for certain instructions we should think about constraining the size of Value
?
are we saying that in could have any width, and we extend it to 8, 16, 32, or 64 bytes?
pretty much, although there is a little more in the extend
term and also just what is implicitly valid or not for CLIF (dest must be wider than source; checked when the clif validator is enabled)
So for certain instructions we should think about constraining the size of Value?
Yes, that sounds right.
We're having some trouble mapping isle types to our internal ir types. Specifically, we want Value
to be constrained in different ways depending on which rule it's being used in (in the below example we want Value
to be a bv64).
(decl extend (Reg bool u8 u8) Reg)
(rule (extend rn signed from_bits to_bits)
(let ((dst WritableReg (temp_writable_reg $I64))
(_ Unit (emit (MInst.Extend dst rn signed from_bits to_bits))))
dst))
(rule (lower (has_type (fits_in_64 out) (uextend x @ (value_type in))))
(extend x $false (ty_bits in) (ty_bits out)))
In other rules, we don't necessarily want this constraint. Do you have any advice on how to map the same type name to different ir types?
Does isle type-checking have any way to differentiate Value
in different rules? How does it distinguish a Value
that represents a u8
from one that represents a u64
?
hi @Monica Pardeshi -- the short answer is that in ISLE those cases aren't different types; there's an abstraction-levels sort of issue going on here, where the type at the "compiler implementation" abstraction level is Value
(an SSA value) but the type at the "IR value" level is u8
or u64
. (In other words, type of compiler SSA identifier vs. type of value it identifies.)
If we want to encode u8
vs u64
somehow, that gets into dependent-type territory; but I would question: do we really need it?
In my mind the approach y'all have been aiming for has always been: map Value
s to a bitvector big enough for the biggest value; then make assertions about its bits. A u8
is stored in the low 8 bits, so we only have assertions about the low 8 bits; the upper bits are undef / "don't-cares". One could think of this sort of like having an abstract machine model where Value
gets stored in a "register" of some width. Does that make sense? Or is there some reason the SMT-solver-level bitvector width has to be narrower?
Last updated: Jan 24 2025 at 00:11 UTC