Procedural Planets

Final project in TDT4230 Graphics and visualization. Procedurally generated planets with varying noise and colours. Focus has been on applying Level of Detail techniques, like scaling, moving coordinate system, and generating terrain with higher resolution depending on distance to planet. Detail level allows flying around in a solar system, as well as landing on and exploring the vast emptiness of each planet.

Running

The project is written in Rust and requires Cargo. Run with cargo run.

Some configurations can be adjusted without recompiling in the settings.conf file. The scene is constructed in scene.rs, and planets parameters can be configured there.

Controls

  • W A S D, shift, space: Movement
  • Up, Down: Increase or decrease movement speed
  • F: Toggle free float or anchor to center of gravity of closest planet
  • I: Toggle GUI
  • M: Cycle polygon modes (fill, lines, points)

State of the project

Starting to get some eye candy 2 Starting to get some eye candy Starting to get some eye candy Starting to get some eye candy Starting to get some eye candy

Behind the scene

Generate a cubesphere, vertices should be quite evenly distributed

The cubesphere. Not perfect, but well suited for my usecase.

Computed normals, but struggling with seams

Adding noise to the sphere makes the foundation for a planet.

Apply noise and add an additional sphere as an ocean

An extra, low-poly sphere does the job as an ocean.

Playing with adding a height dependent colour map

Adding height dependent colour scheme.

LoD, though buggy, is making progress

The varying detail level is shown clearly in wireframe mode.

Things to add

  • [Bug] Fix the shadowing. Haven't gotten shadow from area lightsource to work.
  • [Bug] In anchored and landed state, adjust camera direction as well when moving along the curve of the planet.
  • [Bug] Continue work on precision, still some bugs when walking on a planet.
  • Improve ocean shader. Currently just a blend of dark and light colour on a smooth sphere.
    • Should deform normals and get some better light reflections in waves.
    • Reflect land masses in the ocean.
  • Improve atmosphere, the current solution is quite naive and introduces a few problems with lighting.
    • Figure out how to blend atmosphere with planets showing behind.
    • Light refraction when landed to make daylight and coloured sunset and sunrise.
    • There is a thing called post-processing shaders, that might solve this as well as the oceans.
    • Nvidia GPU gems: atmospheric scattering
    • Rayleigh scattering
    • Lague
  • Clouds, Mie scattering
  • Improve skybox shader. Adding some colours to the stars would help a lot.
  • Pass planet properties through SSBO (shader storage buffer object) instead of uniforms, allowing an infinite amount of planets.
  • Controls: Add gravity to planet properties, and use it in the physics part.
    • Maybe add preset speeds for player states, to avoid having to blast the arrow buttons when switching
  • Add controllable spaceship and maybe a player mesh.
    • Add a spaceship mesh in the viewport when flying around. I have meshes that might work.
    • Make it possible to land the spaceship and walk away from it (reworking the Anchored → Landed state).
    • Player mesh for a 3rd person view, or for shadows in 1st person view.
  • Add planet rotation. Started, but discontinued for the project deadline.
  • Add texturing and materials, currently the plain coloured terrain is very boring. Define a set of selectable material properties and implement in shader.
    • Define and implement some different materials: Sand, snow, gravel, grass, bedrock, dirt. Generate texture/normal map for each kind from noise, saving texture buffer and its mipmaps.
    • Add a tessellation shader to apply transforms. Normal maps generated from noise.
  • Improve terrain generation. Currently just simple fractal noise on Perlin.
    • Define occurence of land masses, continents, smaller islands.
    • Structures: mountains, flatlands, craters, volcanos, ridges, etc. Noise function to define occurence of these.
      • Blend structures when they overlap.
  • More models, add static buildings and constructions on planets, static or moving NPC vehicles, satellites, spacecrafts, land vehicles.
  • More advanced planet trajectories, usually they are ellipses and not perfect circles (Galileo might disagree).
    • Even more, entities in real world can revolve around each other in a pair.
    • Could consider implementing some simplified gravitational physics instead of just deciding the trajectory paths.
  • GUI improvements
    • Add a loading screen (Not that loading takes any long atm), a start screen, and some nice menu/settings screen
    • Consider adding a crosshair to emphasize moving direction
    • Give planets names and display it when pointing at them, also show distance to the planet.
    • Improve the GUI layout (although it's mostly for debugging), maybe create a new layout for normal playing, not displaying information just needed for debugging.
  • Extract general elements to an improved project baseline.
    • mesh module
  • Mesh: merge meshes
  • Do scene graph elements as a trait instead of a single struct.
  • LoD/Memory usage control: Keep track of generated meshes and time since last use / probability of imminent usage, avoiding huge memory consumption on unused meshes.
    • There is an architectural challenge with the application design, that the responsibility for meshes and buffers are somewhat shared between the SceneNode and the Planet structs.
    • Planet should solely describe the properties of a planet, for its mesh to be created.
    • Mesh should have a better defined mandate, maybe only responsible for creating meshes from available constructors and return the mesh - or add the task of making the VAO, returning the VAO index and deleting the (intermediate) mesh from CPU memory.
    • Who should be responsible for handling LoD? Is it a special case, or should it be integrated within everything that will be rendered (if ever some other models were to be added)?
    • Reaching a "LoD enabled" object in the scene graph, would first require determining the current level and render that one, then continue on any children it might have in the scene graph. Meaning: The scene graph should then only be a tree structure for setting rendering order and letting objects have relative transformations.
Structure Responsibility
Scene Graph Relative transformation (scaling, rotation, position), parse tree to render. Track state between render objects
Planet Description of planet properties.
Mesh Class for constructing meshes. Role is done after a mesh is loaded to VAO.
[missing] "Render Object"? Hold the buffer pointers associated with an object, as well as its transform, material, texture, and other uniforms. Different LoD nodes will be different render objects, but they will also be connected in that they represent the same object, and should not co-exist (at least not be rendered at the same time).
[missing] Could be same as above. For an object that supports LoD, decide which level we are on. Keep track of which levels are ready in VRAM, and request meshes if a level should be used but does not contain a mesh. Connect a pointer with a centralised memory manager, which will force the containing buffer to be deleted if it is too old.