-
-
Notifications
You must be signed in to change notification settings - Fork 968
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
Feature: zstandard compression for network games #8845
Conversation
32a4c85
to
2441a28
Compare
I've not combed through this yet, but I've got no objections to what I see so far. It seems well thought out. However, as well speed and size, there is a third axis to think about which is compatibility of persistent save files. |
A very solid point. And looking back, I think indeed that |
After some brain-cycles, I indeed ended up changing "smallest" and "fastest" for "persistent" and "temporary", as that is much more what I wanted to say anyway. I noticed that while documented that "smallest" is not really the smallest we can do, something wasn't fully right yet. I think this change makes a lot more sense, as that is what we want: we always want the smallest and fastest compression we can get, but in a balance with each other. So that is not the axis we are configuring. Indeed, the thing we want to indicate: are you allowed to use methods that are a bit less widespread and more experimental. I think Tnx @JGRennison , very valuable comment :) |
dc33160
to
8f86ca8
Compare
Few things I noticed this pr is lacking compared to #8805. Though most important ones are easy to fix I guess.
|
This is an alternative of what I mention at limitation. I tried to scope this PR to keep it easily reviewable; such extensions can be made in various of ways, but that shouldn't be part of this PR. For example, I considered making
Yes. The LZMA is of course exactly the point; clients that do not have zstd should use LZMA instead. Other formats like "none" are only bad for the bandwidth, but otherwise surprisingly fast. The only one in the list that would be somewhat bad is LZO, but not terrible bad. In general, all compressions we have set are not "bad". Just zstd is better.
As mentioned by limitations. I do not think this is a feature that is required for the average server admin. Expert server admins can compile their own server (and most do anyway), so I do not think it is needed for this to be a setting. But if there is really a request for this, future PRs can extend on this. For example, I considered creating
This is already the case before this PR, so I read this more as a feature request. And if I translate it correctly, you basically would like that you can set the savegame format in the console command. I think this is a reasonable request, but for another PR of course. |
8f86ca8
to
2c6bec0
Compare
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.
Minor amount of thoughts from scrolling through
SGF_NO_THREAD = 0, ///< Do not use threads. | ||
SGF_THREADED = 1 << 0, ///< Save threaded. | ||
SGF_PERSISTENT = 0, ///< Savegame should persistent through time and different clients. | ||
SGF_TEMPORARY = 1 << 1, ///< Savegame will only be used temporary or locally. |
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.
the fact that the enum contains duplicate values with entirely different meanings is a bit odd
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.
They are toggles. The idea is that you either useNO_THREAD
or THREADED
. We could do away with the 0-value ones, but than it is harder to see how a savegame is made and easier to think either one is on-by-default.
As I wrote in the PR description, I use this to avoid calls with true, true
, which are not very descriptive and easy to mess up. But I am fine reverting back to the boolean approach; or of course if you have any other suggestions, I am all ears :)
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.
Alternatively, I can do 0 << 0
and 0 << 1
to make it more clear they are toggles? Not sure that helps :D
98d2959
to
a97a5f6
Compare
src/saveload/saveload.cpp
Outdated
@@ -2422,6 +2424,7 @@ struct ZSTDSaveFilter : SaveFilter { | |||
|
|||
/** The format for a reader/writer type of a savegame */ | |||
struct SaveLoadFormat { | |||
uint8 index; ///< Index of this entry. |
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.
Comment should say that the index has to be unique, unless you actually intended to have multiple formats at one index in the bitmask.
Some compile time assert that index is unique would probably be nice, but I'm not sure it's possible. (Maybe with constexpr/consteval, but not sure).
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 would love some compile time asserts, but I wouldn't even know where to start :D And consteval
is C++20, not?
Use zstd for network savegames, and lzma for all other. This should speed up joining of multiplayer games, especially on large maps.
…ormats This means that if a client doesn't have all formats, it can still join a multiplayer server. The client informs the server of his available formats, after which the server finds the best format given these constraints. If no match is found, the client is disconnected with an error.
a97a5f6
to
ac3deee
Compare
Original author indicates that |
Every time I look at this PR, I don't like what I did myself. Mainly the way I solved flags is just shit. But also the whole way this is wired in is not in a way I like. I also miss a solid way of testing this .. which just annoys me. So I could leave this open as inspiration for others, but honestly .. not sure that is useful. So I am going to close it up .. maybe I revisit this some day. But I doubt it will be this year :D |
Closes #8805.
Closes #8773.
Motivation / Problem
#8773 introduces zstandard, with good argumentation why it should be default for network games. It does, however, only introduce zstandard. It does not enable it by default, nor for network games.
#8805 introduces a handshake between server and client to find the common savegame formats. Alternatively, JGRPP introduces similar functionality in a different way: JGRennison/OpenTTD-patches@c2ae758
Both have their merits, but both also have their issues. #8805 is on one side of the spectrum, making it fully configurable and tunable. JGRPP on the other hand, minimizes it and makes zstd a special case. I did not like either approach (but highly appreciate both views on the matter): to me, it is more than fine to have a clear opinion on the best settings, but I also like to think a bit about the future, in case we ever add more compression libraries; it is likely that in the next few years even better and faster compression algorithms will be found, looking at amount of new methods I already see around us.
Finally, I had the realisation that it is not network games or zstd that makes this special. But it is the fact that sometimes we want to prioritize on savegame filesize, and sometimes on savegame creation time. Later I adjusted this (tnx the conversation below), that it is more about temporary vs persistent storage: one can be a very new compression method, but the persistent storage should be able to survive for years to come.
With this in hand, I present you this alternative. It combines all the above work in a single Pull Request:
Description
After inspiration on the work in JGRPP, I did some rewriting. Instead of passing
threaded
everywhere, and with this PR that would also includetemporary / persistent
andbitmask
, they are now stored in_sl
instead. Additionally, JGRPP uses flags, which is a good idea. But I took a slightly different implementation choice here, and basically make them bitwise switches: eitherSGF_TEMPORARY
orSGF_PERSISTENT
and eitherSGF_NO_THREAD
orSGF_THREADED
. This makes calling save functions a lot easier and more descriptive, instead of a list oftrue, false
, etc.I integrated several fixes JGRPP already made for the zstandard implementation from #8773, fixes some coding style, made
compression_level
signed, and other minor issues, as while I was reading the code why not integrate the review. So it is "mostly" #8773, with some changes.Limitations
savegame_format
in your configuration, it overwrites auto-detection completely. So if a server would be to do that, the handshake is fully ignored and the savegame is made in what-ever format is described. Feature: Allow client and server to negotiate on compression to use for the savegame #8805 in contrast allows multiple formats to be set, which are all tried for network games. I think this is unneeded and only complicates the code. If a server admin really does not want to support a compression or experiment with different compression levels, he might as well recompile OpenTTD with these changes. Personally, I am not a big fan of these endless amount of settings and configuration options, and I think it is fine we have an opinion about what is best. However, if there is truly a demand for this, an additional PR can be created to add support for this.Checklist for review
Some things are not automated, and forgotten often. This list is a reminder for the reviewers.