-
-
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
Template DoCommand and friends #9725
Conversation
f1027b2
to
5540a67
Compare
4102e2d
to
f9a87d6
Compare
src/aircraft_cmd.cpp
Outdated
@@ -1275,7 +1276,7 @@ void HandleMissingAircraftOrders(Aircraft *v) | |||
const Station *st = GetTargetAirportIfValid(v); | |||
if (st == nullptr) { | |||
Backup<CompanyID> cur_company(_current_company, v->owner, FILE_LINE); | |||
CommandCost ret = DoCommand(DC_EXEC, CMD_SEND_VEHICLE_TO_DEPOT, v->tile, v->index, 0); | |||
CommandCost ret = DoCommand<CMD_SEND_VEHICLE_TO_DEPOT>::Do(DC_EXEC, v->tile, v->index, 0, {}); |
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.
Not sure if anything can be done about it since there's still a Command
struct, though DoCommand::Do
sounds double. Command::Do
would look and sound better. Although that change might also affect naming things like DC_EXEC.
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.
Yeah, there can only be one Command
struct(tm). OTOH the current Command
struct is used in one single place and could be easily renamed.
I wouldn't mind a different name, but I don't know what everybody else thinks about it.
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 like the idea of Command::Do
, feels neat.
On the subject, I do think Command::Post
would be a confusing name, but I struggle with finding a good name. From what I remember, Command::Do
can be executed inside a command, but Command::Post
cannot; the latter should also always be the one to execute when starting a command.
So something like Command::Execute
? Or Command::Safe
? It is difficult .. but for sure "Post" confuses me, as you think it is the "post" (vs "pre") handling of the command; which it is not (I think :D) :P
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.
Post
was just something I could come up with that whoever came up with the initial P
might have though of.
If we were to rename it, I like Execute
more then Safe
.
Famous last words: I might have figured MingW64 out. For whatever reason, it fails on linking when casting a function pointer to its own type, but works if I insert a detour to some other function type in between. (Fingers crossed all other compilers stay happy with it). And, to surface one comment from Rb: Should it be |
Renamed to |
|
||
if (p1 >= MapSize()) return CMD_ERROR; | ||
TileIndex end_tile = p1; | ||
if (end_tile >= MapSize()) return CMD_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.
Isn't the tile checking in the command code (DoCommandHelper::Do) preventing a tile from being larger than the map size? Also holds for CmdRemoveLongRoad
.
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.
DoCommandHelper::Do
only checks the first parameter (i.e. what used to be tile
), but not anything that used to be in p1/p2, as that is the current behaviour. I thought about checking all TileIndex parameters, but I'd have to verify if any command does a different TileIndex validation.
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.
Okay. Lets keep those things out of this PR then ;)
I've also had other outlandish(?) ideas. We are passing Vehicle/Town/...IDs and almost always T obj = T::GetIfValid(id); if (obj == nullptr) return CMD_ERROR
, so I was thinking about putting that resolving to object (and validation whether nullptr
is allowed) into some "annotation" (with C++ templating magic), That way most of the common code for resolving and checking (also with enums) can be moved out of the Cmd* functions, making it less likely that mistakes are made with the checks for data that comes from the network.
Though that is far beyond the scope of this PR. However, validating of TileIndex
values falls somewhere in the same category.
…nd function calls.
…templated command traits. This is using a non-intrusive type-traits like templated system, which allows compile-time validation that the command table and the command enum match up.
… cmd/callback combinations. If the arguments of the callback proc don't match with the command parameters, we can't do the proper command execution anyway. As such, don't even generate an unpack function in the first place, saving a bit of unnecessary code bloat. Validate on receive that the cmd/callback combination is supported, rejecting clients that try to send invalid values.
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.
Lets hope the guaranteed bugs are found quickly; I can't find any more.
I wanted to wait until I had a chance to review it properly before wringing angry posts but you blink twice and it's already merged. I still didn't really look much into it so maybe some of it is not that bad or at least salvageable but from what I read in the PR description this is a disaster and should be reverted asap. As it only focuses on code improvements and completely ignores the big picture. Where is any consideration for the actual use cases? Where is any analysis of potential features of the command could use? It's been a constant nuisance for many years, I've lost count of how much stuff could've benefited from a well-thought command system. Highlights, gamescripts, game recording, admin port. And all I see in this PR is just templates are cool, templates go brrrr. Even something as simple as its impact on the bandwidth usage in multiplayer is nowhere to be seen. Or some mention of that admin port shit it does. I get that breaking admin api is already commonplace but what is admin client even gonna do with this "data"? It was already bad enough previously, but at least there was some structure to it. So yeah, changes like this make me want to just quit doing anything OpenTTD-related. You spend years trying to make the best with what you have and then you have to rewrite half of your code every time some core devs decide it's a good idea to disregard any analysis and foresight and break everything for some little gimmick. |
I agree it's definitely not trivial to get your head around without (prior) knowledge, and I'm sure there are things in the documentation that can be improved upon to make things clearer. Having said that, I think that this PR is looking at the big picture of the commands within OpenTTD. The foremost two/three issues that this PR tackles is having a fixed size of 2x32 bits (+ 1 string + 1 tile index) for commands, therefor untyped parameters and a lack of flexibility with commands. Lets take CmdPause as an example. Previously it looked like: Due to the "unlimited" amount of parameters you can now pass, there is no need to perform bitstuffing before calling a command and undoing the bitstuffing within the command. You just pass the right values into the parameters. Now, if someone changes the parameters of the function, the compilation for the callers will break instead so you can go to those locations to pass the right information, instead of hoping that you fixed all callers. That alone gives a lot more flexibility, but there is more. You are not limited to returning "only" a I do agree that the command logging for the admin port could use some love. We have to check what to do there, but it was already a random (meaningless) blob of data that could be changed at any time. I wonder whether we might now be able to at least give some meaning to the packet; maybe the actual name of the command, and per parameter the type and value in some more useful format. Then an user of the admin logging can at least gain some more knowledge out of the data which would be more stable than it was before. |
@ldpl Regarding your final point, the storybook and goal commands in specific would probably now benefit from a big pass to perhaps change them fundamentally. For example, allow them to send bulk updates so a single command could create an entire storybook, or update all the goals for a company. That kind of change would be out of scope for this PR, but I'm quite sure it would be worth the work to do. |
@nielsmh Technically there is not much difference between sending a list of changes in one command and sending a list of commands with changes. Except you'll need to add complex structures to DoCommand templates, implement new GS api methods, etc. And, tbh, no matter how you wrap it up updating UI from the server side is going to suck no matter what. So this overcomplication will ultimately be for nothing. Also, it's not like UI is the only thing that can benefit from the command spam, it's just the most obvious one. |
You mention the update rate as a specific problem, due to the current limit of one simple update per command. If the updates could be batched into a single command, the updates that currently may take several real-time minutes could perhaps be reduced to take 32 game ticks. |
You'll have to do batching for each command for every specific usecase instead of solvng the issue once and for all. Slow story book - batch story book, slow goals - batch goals, slow town updates - batch town text, batch industry text, batch object placement, etc, etc, etc. Every time someone comes up with something new you'll have to make a batched command. |
I can see why batching would make sense, but what does it have to do with how commands are handled in the source code? Currently scripts are suspended after each issued command because commands are delayed at least one tick in networked games. So the problem with any kind of command batching isn't any templated code, but that you'd have to completely redesign the script API to make command execution asynchronous from the scripts POV. This is a problem of a totally different layer than the command layer changed by this PR. |
@michicc yes, that problem is of a wider scope than just command layer, but it's related to it nonetheless. If there is no way to identify commands on the network it will be impossible to do asynchronous execution no matter what you do to the script api. So it's something that, at the very least, should be considered when designing the command layer, otherwise you may end up in a situation where for someone to make a progress on that script api redesign he'll have to completely redo your command layer first. |
Motivation / Problem
Command handlers are passed their data in bit-stuffed form, and with a fixed amount of bits available. This is a) prone to errors as there is no compile-time check if parameters where stuffed into the right spot, and b) limits extensibility as there's some reservation to blanket increase the parameters for one single thing.
Both problems can be solved if one allows arbitrary, typed parameters for commands using some C++ magic.
Description and Discussion
With this PR, DoCommand and friends are spiced up using some C++17 template magic to achieve variable amounts of typed parameters. Main implementation points:
CommandCost
, making most of the previously used globals obsolete.DoCommand
) directly returns the command proc returns.DoCommandP
) passes the return values on to the command callbacks.if constexpr
, the compiler can be tasked with generating matching function calls.Some additional random comments:
struct Do
execution layer is completely eliminated (including e.g. the various tuple gymnastics), leaving just a plain old static function call.TileIndex
is changed into a struct to be able to reliably used in template specialisation.Limitations
There are bugs, 100% guaranteed 😄
This changes the on-network command packet, no idea if this necessitates any changes to the GC or the TURN relay.
I might have forgotten to update some documentation somewhere.
Checklist for review
Some things are not automated, and forgotten often. This list is a reminder for the reviewers.