-
-
Notifications
You must be signed in to change notification settings - Fork 322
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
Unclear intent regarding optimizations for some profiles #1477
Comments
Thanks for opening this issue to figure out how releases should be configured.
Sorry for the mess. As far as I remember, the LTO setting was to slow to compile so eventually I commented it out, leaving it to the standard that should be fine. For optimizing build times, In theory, the choice of build profiles should be made so that nobody is surprised and gets served well.
Yes, please feel free to clean that up and remove commented-out parts with impunity. If I remember correctly, optimizing all dependencies unconditionally felt too slow at the time, so I settled for a subset which seems to still be the right choice in retrospect. However, it's too long to really know for sure it has the desired performance impact, even though the current setting doesn't seem to make it too bad. Some additional thoughtsThe real question is: what should happen if For instance, when building for GitHub releases, one might want to use all optimizations aggressively and strip symbols. When profiling, one might want additional debug assertions but also release optimizations (but nothing that increases build times crazily). |
I've opened #1495, but as alluded to there and detailed below, I think significant changes might be wanted before considering that to address this issue in the best way.
How slow is too slow? With I tested three configurations, with results and some further explanation in this gist:
Please let me know if you would like more of the information from that gist to be here (or perhaps in a new discussion post or somewhere).
Most of the slowdown seems due to less parallelism. I am inferring this from the total processor time versus total elapsed time. The bulk of the effect may be due to To be clear, I did not benchmark the code itself, I only tested how long it took to build it. I suspect that I am wondering if
In the same way that one of the profiles So I think the least surprising way to do it might be to go with the original idea of having
This could be revisited along with the Another option would be to not use any optimized dependencies in the
When users install that way, do they want that additional information? If so, then perhaps the debug info should be kept in even in the GitHub releases. I think I would usually expect the practical differences between
Knowing that various build workflows such as those of ripgrep and gitoxide strip binaries causes me, as a result of that, to be aware that another difference could be the absence of debug info in the binaries. But this is not something I would have expected before, or that I would expect in general or across languages. How great is the benefit of stripping the symbols? And is there any way to provide them in separate files instead, as downstream distributions sometimes do? Is that worth doing? |
I arrived here late, after merging the corresponding PR, but want to pick up on a few parts nonetheless.
To my mind, a good way to go about this is to crank up everything to 11 for GitHub releases as build-speed doesn't really matter there or is well within limits. To my mind, if builds take a little less than an hour, it's still good enough. Important operations, like cloning and checkouts, could then be compared from time to time to see if GitHub builds are measurably faster than local
That's very interesting and I wan't aware of that.
Indeed, using
I also think that having these optimizations in
Reducing the binary size probably is the primary driver for stripping symbols, and as panics are rare or never happen, they might effectively be useless. Those who want symbols for profiling or debugging probably build their own binary (or can be expected to install with So probably these binaries should still be stripped based on that valuation. Personally, I wouldn't provide debug symbols either as it's even more files on the downloads page then, and people probably are faster just recompiling with |
Sorry about the confusion there. Providing them as separate files (and obtained by separate downloads) is, in hindsight, the most salient element of the example I linked to, but actually I did not mean to advocate that Instead, I was thinking that if the I think that at least if I would still not ship the symbols as separate downloads, and likely not even include them in the existing archives either. Though there might be an argument for doing the latter, since while it would increase the size of the downloads, it would not increase the amount of memory the binary image takes up when the program is run. This it would not have even a slight negative effect on CPU cache performance, paging, or anything else affected by executable size. Having the symbols is useful in case of panics even outside of profiling, but it is not obvious to me why users would want this when installing from source with (Even if so, I should probably benchmark runtime performance before doing anything that would either commit gitoxide to maintaining a separate profile or even that users might wrongly assume commits gitoxide to doing so.) |
It's really just me who would want to have panics and Thus far, GitHub releases were optimized for download size, so maybe, to unify this, we just don't strip the binaries as a start. Those who don't want them can then strip them themselves. Finally, if there is a benefit in having minimally-sized archived, these could be created separately, for instance for use in GitHub actions that would need to startup as fast as possible. |
Current behavior 😯
There are a few conditions related to build profiles for
gitoxide
defined in the top-levelCargo.toml
that seem like they can be addressed when—as planned in the comment discussion in #1475—splitting therelease
profile into separaterelease
andprof
(orprofile
orprofiling
) profiles.#lto = "fat"
.1. Debug symbols
Currently the
release
profile leaves debug symbols in because they are useful for profiling, but this creates the need to strip them in a separate step when they are not wanted, such as when building binaries to publish in releases on GitHub.This is as discussed in comments in #1475, with a suggested plan in #1475 (comment).
2. Unclear intent for LTO
It is not clear if link-time optimization, at higher than the default very limited level, should be done. It's possible this might even differ between
release
andprof
profiles. (This is the main sub-issue that may need examination and discussion, and the main reason I opened this issue.)Early on, in 07859cb, various performance-related configuration was set for the
release
profile. This included the most aggressive link-time optimization available—short of LTO across different languages—which was still in place about a year ago:/~https://github.com/Byron/gitoxide/blob/71efcbba112376b4acaf37d662cdb38d369462be/Cargo.toml#L185
Then in bd32e39 (#893) it was commented out, with the effect of changing the
lto
setting fromfat
to the default offalse
, which somewhat unintuitively still performs link-time optimization, just only within the crate itself./~https://github.com/Byron/gitoxide/blob/1d37bf6a773d56eea9003aa626ced413e8e0eaa3/Cargo.toml#L209
For the last year, it's been that way. Since the
gitoxide
project is made up of many crates, which thegitoxide
binary crate itself uses, I suspect this does not leverage most of the benefit of LTO. That is, the word "false
" may be intuitively accurate in this case.Perhaps benchmarking was done and found LTO not to be helpful for
gix
andein
. (In other software that uses anygitoxide
crates, that software would of course set whatever LTO-related configuration is wanted.) But the commit and PR do not state a reason for the change, and they are otherwise not conceptually related to LTO or optimization. If the goal is to decrease build times, thenthin
could be used, which is documented to be almost as good asfat
while being faster to perform.Another possibility is that LTO was turned off because the resulting optimizations decreased the detail available when profiling or made the results hard to interpret. If that was the reason, then I think that, when splitting
release
intorelease
andprof
, this would be resolved by setting different values oflto
in the two different profiles.release
could havefat
orthin
, whileprof
could stick with the default offalse
.3. Old plan for optimizations even in debug builds?
This would not affect the
release
orprof
configuration, but I suspect it could be clarified at the same time.a5b1cd3 added:
/~https://github.com/Byron/gitoxide/blob/1d37bf6a773d56eea9003aa626ced413e8e0eaa3/Cargo.toml#L217-L219
It seems intuitively odd to compile dependencies with a higher optimization level than the code that is using them, especially since this may make stack traces less rich. On the other hand, maybe it is to make test runs faster.
We are already doing it, with the even more aggressive level of
3
, in thedev
profile for particular dependencies, most of which are evengix-*
crates. This was done in 4a948d0 (after having been attempted in #190 per a22c95b and 85f1a48) with the explicit purpose of making some tests faster. I am not suggesting that this be changed, since its purpose is clear from the commit message and running tests faster has a number of benefits./~https://github.com/Byron/gitoxide/blob/1d37bf6a773d56eea9003aa626ced413e8e0eaa3/Cargo.toml#L196-L205
But one would expect that and the commented-out
opt-level = 2
for*
to be right next to each other. Instead, therelease
profile is written in between them. Although the commit message in a5b1cd3 makes clear that this was not why that commented-out code was introduced, I wonder if part of the reason it has been preserved as a comment for so long, rather than eventually being uncommented or removed, could be confusion with this other code in therelease
configuration:/~https://github.com/Byron/gitoxide/blob/1d37bf6a773d56eea9003aa626ced413e8e0eaa3/Cargo.toml#L215
To be clear, I'm not at all advocating changing that, which was introduced in add2e3e (#2) and makes it so dependencies that are only used when building and not used when running the code are not needlessly optimized.
When adding a
prof
profile, it would go right next to therelease
profile, and having both of them split the two sections that customizedev
profile configuration (doing so even in a related way) would exacerbate the current unclear situation. I'd like to remove whatever is not planned or adjust comments to reflect the goal of code that is wanted for the future but not yet, and then if applicable put the most related things together.Expected behavior 🤔
See above. It seemed clearer to include this for each subsection.
(I considered opening multiple issues or opening no issue for anything here and just bringing it all up in comments, but I figured this might still be best, even though it goes a bit against the format, because there is at least one other issue related to the comments there, #1478, that should be covered separately and should not be confused with any of this.)
Git behavior
I have not researched this. Some choices of optimization level or the treatment of debug symbols might be illuminating. But the tooling and language are different, and for the upstream Git project, I believe there are no official binary releases. So it seems largely inapplicable.
Steps to reproduce 🕹
Since this is a clarity and release-engineering issue opened for tracking and feedback about things I think are mostly known, reproduction consists mainly of examining the code shown, linked, or otherwise discussed above and in the related portion of the #1475 comment discussion.
One practical note, though: On some platforms, the
file
command will reveal debug symbols, including text like "not stripped" in its output when they are present. But it seems thefile
command on macOS may not do this.The text was updated successfully, but these errors were encountered: