On 20/07/20 14:17, pet...@infradead.org wrote:
> On Mon, Jul 20, 2020 at 01:20:26PM +0100, Valentin Schneider wrote:
>> On 20/07/20 12:26, pet...@infradead.org wrote:
>
>> > +  /*
>> > +   * We must re-load prev->state in case ttwu_remote() changed it
>> > +   * before we acquired rq->lock.
>> > +   */
>> > +  tmp_state = prev->state;
>> > +  if (unlikely(prev_state != tmp_state)) {
>> > +          /*
>> > +           * ptrace_{,un}freeze_traced() think it is cool to change
>> > +           * ->state around behind our backs between TASK_TRACED and
>> > +           *  __TASK_TRACED.
>> > +           *
>> > +           * This is safe because this, as well as any __TASK_TRACED
>> > +           * wakeups are under siglock.
>> > +           *
>> > +           * For any other case, a changed prev_state must be to
>> > +           * TASK_RUNNING, such that when it blocks, the load has
>> > +           * happened before the smp_mb().
>> > +           *
>> > +           * Also see the comment with deactivate_task().
>> > +           */
>> > +          SCHED_WARN_ON(tmp_state && (prev_state & __TASK_TRACED &&
>> > +                                     !(tmp_state & __TASK_TRACED)));
>> > +
>>
>> IIUC if the state changed and isn't TASK_RUNNING it *has* to have
>> __TASK_TRACED, so can't that be
>>
>>   SCHED_WARN_ON(tmp_state && !(tmp_state & __TASK_TRACED));
>
> Suppose task->state == TASK_UNINTERRUPTIBLE, and task != current, and
> then someone goes and does task->state = __TASK_TRACED.
>
> That is, your statement is correct given the current code, but we also
> want to verify no new code comes along and does something 'creative'.
>

That is what I was trying to go for; AFAICT your approach only warns if
__TASK_TRACED gets removed between the two loads (IOW it was already
there). The way I was seeing it is:

- We only get here if prev_state != tmp_state; IOW we know we raced with
  something
- if (tmp_state), then it wasn't with ttwu_remote()
- thus it must only be with ptrace shenanigans, IOW __TASK_TRACED must be
  there.

Now, what I suggested still doesn't detect what you pointed out, or some
crazier thing that sets __TASK_TRACED *and* some other stuff. IIUC the
ptrace transformation does TASK_TRACED -> __TASK_TRACED, so we could have
it as:

  /* TODO: name me */
  #define foobar TASK_TRACED ^ __TASK_TRACED

  ...

  /* not TASK_RUNNING; check against allowed transformations
  SCHED_WARN_ON(tmp_state && ((prev_state ^ tmp_state) & ~foobar));


That said...

> Or is the heat getting to me?

... that may apply to me as well :-)

Reply via email to