Jump to content

[nerds only] if I was in charge of the KSP physics engine


Recommended Posts

I am in no way saying that this is the only way to make a better KSP or an actual KSP 2; this is just what *I* would do if I had the time, budget and expertise.

Please take in consideration that I know how to program professionally; this is not coming out of nowhere. Finally, this is a technical post, this is not at all about the game mechanics. It will be nerdy and boring.

IMPORTANT: the model below would not leave enough computing power for a "pretty" game. All the power goes to computing physics. The game would look ugly.


1) I would not start with implementing space,  instead I would start with TIME.

I would implement the ability to have more than one timeline, and branching between those timelines. Or at least, I would lay clear concepts to divide that paradigm into smaller problems. Think "git" but applied to a KSP game: just like git has branches, commits, merge, rebase, squash, etc... Here we would have timelines, snapshots, etc.

Each parallel timeline would be a succession of "snapshots" of the player's universe.


You think it's weird and impossible? Well, this exists more or less already in KSP 1 as part of the so-called "multiplayer" plugin. Vanilla KSP lets you fast forward time, but on top of it the plugin lets you do more: you can create alternate timelines and it lets you "resync" them. Implicitly, it means a primitive form of "rewind" by reverting to snapshots.

2) I would think about how to SAVE those snapshots

Those timelines are a succession of snapshots.

Unfortunately, the player spends most of their time playing "real time" in KSP. It would be too resource-intensive to save the entire flow of events.

The only way is a) to restrict the moments in time when the engine can take a snapshot, and b) to make it possible to reconstruct/interpolate what comes after until the next snapshot.

It's a matter of making decisions (i.e. RULES), and sticking to it.

Consider KSP 1: it already forbids you from saving when you accelerate, or are in the atmosphere, etc. (I can't remember exactly if those restrictions have been worked around, but my point is that there are rules). It means that the INITIAL CONDITIONS after a savinga snapshot are UNAMBIGUOUS; with a bit of effort, the engine could more it less reconstruct what happens to a craft until the next snapshot

To give you an idea (among many real-life examples), you can imagine how a video codec works: it can compute the next few seconds of video by using (a) an initial state (a full screen image) and (b) a little bit of computation/incremental data (the minimum amount of changes between each video frame)

I did not detail here the tricks I would use to work around complicated situations, such as a craft under acceleration, or a craft entering/exciting a sphere of influence, etc. That will come after .

3) I would start with applying this discretization of time to CELESTIAL BODIES

it would be absolutely crucial to be able to find the position of every celestial body relatively quickly, as far in the future as we want.

If you ask an astronomy geek, they will tell you that it's impossible, because computations on multiple bodies creates a chaotic system (in the physics sense), i.e. incredibly heavy (nearly impossible) calculations

But again, for a videogame it's a matter of setting rules, to simplify and calculate only what's needed (while swiping the difficult bits under the rug). Illusion!

In the case of orbits I would stick to simple systems: a) the sun is central and immutable, b) the planets don't interact with one another, c) satellites don't interact with one another.

If you need a two-bodies system (i.e. Neptune so heavy that it deviates the sun) then hard-code it.


The position of each satellite is an offset to add to the position of its parent body. No complicated conics, maybe a simplification relying on segments of hyperboles or a handful of carefully chosen Bézier curves.

But remember that the whole engine is snapshots-oriented. You can't save every point (in time and space) of every orbit of orbits. Therefore, even the "simplified" orbits would eventually be broken down to a set of points with a LINEAR resolution between each point. between point A and point B, we decide that the celestial body travels at constant speed, in a straight line. We can smooth out the change in speed between each point (again: linearly).

Here I don't explain how to save the points' positions with enough precision; I will explain further.

Side note 1: A good idea would be to snapshot the positions of the Lagrange points, additionally to the positions of the celestial bodies.

Side note 2: there's still the case of random asteroids, like in KSP 1. I would make them pop in and out of existence randomly, but following a system of seeds (i.e. you can predict where and when it appears). then the engine can compute the asteroid's position based on that initial position and speed, but only IF it needs to, i.e. IF the player decides to interact with it.

4) Discretization of space

Like the original Dev of KSP wrote in an old blog post, one of the challenges of such a game is that the engine needs to manage a 3D space at a scale a of a few centimetres (does your kerbal's foot touch the ladder?) all the way up to astronomical distances.

This is resolved (on paper and in some way in KSP 1) by managing space at TWO different scales: the "SMALL-ISH" scale, i.e. everything that's close enough to the craft to be managed natively by the physics engine with sufficient precision. And the "LARGE" scale, which is more or less doubling or quadrupling the precision of floating point numbers by adding an extra 32, 64, etc. bits. Possibly in a separate variable that rarely changes.

In other words, the 3D space is a kind of VOXEL. The "secondary" number where you store the "large" position of a craft is the equivalent of the CHUNKS in Minecraft, and rarely changes.

But I would push the concept further, by developing a kind of space discretization more suitable for orbital mechanics.

Let's call it SPHERICAL VOXEL. For as long as a craft is within a sphere of influence, you could define its location above the surface by using its latitude chunk and longitude chunk. And then of course within each chunk, use a regular floating point position. The interesting part would be the altitude: you could define vertical chunks that are NOT divided linearly, but instead exponentially, to reflect the rapid decrease of gravity as you go up.

However, still following our concern for snapshots and requirement for minimal storage needs, the gravity change would be resolved linearly within each vertical chunk (with a linear smoothing as we transition from one chunk to the next)

Going back to discretization of space: when the craft is NOT within a sphere of influence, we save its position in a regular cubic voxel, with its initial position, direction and speed at the edge of the voxel, and since we know what forces are applied to it (both because of our simplistic model for celestial bodies positions and because we have strict rules for snapshots) we can calculate its position as simply as possible at any point in time until the next snapshot.

It's trivial to determine when an object leaves the cubical voxel to enter the spherical voxel, as it's directly related to its distance to the celestial body.

5) collisions

The double voxel system described previously (both the cubical voxel and the spherical voxel) lets us determine quickly when two objects are far enough apart from one another to NEVER collide. That is, when they are in two different chunks.

Any object going at extreme speeds might require extra care (the problem arises when it's going so fast that it skips several chunks between each game tick) but that's still resolved reasonably easily by following standard algorithms -- similar to netcode algorithms (simulating continuous movements from discrete data, like when counterstrike's net code determines if a projectile hit you). As a fallback you can decide that if a craft goes that fast then it should disintegrate! Or that you "lost connection" to it. Or create an entirely new model for crafts going at extreme speeds.

Anyways, once you've asserted that a craft can only go from one chunk to its neighbour chunk then you can compute what craft might collide with what other object within one chunk, both in time and space. Once again we rely on our extensive use of linear resolution of trajectories (or, at worst, polynomial) to cast "rays" (in the 3D graphics sense) and determine if the relatively tiny objects that are crafts (in the immensity of space) do collide within the large chunk. Let the physics engine do the rest, as the crafts are in fact rigid bodies.

6) rigid bodies

This paragraph is secondary to the whole system, but here's an idea: each craft part (tank, engine, antenna...) should in fact be designed from the beginning as a set of attachment points and struts/metal beams. Not literal struts or literal metal beams, of course (we don't need to see what's inside the craft's part!). But the physics engine should be fed the rough structure of the part, to know where it should bend and where it should break.

For example, a tank should not secretly be just two attachment points connected by one single vertice (with a pretty cylinder mesh slapped onto it as a skin). Instead it should be a handful of vertices arranged in cylinder shape, representing its inner metal beams. The physics engine would work with those so-called beams and calculate the constraints on them, like in the video game Polybridge. Pretty standard stuff.

KSP 1 already did an embryo of that when it introduced actual struts as well as autostrut. But while that "physical mesh" should remain hidden to the player during a flight, its existence should be made explicit (for example, through a new overlay in the VAB). The actual mesh of the part (the one visible to the player) would only be cosmetic. That mesh could even distort, following the "physical" mesh's distortions, by using a simple displacement shader. Only if the physical mesh gets "too distorted" then the part gets ripped off the rest of the craft, or explodes.

The rigid body that is the part doesn't need to have MANY physical points (that would require too much GPU). In fact, it needs very few points, the same way a collision is very often just a cube. Just enough to make it interesting. This way ships could break in other places than just the attachment points between parts. And they could bend in unpredictable ways.

no, I'm not advocating for KSP2's floppy rockets. Quite the opposite.

7) weather

use the standard way of generating weather and volumetric clouds in a game (seed-based pseudo-random noise) and keep it as deterministic as possible to shoehorn it into the snapshots and voxel systems. To put it shortly: 3D Perlin noise creates shapes in small cubes, that you can interpolate linearly. Just like our voxel. They fit together.

8) aerodynamics

The "Ferram aerospace" plugin lays the path for proper aerodynamics. It uses voxel, but it has nothing to do with our voxel system.

The main goal would not be to have "accurate" aerodynamics (that comes as a bonus) but instead to pre-bake as much aerodynamics-related data, in order to alleviate the GPU 's work as much as possible when resolving/predicting the trajectory of a craft. In other words: from the pre-baked model, try to extract some discrete aerodynamics behaviours (craft angled 0°, craft angled 10°, etc.) and make heavy use of those simplifications while maintaining a reasonable level of detail in the flight behaviour.

10) resync'ing of timelines

I've come full circle from time to "orbit-scale" space to "craft-scale" space.

It is now obvious that some aspects of the game can be resolved on-demand by the engine (snapshot, rewind, fast-forward, etc.), such as the position of celestial bodies and the position of crafts that are in orbit, while some aspects are still out of reach (such as the position of a craft submitted to aerodynamic forces).

The time has come to explain how I would resolve that.

Imagine you're flying a craft in the atmosphere and it separates into two crafts, each with a guidance system.
1. Both crafts' flights still need to be managed by the game engine. None of them becomes a falling brick. The engine needs to be robust enough to handle as many crafts as are present in the current voxel chunk, even if it means a drop in frame rate (no skipping allowed for the physics engine!!!)
2. When one of the crafts gets "far enough" from the craft currently being flown, the engine applies a simplified flight trajectory to it just to show it on screen, but that trajectory means nothing. The engine marks that "fake flight" as "unresolved".

The player can no longer switch to it.

3. In order to "resolve" what became of that craft, the player needs to (a) finish flying the current craft until it becomes possible to create a new snapshot of this timeline, then (b) tell the game that he/she now wants to "resolve" the flight of that other craft. Then (c) the game obeys by reverting to the snapshot taken when that craft went too far from the original craft, and (d) lets you fly it until it becomes possible to create a new snapshot for that craft.

Those two crafts now have different timelines, and both are resolved as far as the player flew those crafts.

But what if the player flies the second craft close enough to the first craft again?

That's the interesting part. The two timelines become reunited temporarily. If the crafts don't collide then the timelines separate again, possibly without changing the snapshot created previously (i.e., created AFTER flying the first craft to completion). But if the crafts collide, then the two timelines get reunited permanently (and any subsequent snapshots they might have had get deleted). After the collision, if they get separated again, then Both crafts' flights need to be resolved again, separately, just like before.

Please note that an "unresolved" flight can remain unresolved indefinitely. The incentive for the player to resolve a flight is that if they don't, then the craft never lands or never crashes, and the player can never recover the parts.

unresolved flights could be displayed in the tracking station,  just like any other craft.

11) resync'ing of timelines in the distant future

Everything I'm about to describe was already a paradigm in the multiplayer plugin of KSP 1, and we know from Dev interviews that the KSP 2 team had similar ideas as what I'm about to present.

Imagine this scenario:
1. you take 10 days to fly a craft to, let's say, the Mun, and you lend it there. That craft is named "MISSION A".
2. The game engine can assert that could repeat that flight as many times as you want, given the same resources and the same time to travel. Therefore, if the player requests it, the game can create a "commercial route". It can even assume that the same amount of fuel that was left in your craft when you landed will be present on any subsequent "automated" craft landing on the moon. It does not need to simulate each and every craft .
That's the trivial part.
now for the fun part:
3. Imagine you revert to the timeline snapshot just before MISSION A departed. Now you decide to launch a new MISSION B.
4. that second fight takes only 4 days to fly to Mun, because this time you flew better. It lands in the exact same spot where MISSION A will land 6 days from now.
5. We can let the player play with MISSION B as long as he/she wants. If that craft leaves the surface in less than 6 days, then nothing special happens. BUT.... Here comes the catch! If MISSION B stays 6 days then we know that it occupies the same surface chunk as MISSION A. the two timelines need to be resolved! The player is notified of it and needs to keep playing one of those crafts for as long as they're within potential collision distance, otherwise the flight is marked as "unresolved" as we explained before.

12) actual cooperative or competitive multiplayer?

By the way, this mechanics allows for real multiplayer in KSP. However, not real time.
It would be fun to imagine game modes where there's a game master who can decide who can rewind time, and how far back. Or each player can rewind time only a limited amount of times. Etc.

13) landmarks

[A somewhat dumb and out-of-place] addition to this long post.]

I find it strange that KSP 2's engine was not immediately able to handle the KSC properly (from a rendering perspective).
Each large landmark at the surface of a planet (whether be building, arch, cave, solar flare, etc ) should be modelled at several levels of detail. Low-poly version for everyone as soon as you get away from it! The collisions work with the low poly version at high speed/large distance, or with the full details at short range/low speed.

 
 
 
Edited by jeancallisti
better title
Link to comment
Share on other sites

According to the forums' descriptions, I don't think you should have moved it. It's about neither directly KSP 1 nor directly KSP 2.

It's more about a fantasy KSP clone. What could have been done of the KSP1 code base when the development of KSP 2 started.

Edited by jeancallisti
Link to comment
Share on other sites

This thread is quite old. Please consider starting a new thread rather than reviving this one.

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

×
×
  • Create New...