Page MenuHomeFreeBSD

fread: improve performance for unbuffered reads
ClosedPublic

Authored by pfg on May 29 2021, 9:46 PM.
Tags
None
Referenced Files
Unknown Object (File)
Dec 16 2024, 1:52 PM
Unknown Object (File)
Dec 13 2024, 1:30 AM
Unknown Object (File)
Nov 21 2024, 11:37 AM
Unknown Object (File)
Nov 15 2024, 6:02 PM
Unknown Object (File)
Oct 30 2024, 5:47 AM
Unknown Object (File)
Oct 21 2024, 11:41 AM
Unknown Object (File)
Oct 19 2024, 5:57 PM
Unknown Object (File)
Oct 18 2024, 1:43 PM
Subscribers

Details

Summary

We can use the buffer passed to fread(3) directly in the FILE *.
The buffer needs to be reset before each call to __srefill().
This preserves the expected behavior in all cases.

THe change was found originally in OpenBSD and later adopted by NetBSD.

Obtained from: OpenBSD (CVS 1.18)

Diff Detail

Repository
rS FreeBSD src repository - subversion
Lint
Lint Passed
Unit
No Test Coverage
Build Status
Buildable 39577
Build 36466: arc lint + arc unit

Event Timeline

pfg requested review of this revision.May 29 2021, 9:46 PM

In principle, the optimization is perhaps useful for unbuffered consumers.

The locking in your patch is incompatible with the FreeBSD implementation. Test the patch in multithreaded program.

lib/libc/stdio/fread.c
128

This cannot be right.

In D30548#686024, @kib wrote:

In principle, the optimization is perhaps useful for unbuffered consumers.

The locking in your patch is incompatible with the FreeBSD implementation. Test the patch in multithreaded program.

Bah .. you are right, I am quite rusty on FreeBSD and I just copy-pasted. This shouldn't have compiled.

In D30548#686034, @pfg wrote:
In D30548#686024, @kib wrote:

In principle, the optimization is perhaps useful for unbuffered consumers.

The locking in your patch is incompatible with the FreeBSD implementation. Test the patch in multithreaded program.

Bah .. you are right, I am quite rusty on FreeBSD and I just copy-pasted. This shouldn't have compiled.

It should compile fine, but it cannot work.

What you need to do is to remove the wrong unlock line.

In D30548#686179, @kib wrote:
In D30548#686034, @pfg wrote:
In D30548#686024, @kib wrote:

In principle, the optimization is perhaps useful for unbuffered consumers.

The locking in your patch is incompatible with the FreeBSD implementation. Test the patch in multithreaded program.

Bah .. you are right, I am quite rusty on FreeBSD and I just copy-pasted. This shouldn't have compiled.

It should compile fine, but it cannot work.

What you need to do is to remove the wrong unlock line.

Remove or replace?

The change seems to have made an important difference in other BSDs but I'd honestly love to have someone that knows more about locking take over ;).

lib/libc/stdio/fread.c
128

Should it just be?

FLOCKFILE_CANCELSAFE(fp);
lib/libc/stdio/fread.c
128

I meant

FUNLOCKFILE_CANCELSAFE();

but I guess one has to lock before unlocking ?

lib/libc/stdio/fread.c
128

Look at fread() and __fread(). You patched __fread(), which assumes that the locks are taken externally and unlocked externally as well, when needed.

In other words, just delete (any) unlock you added.

Duh ! I see now.

Will fix soon. and run some more testing in my system. Thanks!

Remove deprecated unlocking, pointed out by kib

pfg marked 2 inline comments as done.May 31 2021, 1:52 AM

This looks fine to me, but I did not tested.

This revision is now accepted and ready to land.May 31 2021, 7:52 PM

Commit log nit: s/THe/The/

This revision was automatically updated to reflect the committed changes.