Re: [c++std-parallel-1651] Re: Compilers and RCU readers: Once more unto the breach!
From: Paul E. McKenney
Date: Tue May 26 2015 - 21:41:58 EST
On Tue, May 26, 2015 at 07:08:35PM +0200, Torvald Riegel wrote:
> On Tue, 2015-05-19 at 17:55 -0700, Paul E. McKenney wrote:
> > http://www.rdrop.com/users/paulmck/RCU/consume.2015.05.18a.pdf
>
> I have been discussing Section 7.9 with Paul during last week.
>
> While I think that 7.9 helps narrow down the problem somewhat, I'm still
> concerned that it effectively requires compilers to either track
> dependencies or conservatively prevent optimizations like value
> speculation and specialization based on that. Neither is a good thing
> for a compiler.
I do believe that we can find some middle ground.
> 7.9 adds requirements that dependency chains stop if the program itself
> informs the compiler about the value of something in the dependency
> chain (e.g., as shown in Figure 33). Also, if a correct program that
> does not have undefined behavior must use a particular value, this is
> also considered as "informing" the compiler about that value. For
> example:
> int arr[2];
> int* x = foo.load(mo_consume);
> if (x > arr) // implies same object/array, so x is in arr[]
> int r1 = *x; // compiler knows x == arr + 1
> The program, assuming no undefined behavior, first tells the compiler
> that x should be within arr, and then the comparison tells the compiler
> that x!=arr, so x==arr+1 must hold because there are just two elements
> in the array.
The updated version of Section 7.9 says that if undefined behavior
allows the compiler to deduce the exact pointer value, as in the
case you show above, the dependency chain is broken.
> Having these requirements is generally good, but we don't yet know how
> to specify this properly. For example, I suppose we'd need to also say
> that the compiler cannot assume to know anything about a value returned
> from an mo_consume load; otherwise, nothing prevents a compiler from
> using knowledge about the stores that the mo_consume load can read from
> (see Section 7.2).
I expect that the Linux kernel's rcu_dereference() primitives would
map to volatile memory_order_consume loads for this reason.
> Also, a compiler is still required to not do value-speculation or
> optimizations based on that. For example, this program:
>
> void op(type *p)
> {
> foo /= p->a;
> bar = p->b;
> }
> void bar()
> {
> pointer = ppp.load(mo_consume);
> op(pointer);
> }
>
> ... must not be transformed into this program, even if the compiler
> knows that global_var->a == 1:
>
> void op(type *p) { /* unchanged */}
> void bar()
> {
> pointer = ppp.load(mo_consume);
> if (pointer != global_var) {
> op(pointer);
> else // specialization for global_var
> {
> // compiler knows global_var->a==1;
> // compiler uses global_var directly, inlines, optimizes:
> bar = global_var->b;
> }
>
> The compiler could optimize out the division if pointer==global_var but
> it must not access field b directly through global_var. This would be
> pretty awkwaard; the compiler may work based on an optimized expression
> in the specialization (ie, create code that assumes global_var instead
> of pointer) but it would still have to carry around and use the
> non-optimized expression.
Exactly how valuable is this sort of optimization in real code? And
how likely is the compiler to actually be able to apply it?
(I nevertheless will take another pass through the Linux kernel looking
for global variables being added to RCU-protected linked structures.
Putting a global into an RCU-protected structure seems more likely than
is an RCU-protected pointer into a two-element array.)
> This wouldn't be as bad if it were easily constrained to code sequences
> that really need the dependencies. However, 7.9 does not effectively
> contain dependencies to only the code that really needs them, IMO.
> Unless the compiler proves otherwise, it has to assume that a load from
> a pointer carries a dependency. Proving that is often very hard because
> it requires points-to analysis; 7.9 restricts this to intra-thread
> analysis but that is still nontrivial.
> Michael Matz' had a similar concern (in terms of what it results in).
Again, I will be looking through the Linux kernel for vulnerabilities to
this sort of transformation. However, I am having a very hard time seeing
how the compiler is going to know that much about the vast majority of
the Linux-kernel use cases. The linked structures are allocated on the
heap, not in arrays or in globals.
> Given that mo_consume is useful but a very specialized feature, I
> wouldn't be optimistic that 7.9 would actually be supported by many
> compilers. The trade-off between having to track dependencies or having
> to disallow optimizations is a bad one to make. The simple way out for
> a compiler would be to just emit mo_acquire instead of mo_consume and be
> done with all -- and this might be the most practical decision overall,
> or the default general-purpose implementation. At least I haven't heard
> any compiler implementer say that they think it's obviously worth
> implementing.
I believe that we need to balance the specialization of the
memory_order_consume feature against the real-world usefulness of
the optimizations. In addition, I believe that we can rule out some
of the more ornate situations, which would allow the compiler to do
the optimizations and allow the designated set of dependencies to be
preserved.
> I also don't think 7.9 is ready for ISO standardization yet (or any of
> the other alternatives mentioned in the paper). Standardizing a feature
> that we're not sure whether it will actually be implemented is not a
> good thing to do; it's too costly for all involved parties (compiler
> writers *and* users).
Understood, but it seems premature to give up on 7.9. I believe that we
are actually quite close.
> IMO, the approach outlined in Section 7.7 is still the most promising
> contender in the long run. It avoid the perhaps more pervasive changes
> that a type-system-based approach as the one in Section 7.2 might result
> in, yet still informs the compiler where dependencies are actually used
> and which chain of expressions would be involved in that. Tracking is
> probably simplified, as dependencies are never open-ended and
> potentially leaking into various other regions of code. It seems easier
> to specify in a standard because we just need the programmer to annotate
> the intent and the rest is compiler QoI. It would require users to
> annotate their use of dependencies, but they don't need to follow
> further rules; performance tuning of the code so it actually makes use
> of dependencies is mostly a compiler QoI thing, and if the compiler
> can't maintain a dependency, it can issue warnings and thus make the
> tuning interactive for the user.
>
> Of course, finding a good way to reference the source of the dependency
> in the API (see the paper for some of the sub-issues) will be a
> challenge. But, currently, I'm more optimistic we can find a useful
> solution for this than finding a standardizable version of 7.9.
As discussed before, I have no objection to exploring the other options,
including the one in Section 7.7, but we really do need to be able to
handle the common-case dependency chain without overdosing on syntactic
sugar.
Thanx, Paul
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/