Page MenuHomeFreeBSD

pipe: try to skip locking the pipe if a non-blocking fd is used
ClosedPublic

Authored by mjg on Aug 8 2022, 7:20 PM.
Tags
None
Referenced Files
Unknown Object (File)
Tue, Jan 21, 6:15 AM
Unknown Object (File)
Mon, Jan 20, 2:05 AM
Unknown Object (File)
Sat, Jan 18, 3:29 AM
Unknown Object (File)
Thu, Jan 16, 4:17 PM
Unknown Object (File)
Thu, Jan 16, 4:08 PM
Unknown Object (File)
Thu, Jan 16, 4:03 PM
Unknown Object (File)
Thu, Jan 16, 3:58 PM
Unknown Object (File)
Thu, Jan 16, 1:57 PM
Subscribers

Details

Summary

make -j 104 tinderbox has over half of the calls to read find an empty pipe. As the pipe is shared, this contends a lot.

commit d7cfa63d1de31e176e50562cdb57efa596792d93 (HEAD, mjgzoo/pipenb)
Author: Mateusz Guzik <mjg@FreeBSD.org>
Date:   Mon Aug 8 18:02:15 2022 +0000

    pipe: try to skip locking the pipe if a non-blocking fd is used
    
    Reviewed by:
    Differential Revision:

commit a5ee622427f86017939bc9ef47e5a7c804072825
Author: Mateusz Guzik <mjg@FreeBSD.org>
Date:   Mon Aug 8 19:14:09 2022 +0000

    mac: cheaper check for mac_pipe_check_read
    
    Reviewed by:
    Differential Revision:

Diff Detail

Lint
Lint Skipped
Unit
Tests Skipped

Event Timeline

mjg requested review of this revision.Aug 8 2022, 7:20 PM
mjg created this revision.
sys/kern/sys_pipe.c
732

(fp->f_flag & FNONBLOCK) != 0

733

If this check is worth it, why it is done under FNONBLOCK/mac predicates? It should be performed unconditionally, it seems.

735

!= 0

sys/security/mac/mac_framework.h
270

Why is it defined as 0 and not as false?

sys/kern/sys_pipe.c
726
730

This comment doesn't make much sense to someone just reading the code: what is being avoided? Doesn't the lockless check avoid the contention?

I'd suggest explaining it at a higher level: some programs use pipes to distribute jobs to workers, so polling can induce lock contention if we don't check the buffer here.

735

In the locked path, the reader might clear PIPE_WANTW. Why is it not necessary to handle that here?

737

Please use atomic_load_* for these unlocked loads.

mjg marked 4 inline comments as done.
mjg added inline comments.
sys/kern/sys_pipe.c
735

it should be an invariant that if the pipe is empty someone can write to it

737

note atomic access on only one end is not really a full solution.

perhaps it is time to introduce atomic_int_t and other types.

mjg marked an inline comment as done.Aug 11 2022, 10:35 PM
mjg added inline comments.
sys/kern/sys_pipe.c
733

there is some processing done right now even when literally nothing got read. I do think it is questionable, but I don't want to remove it in this patch. I consider it fine for the non-blocking case which just takes a peek.

sys/kern/sys_pipe.c
733

also it's this check is not an optimization, it is needed to return the right error code -- 0 instead of EAGAIN

sys/kern/sys_pipe.c
730

Missing something after "over".

737

Indeed, atomic_load does not solve a problem per se, it's just a hint to someone reading the code.

To make KCSAN avoid reporting false positives, the stores need to be atomic-qualified too.

737

This should be atomic_load too.

  • fix 32-bit build
  • tidy up comment
  • atomic for pipe state
sys/kern/sys_pipe.c
744

We could use u_int for cnt in struct pipemapping. pipe_build_write_buffer() and pipe_clone_write_buffer() use an int already. Then there's no need for this ugly ifdef.

This revision is now accepted and ready to land.Aug 17 2022, 1:42 PM
This revision was automatically updated to reflect the committed changes.