Re: [PATCH] futex: Support smaller futexes of one byte or two byte size.
From: Peter Zijlstra
Date: Wed Dec 11 2019 - 08:44:56 EST
Please read:
https://people.kernel.org/tglx/notes-about-netiquette-qw89
On Sun, Dec 08, 2019 at 05:18:12PM -0500, Malte Skarupke wrote:
> In my first version WAKE also took a size parameter, however I chose
> to not send that version because there are two problems with that. You
> already noticed the first problem: we would have to store the size and
> verify that it matches on WAKE.
>
> The second problem is with REQUEUE and CMP_REQUEUE: The usual case
> there will be that you have a mutex that's 8 bits in size, but the
> condition variable is harder to fit into 8 bits and will be larger. So
> you would need to pass in two different size flags. But REQUEUE
> doesn't actually care what the size is at all. It just moves waiters
> around. CMP_REQUEUE does care, but only for one of the arguments. So
> it works out perfectly that there is one flag for the size.
>
> Those two cases really helped me clarify what the size argument
> actually is needed for. It's only needed for the "compare" part of
> CMP_REQUEUE. Similarly WAIT could have also been named CMP_WAIT and if
> it was named that, it would also be obvious that the size argument is
> only needed for the "compare" part of WAIT. All the other work that
> WAIT does doesn't actually care about the memory behind the pointer at
> all. It only cares about the address.
>
> That also illustrates what should happen if we splice a futex and call
> WAIT on @ptr and WAKE on @ptr+1: If I understand you correctly, (and
> correct me if I'm wrong here) your point is that there would be an
> inconsistency in the API there. You say it would be inconsistent for a
> size-less WAKE to not wake a futex that it's pointing in the middle
> of.
>
> But with the above thoughts, we realize that since those are two
> different addresses, they refer to two different futexes. It doesn't
> matter that the WAIT was for a four byte futex. That information was
> only relevant for the "compare" part of WAIT. It's not relevant for
> anything else, and therefore it's not relevant for the identity of the
> futex. So the fact that splicing a futex doesn't work (and can't
> easily be made to work) does not point at an inconsistency in the API.
See, I disagree. The WAIT/WAKE pair is an implementation of the MONITOR
pattern. Similar to x86's MONITOR/MWAIT or ARMv8's LDXR/WFE. It allows a
(task vs cpu) blocking implementation of:
WAIT:
while (READ_ONCE(*ptr) != val)
cpu_relax;
WAKE:
WRITE_ONCE(*ptr, val);
With futex that turns into:
WAIT:
for (;;) {
u32 t = READ_ONCE(*ptr);
if (t == val)
break;
futex(WAIT, ptr, t);
}
WAKE:
WRITE_ONCE(*ptr, val);
futex(WAKE, ptr);
IOW, each WAKE corresponds to a STORE.
Extending the above to variable width, you'll see that:
u32 var = 0xff;
u32 *ptr = &var, val = 0xffff;
WAIT:
while (READ_ONCE(*ptr) != val)
cpu_relax();
WAKE:
WRITE_ONCE(*(u8*)ptr+1, 0xff); // let's assume LE
will in fact release the WAIT. IOW, all WAIT cares about is that the
value it went to sleep on has changed. It doesn't care how many bits of
the variable changed or what size store made it happen.
It also works the other way:
u8 var = 0, val = 1;
u8 *ptr = &var;
WAIT:
while (READ_ONCE(*ptr) != val)
cpu_relax();
WAKE:
WRITE_ONCE(*(u32 *)((unsigned long)ptr & ~0x3), 0x01010101);
That u32 store fills our @var with 1 and the WAIT terminates.
In neither example does @ptr (necessarily) match.
> Taking all that into account, I believe there are two possible
> consistent implementations for sized futexes. One of them is the one
> you're asking for, where the size is always passed and always verified
> to be correct. The other one is the one I'm proposing, where the size
> argument only applies to the "compare" part of WAIT and CMP_REQUEUE,
> and all the other work of futexes is size-less and only works on the
> address. (and I think similar reasoning will work for the operations
> that are not supported yet)
>
> I believe that between those two consistent implementations, the one
> with size-less WAKE and REQUEUE is preferable. REQUEUE in particular
> makes clear how we really don't care about the size in these
> operations. There is no difference in behavior when moving between
> futexes of different sizes or the same size. It just doesn't matter.
> But if REQUEUE is size-less, it would be inconsistent for WAKE to
> require a size since REQUEUE is just a WAKE with extra features.
>
> The other downside of the version that checks the size is that we,
> well, have to check the size. That's extra work we have to do and
> extra data we have to store, and I can't come up with any case where a
> user would actually benefit from us doing that extra work.
So I'll argue that, while the pure address mode is perhaps convenient,
it is not consistent. In particular it allows some but disallows other
mixed size overlapping operations. Consistency would either allow or
disallow all.
Yes, being strict is perhaps a little more expensive, but I so prefer
strict and narrow semantics over weird and fuzzy semantics. Also, I
don't think it is actually that much more expensive.
Specifically, as you found, we have available space in
futex_key::offset, 2 more FUT_OFF_ bits would allow encoding 4 different
sizes (1,2,4,8 bytes).
Re REQUEUE, using 4 instead of 2 extra bits in the command word would
allow encoding 2 sizes, one for uaddr and one for uaddr2.
Thomas?