-
Notifications
You must be signed in to change notification settings - Fork 18.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
pkg/mount improvements #36091
pkg/mount improvements #36091
Conversation
41cf809
to
2b8bd69
Compare
8182b38
to
1a83fd7
Compare
OK this is no longer WIP |
ping @tonistiigi @cpuguy83 needs some review love 😽 |
Some measurements relevant to this PR, comparing |
@@ -15,7 +15,7 @@ import ( | |||
|
|||
// Parse /proc/self/mountinfo because comparing Dev and ino does not work from | |||
// bind mounts. | |||
func parseMountTable() ([]*Info, error) { | |||
func parseMountTable(filter FilterFunc) ([]*Info, error) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we can get rid of the NoFilter()
option by changing this to ...FilterFunc
. This also allows for multiple filters... for whatever reason.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
FWIW, just passing nil
instead of NoFilter()
would be an option as well (given that we currently don't have a use for multiple filters to be passed)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, my original intention was to use nil
and it didn't work (a function pointer can not be nil
, if I remember correctly it's because of a type mismatch).
...FilterFunc
makes sense from the first glance.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
a function pointer can not be nil, if I remember correctly
hm, I think if you check for nil
and don't call it, it would work https://play.golang.org/p/5Q_9hk-Ixfc
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, that was exactly my plan but for some reason it did not work (guess my C background played a trick on me here). Thanks for pointing it out. Redid it with nil
, patchset updated
3147074
to
6c5f7cd
Compare
Rebased to current git tip, added one more patch due to #36107 being merged (which uses a function I removed) |
@kolyshkin are the z and powerpc failures related ? |
Same test case failed on ppc and z, perhaps some other recent moby changes affected it? 00:26:42 FAIL: docker_cli_swarm_test.go:341: DockerSwarmSuite.TestSwarmContainerEndpointOptions |
/cc @tophj-ibm |
P/Z issues were a dockerhub issue where the manifest list hadn't been updated for the multi-arch versions of busybox:glibc yet.
Also looks like a bug where the |
You mean Using |
} | ||
|
||
// Mounted determines if a specified mountpoint has been mounted. | ||
// On Linux it looks at /proc/self/mountinfo. | ||
func Mounted(mountpoint string) (bool, error) { | ||
entries, err := parseMountTable() | ||
entries, err := GetMounts(SingleEntryFilter(mountpoint)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps better/cleaner to use the non-exported function here
parseMountTable(SingleEntryFilter(mountpoint))
Ok, what about:
We can un-export the |
Other than that, your proposal looks good, thanks for the input. By the way, I had |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm ok with the design as is.
LGTM
This needs an update, hold on |
Functions `GetMounts()` and `parseMountTable()` return all the entries as read and parsed from /proc/self/mountinfo. In many cases the caller is only interested only one or a few entries, not all of them. One good example is `Mounted()` function, which looks for a specific entry only. Another example is `RecursiveUnmount()` which is only interested in mount under a specific path. This commit adds `filter` argument to `GetMounts()` to implement two things: 1. filter out entries a caller is not interested in 2. stop processing if a caller is found what it wanted `nil` can be passed to get a backward-compatible behavior, i.e. return all the entries. A few filters are implemented: - `PrefixFilter`: filters out all entries not under `prefix` - `SingleEntryFilter`: looks for a specific entry Finally, `Mounted()` is modified to use `SingleEntryFilter()`, and `RecursiveUnmount()` is using `PrefixFilter()`. Unit tests are added to check filters are working. [v2: ditch NoFilter, use nil] [v3: ditch GetMountsFiltered()] [v4: add unit test for filters] [v5: switch to gotestyourself] Signed-off-by: Kir Kolyshkin <[email protected]>
Use mount.SingleEntryFilter as we're only interested in a single entry. Test case data of TestShouldUnmountRoot is modified accordingly, as from now on: 1. `info` can't be nil; 2. the mountpoint check is not performed (as SingleEntryFilter guarantees it to be equal to daemon.root). Signed-off-by: Kir Kolyshkin <[email protected]>
The flow of getSourceMount was: 1 get all entries from /proc/self/mountinfo 2 do a linear search for the `source` directory 3 if found, return its data 4 get the parent directory of `source`, goto 2 The repeated linear search through the whole mountinfo (which can have thousands of records) is inefficient. Instead, let's just 1 collect all the relevant records (only those mount points that can be a parent of `source`) 2 find the record with the longest mountpath, return its data This was tested manually with something like ```go func TestGetSourceMount(t *testing.T) { mnt, flags, err := getSourceMount("/sys/devices/msr/") assert.NoError(t, err) t.Logf("mnt: %v, flags: %v", mnt, flags) } ``` ...but it relies on having a specific mount points on the system being used for testing. [v2: add unit tests for ParentsFilter] Signed-off-by: Kir Kolyshkin <[email protected]>
The mountinfo parser implemented via `fmt.Sscanf()` is slower than the one using `strings.Split()` and `strconv.Atoi()`. This rewrite helps to speed it up to a factor of 8x, here is a result from go bench: > BenchmarkParsingScanf-4 300 22294112 ns/op > BenchmarkParsingSplit-4 3000 2780703 ns/op I tried other approaches, such as using `fmt.Sscanf()` for the first three (integer) fields and `strings.Split()` for the rest, but it slows things down considerably: > BenchmarkParsingMixed-4 1000 8827058 ns/op Note the old code uses `fmt.Sscanf`, when a linear search for '-' field, when a split for the last 3 fields. The new code relies on a single split. I have also added more comments to aid in future development. Finally, the test data is fixed to now have white space before the first field. Signed-off-by: Kir Kolyshkin <[email protected]>
Now, every Unmount() call takes a burden to parse the whole nine yards of /proc/self/mountinfo to figure out whether the given mount point is mounted or not (and returns an error in case parsing fails somehow). Instead, let's just call umount() and ignore EINVAL, which results in the same behavior, but much better performance. Note that EINVAL is returned from umount(2) not only in the case when `target` is not mounted, but also for invalid flags. As the flags are hardcoded here, it can't be the case. Signed-off-by: Kir Kolyshkin <[email protected]>
There is no need to parse mount table and iterate through the list of mounts, and then call Unmount() which again parses the mount table and iterates through the list of mounts. It is totally OK to call Unmount() unconditionally. Signed-off-by: Kir Kolyshkin <[email protected]>
This is not for the sake of test to run faster of course; this is to simplify the code as well as have some more testing for mount.SingleEntryFilter(). Signed-off-by: Kir Kolyshkin <[email protected]>
Updated; this is basically - "github.com/stretchr/testify/assert"
+ "github.com/gotestyourself/gotestyourself/assert" and - assert.NoError(t, err)
+ assert.NilError(t, err) spread over a few commits from the set. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
still LGTM, thanks for updating those
The mountinfo parser implemented via `fmt.Sscanf()` is slower than the one using `strings.Split()` and `strconv.Atoi()`. This rewrite helps to speed it up to a factor of 8x, here is a result from `go bench`: > BenchmarkParsingScanf-4 300 22294112 ns/op > BenchmarkParsingSplit-4 3000 2780703 ns/op I tried other approaches, such as using `fmt.Sscanf()` for the first three (integer) fields and `strings.Split()` for the rest, but it slows things down considerably: > BenchmarkParsingMixed-4 1000 8827058 ns/op Note the old code uses `fmt.Sscanf` first, then a linear search for the '-' field, then a split for the last 3 fields. The new code relies on a single split. One other thing is, the new code is more future proof as it skips extra optional fields before the separator (currently there are none). I have also added more comments to aid in future development. Finally, the test data is fixed to not have white space before the first field. Based on a similar change in Moby, moby/moby#36091 [v2: remove no-op break statement to silence staticcheck] Signed-off-by: Kir Kolyshkin <[email protected]>
This is a set of improvement to speed up pkg/mount.
Here is a short description; for more details see individual commit messages.
Implement filters for mountinfo parsing. In case a user is not interested in all the entries but a selected few, a filter can be applied during parsing.
Rewrite Linux mountinfo parser to use
Strings.Split()
(rather thanfmt.Sscanf()
), speeding it up 8x.volume/local, mount.Unmount(): no need to parse mountinfo before calling unmount.
miscellaneous(moved out into separate PRs: pkg/mount: use sort.Slice #36506 pkg/mount unit tests: skip some test under non-root #36505 devmapper cleanup improvements #36307)