Skip to content
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

Parallel semantics for stepping robots #1925

Open
byorgey opened this issue Jun 10, 2024 · 10 comments
Open

Parallel semantics for stepping robots #1925

byorgey opened this issue Jun 10, 2024 · 10 comments
Labels
C-Project A larger project, more suitable for experienced contributors. G-Robots An issue having to do with robots. S-Nice to have The bug fix or feature would be nice but doesn't currently have much negative impact. Z-Feature A new feature to be added to the game.

Comments

@byorgey
Copy link
Member

byorgey commented Jun 10, 2024

#1920 really got me thinking. The main reason concurrency is tricky, as @kostmo says there, is because robots take turns executing, and each robot can see modifications to the world by the previous robot. Thus, currently, if we want to run CESK machines in parallel we would have to work very hard to make sure we respect such dependencies (or else give up on determinism, which I definitely don't want to do).

However, perhaps this is an opportunity to rethink our semantics. I'm kind of thinking out loud here, so I'm not yet sure how strongly I want to argue for this, but this issue is a proposal just for the sake of discussion.

The main idea: parallel semantics

The main idea is to adopt a parallel semantics in which, every tick,

  1. All robots look at the same state of the world from the previous tick, do some computation, and decide on at most one tangible command they want to execute.
  2. All updates to the world then happen simultaneously (with appropriate conflict detection).

i.e. think of the way cellular automata are typically defined: every cell decides what its state is going to be in the next tick based on the state of neighboring cells on the current tick, and then all the cells update at once.

Benefits

I think adopting this semantics could have a lot of benefits:

  • It makes concurrency (Execute robots concurrently #1920) trivial: by definition, we can easily run all robot machines in parallel since they all look at the same world state from the previous tick. We just need a synchronous conflict resolution and world update phase at the end of each tick once all robot machines have finished executing. But that will still save a ton of time since most of the work each tick is running robot CESK machines; I think the relative amount of time needed to check for conflicts and update the world will be very small.
  • Relatedly, thinking ahead to the future, it seems like it would make implementing networked play (Networked play #2205) much easier. Imagine you are playing with someone over a network in a shared world. You can simulate your robots, and they can simulate their robots, and the only network traffic necessary is to transmit all planned world updates; then once each of you has the full set of planned world updates for the tick, you can each independently do conflict detection and update your copy of the shared world. (Of course gracefully handling network lag, dropped packets, etc. would still be nontrivial, but the point is that it still seems much easier conceptually.)
  • It seems simpler to understand and explain, and less "magical". Currently, you cannot predict exactly what some robots are going to do unless you know the secret order in which they will execute, which just so happens to be in order of their robot ID. It is weird to have implementation details leak into the semantics like this. In contrast, with a parallel semantics you can predict exactly what some robots will do knowing only their programs; the result does not depend on any kind of hidden order of execution.
  • It also seems nicer theoretically. For example, sets of world updates are a monoid (as long as you keep conflicts around as a special kind of "world update" indicating that there was a conflict at a given location), with the empty/identity update as the identity, and union + conflict detection as the binary operation. Of course, this is exactly what would allow us to run things concurrently.

Downsides

  • We would need some new kind of "conflict exceptions" that get thrown when robots try to take conflicting actions. For example, currently, if two robots try to pick up the same rock at the same time, one of them will succeed and the other one will get a "there is nothing here to grab" exception. With a parallel semantics, both robots would get a new "conflicting grab" exception (the "nothing to grab" exception could still happen if a robot executed a grab in an empty cell).
  • It is "stricter" than our current sequential semantics in the sense that it will cause more exceptions to be thrown. If two robots try to grab the same thing, under a sequential semantics at least one of them succeeds, whereas under a parallel semantics both of them must fail.
    • Personally, I do not find this to be much of a downside at all --- I think I actually prefer a more "symmetric" system where winners and losers are not chosen arbitrarily --- but I mention it since some people might.
  • It would be a lot of work to implement. Especially the debugger would need a complete overhaul. However, I am hopeful it would result in a big simplification to the debugger.

Practical implementation details

Practically, it might work as follows:

  1. Pass the current GameState to all robots and run them for one tick, each one resulting in an updated robot and a set of planned world updates (probably at most one world update, but in theory I suppose there could be more than one). There will be lots of kinds of world updates but they will be things like "place entity E at location (X,Y)", and so on. Each world update also records the robot ID of the robot that caused it.
  2. Collect all the world updates and check for any that conflict (this should be possible to implement efficiently --- it does not require quadratically looking at every pair of updates --- since e.g. we can put the updates in a map by location and check for any that are at the same location, etc.).
    • Note that I don't think robot moves should ever conflict: if robot 1 moves and robot 2 gives robot 1 an item at the same time, that is OK and will not conflict.
    • Really the only thing I can think of that can cause conflicts at the moment is picking up or putting down entities.
  3. Perhaps the trickiest part is that if there is a conflict, we have to go back and update the machines of the robots which issued the conflicting updates (to turn their current machine state from a success into an exception, to put back an entity that the robot thought it was going to place, etc.). But I don't think this would be too bad.
    • Perhaps the right way to do this is that simulating a robot for one tick returns a triple of a world update and two CESK machines: one to use if the update is successful and one to use if the update failed (or perhaps the second thing is not just a CESK machine, but a function from exception to CESK machine, or something like that). Then we just select the correct one to continue with depending on the outcome of conflict resolution.
  4. Once we have dealt with any conflicts, we take the resulting set of world updates and apply them to the world all at once.

Impacts

Even though this would technically be a breaking change, in most cases I don't think it would make a difference. Differences would only be observable in the case of conflicts, which are rare. So I expect the vast majority of our test suite would continue to work. Perhaps one or two things where we specifically measure tick timings etc. might need some adjustment.

Having written all this, I think I'm still pretty positive about this idea. Eager to hear others' thoughts.

@byorgey byorgey added Z-Feature A new feature to be added to the game. C-Project A larger project, more suitable for experienced contributors. S-Nice to have The bug fix or feature would be nice but doesn't currently have much negative impact. G-Robots An issue having to do with robots. labels Jun 10, 2024
@kostmo
Copy link
Member

kostmo commented Jun 10, 2024

System robots would need special consideration, since multiple commands can run in one tick with instant.

And in #1736 we achieved the ability for a robot's actions in one tick to be fully responded to by the system before that robot's next tick. There should be some way to preserve that ability.

@byorgey
Copy link
Member Author

byorgey commented Jun 10, 2024

Very good points, I had not thought of these! But I think they are ultimately solvable:

  • For system robots running multiple commands with instant, I don't think there's any real difficulty---they can keep track of the hypothetical world they are creating (in case they want to e.g. put something down then pick it back up again) and ultimately return a set of (potentially) many world updates.
  • As for react immediately to wakeups #1736, that takes a bit more work but I think it is ultimately doable. During the conflict detection phase, we can also look for any world updates that cause wakeups. If there are any, we wake those robots up, let them run, and add their updates to the pending update set (potentially triggering additional wakeups, and so on). Once there are no more robots to be woken we can finally apply all the updates.
    • I suppose the really tricky thing here is circular dependencies: e.g. what if a robot A places something in a cell, which causes robot B to wake up and also place something in the same cell? Now there is a conflict, so robot A throws an exception and does not actually place anything in the cell --- which means robot B does not actually wake up after all, which means there is no conflict, ... I suppose the way to deal with this is that robots cannot be "unwoken", even if waking them up causes a conflict which would invalidate their waking condition. i.e. the act of trying to place something in a cell is enough to wake them up; it doesn't matter whether the place is ultimately successful. This seems like a very unlikely corner case anyway.

@ltouro
Copy link

ltouro commented Jun 10, 2024

We would need some new kind of "conflict exceptions" that get thrown when robots try to take conflicting actions. For example, currently, if two robots try to pick up the same rock at the same time, one of them will succeed and the other one will get a "there is nothing here to grab" exception. With a parallel semantics, both robots would get a new "conflicting grab" exception (the "nothing to grab" exception could still happen if a robot executed a grab in an empty cell).

Why not fallback to sequential processing for the conflicting actions? Most of the work would be done at that time and you would get less exceptions for a bit of perf cost. Maybe even derive conflicting sets of actions and process different sets concurrently (if this is indeed need)

@byorgey
Copy link
Member Author

byorgey commented Jun 11, 2024

Why not fallback to sequential processing for the conflicting actions?

That would certainly be possible. My main argument against it would simply be that it is less theoretically elegant, and it introduces extra complication for little benefit. If we fall back to sequential processing for conflicting actions, then once again to predict what will happen you need to know the secret order in which the conflicting robots will be executed. And if we think ahead to networked play, how will you even decide how to order the robots (since the ID numbers may not be globally unique across different clients)? You cannot just say "mine first, then theirs" since all the participating clients would have to agree on a sequential order to use in the case of conflicts. With a more symmetric, purely parallel approach, on the other hand, there is nothing to agree on, and the robots are truly unordered.

@xsebek
Copy link
Member

xsebek commented Jun 11, 2024

Nice write-up @byorgey! Splitting out the world updates from CESK evaluation sounds great.

Like @kostmo I also immediately thought about instant, but maybe it's not that hard. We could apply a set of primitive world updates and check if they are in conflict with other sets of updates.

I think we could even retrict the instant command if it simplifies this feature. Because of hypothetical computation we should not lose any power.

@ltouro
Copy link

ltouro commented Jun 11, 2024

If we fall back to sequential processing for conflicting actions, then once again to predict what will happen you need to know the secret order in which the conflicting robots will be executed.

I was thinking making this order intentionally random. It would introduce some uncertainty into the environment that might be a fun/challenging new feature in the game itself. I was not considering networked play though.

@byorgey
Copy link
Member Author

byorgey commented Jun 11, 2024

Like @kostmo I also immediately thought about instant, but maybe it's not that hard. We could apply a set of primitive world updates and check if they are in conflict with other sets of updates.

Yes, exactly, we are going to be dealing with sets of world updates anyway, so it makes sense for one robot to be able to generate an entire set. However, I just realized one tricky thing about this: let's say a system robot executes instant (move; grab; move). What if the grab fails because it conflicts? We actually need to have an ordered list of world updates generated by the system robot, so that if one conflicts we can invalidate all the ones that came after it. And in that case I guess it's not enough to just replace the robot's CESK machine with an error state: we have to actually wind it back to the state right when it executed the conflicting command, and then keep running it from there with an exception (since it might e.g. have a try block around it?).

I don't know, this sounds rather complex. Maybe there is a simpler approach... perhaps system robots never conflict? e.g. if a system robot grabs a rock at the same time as a user robot, then the system robot gets the rock and the user robot gets a "no rock to grab" error. And if two system robots grab a rock at the same time, they both get a rock.

@byorgey
Copy link
Member Author

byorgey commented Jun 11, 2024

I was thinking making this order intentionally random. It would introduce some uncertainty into the environment that might be a fun/challenging new feature in the game itself.

Ah, I see. That's definitely better than having the order depend on robot ID. Then at least it would be "predictably unpredictable". However, the game is currently completely deterministic, in the sense that executing the exact same commands from the exact same starting state (i.e. same world with same seed) will always yield exactly the same results. I would be sad to lose that property---partly on aesthetic grounds, but partly also practical grounds, e.g. determinism makes possible (or at least easier) potential features like networked play and playback of recorded game sessions.

@kostmo
Copy link
Member

kostmo commented Jun 15, 2024

We should probably develop a benchmark up front to estimate the best-case performance improvement of parallelism. One concern I have is that of "context switching". In some games, the parallelism is distributed "vertically", with one dedicated processor to physics, another to rendering, etc. which minimizes context switching. In our case, "horizontal" parallelism may entail evaluating a batch of robots for a single tick with mapConcurrently. This runs the risk of context switching outweighing the gains of parallelism.

Update: parMap may be pretty lightweight in terms of context switching

@byorgey
Copy link
Member Author

byorgey commented Jun 15, 2024

Just to record some points from our discussion today:

  • This issue in and of itself is independent of whether we actually add concurrency. I still think implementing this is a good idea even if we were to keep everything single-threaded.
  • However, I agree we should be careful about performance if we do add concurrency. But yes, for running all the robot CESK machines, using Haskell's native lightweight threads/sparks/etc. should be a good way to go, rather than async.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
C-Project A larger project, more suitable for experienced contributors. G-Robots An issue having to do with robots. S-Nice to have The bug fix or feature would be nice but doesn't currently have much negative impact. Z-Feature A new feature to be added to the game.
Projects
None yet
Development

No branches or pull requests

4 participants