#ecs #abstraction #compile-time #zero #zero-cost #entities #box

zero_ecs

Entity Component System (ECS), using only zero-cost abstractions

23 releases

0.2.22 May 13, 2024
0.2.21 Apr 14, 2024
0.2.18 Mar 12, 2024
0.1.3-beta.2 Mar 5, 2024

#218 in Game dev

MIT/Apache

15KB
58 lines

Zero ECS

Zero ECS is an Entity Component System that is written with 4 goals

  1. Only use zero cost abstractions - no use of dyn and Box and stuff zero-cost-abstractions.
  2. No use of unsafe rust code.
  3. Be very user friendly. The user should write as little boilerplate as possible.
  4. Be very fast

It achieves this by generating all code at compile time, using a combination of macros and build scripts.

Instructions

Create a new project

cargo new zero_ecs_example
cd zero_ecs_example

Add the dependencies

cargo add zero_ecs
cargo add zero_ecs_build --build

Your Cargo.toml should look something like this:

[dependencies]
zero_ecs = "*"

[build-dependencies]
zero_ecs_build = "*"

Create build.rs

touch build.rs

Edit build.rs to call the zero_ecs's build generation code.

use zero_ecs_build::*;
fn main() {
    generate_ecs("src/main.rs"); // look for components, entities and systems in main.rs
}

This will generate the entity component system based on the component, entities and systems in main.rs. It accepts a glob so you can use wild cards.

use zero_ecs_build::*;
fn main() {
    generate_ecs("src/**/*.rs"); // look in all *.rs files in src. 
}

Using the ECS

Include ECS

In main.rs Include the ECS like so:

include!(concat!(env!("OUT_DIR"), "/zero_ecs.rs"));

Components

Define some components:

Position and velocity has x and y

#[component]
struct Position(f32, f32);

#[component]
struct Velocity(f32, f32);

It is normal to "tag" entities with a component in ECS to be able to single out those entities in systems.

#[component]
struct EnemyComponent;

#[component]
struct PlayerComponent;

Entities

Entities are a collection of components, they may also be referred to as archetypes, or bundles, or game objects. Note that once "in" the ECS. An Entity is simply an ID that can be copied.

In our example, we define an enemy and a player, they both have position and velocity but can be differentiated by their "tag" components.

#[entity]
struct Enemy {
    position: Position,
    velocity: Velocity,
    enemy_component: EnemyComponent,
}

#[entity]
struct Player {
    position: Position,
    velocity: Velocity,
    player_component: PlayerComponent,
}

Systems

Systems run the logic for the application. They can accept references, mutable references and queries.

In our example we can create a system that simply prints the position of all entities

#[system]
fn print_positions(world: &World, query: Query<&Position>) {
    world.with_query(query).iter().for_each(|position| {
        println!("Position: {:?}", position);
    });
}

Explained:

  • world: &World - Since the system doesn't modify anything, it can be an immutable reference
  • query: Query<&Position> - We want to query the world for all positions
  • world.with_query(query).iter() - creates an iterator over all Position components

Creating entities and calling system

In our fn main change it to create 10 enemies and 10 players, Also add the systems_main(&world); to call all systems.

fn main() {
    let mut world = World::default();

    for i in 0..10 {
        world.create(Enemy {
            position: Position(i as f32, 5.0),
            velocity: Velocity(0.0, 1.0),
            ..Default::default()
        });

        world.create(Player {
            position: Position(5.0, i as f32),
            velocity: Velocity(1.0, 0.0),
            ..Default::default()
        });
    }

    systems_main(&world);
}

Running the program now, will print the positions of the entities.

More advanced

Continuing our example

mutating systems

Most systems will mutate the world state and needs additional resources, like texture managers, time managers, input managers etc. A good practice is to group them in a Resources struct. (But Not nescessary)

struct Resources {
    delta_time: f32,
}

#[system]
fn apply_velocity(
    world: &mut World, // world mut be mutable
    resources: &Resources, // we need the delta time
    query: Query<(&mut Position, &Velocity)>, // position should be mutable, velocity not.
) {
    world
        .with_query_mut(query) // we call with_query_mut because it's a mutable query
        .iter_mut() // iterating mutable
        .for_each(|(position, velocity)| {
            position.0 += velocity.0 * resources.delta_time;
            position.1 += velocity.1 * resources.delta_time;
        });
}

We also have to change the main function to include resources in the call.

let resources = Resources { delta_time: 0.1 };

systems_main(&resources, &mut world);

Destroying entities

Let's say we want to create a rule that if player and enemies get within 3 units of eachother they should both be destroyed. This is how we might implement that:

#[system]
fn collide_enemy_and_players(
    world: &mut World, // we are destorying entities so it needs to be mutable
    players: Query<(&Entity, &Position, &PlayerComponent)>, // include the Entity to be able to identify entities
    enemies: Query<(&Entity, &Position, &EnemyComponent)>, // same but for enemies
) {
    let mut entities_to_destroy: Vec<Entity> = vec![]; // we can't (for obvious reasons) destroy entities from within an iteration.

    world
        .with_query(players)
        .iter()
        .for_each(|(player_entity, player_position, _)| {
            world
                .with_query(enemies)
                .iter()
                .for_each(|(enemy_entity, enemy_position, _)| {
                    if (player_position.0 - enemy_position.0).abs() < 3.0
                        && (player_position.1 - enemy_position.1).abs() < 3.0
                    {
                        entities_to_destroy.push(*player_entity);
                        entities_to_destroy.push(*enemy_entity);
                    }
                });
        });

    for entity in entities_to_destroy {
        world.destroy(entity);
    }
}

Get & At entities

Get is identical to query but takes an Entity. At is identical to query but takes an index.

Let's say you wanted an entity that follows a player. This is how you could implement that:

Define a component for the companion

#[component]
struct CompanionComponent {
    target_entity: Option<Entity>,
}

Define the Companion Entity. It has a position and a companion component:

#[entity]
struct Companion {
    position: Position,
    companion_component: CompanionComponent,
}

Now we need to write the companion system. For every companion we need to check if it has a target. If it has a target we need to check if target exists (it could have been deleted). If the target exists we get the value of target's position and set the companion's position with that value.

We need to query for companions and their position as mutable. And we need to query for every entity that has a position. This means a companion could technically follow it self.

#[system]
fn companion_follow(
    world: &mut World,
    companions: Query<(&mut Position, &CompanionComponent)>,
    positions: Query<&Position>,
) {

Implementation: We can't simply iterate through the companions, get the target position and update the position because we can only have one borrow if the borrow is mutable (unless we use unsafe code).

We can do what we did with destroying entities, but it will be slow.

The solution is to iterate using index, only borrowing what we need for a short time:

#[system]
fn companion_follow(
    world: &mut World,
    companions: Query<(&mut Position, &CompanionComponent)>,
    positions: Query<&Position>,
) {
    for companion_idx in 0..world.with_query_mut(companions).len() {
        // iterate the count of companions
        if let Some(target_position) = world
            .with_query_mut(companions)
            .at_mut(companion_idx) // get the companion at index companion_idx
            .and_then(|(_, companion)| companion.target_entity) // then get the target entity, if it is not none
            .and_then(|companion_target_entity| {
                // then get the VALUE of target position (meaning we don't use a reference to the position)
                world
                    .with_query(positions)
                    .get(companion_target_entity) // get the position for the companion_target_entity
                    .map(|p| (p.0, p.1)) // map to get the VALUE
            })
        {
            if let Some((companion_position, _)) =
                world.with_query_mut(companions).at_mut(companion_idx)
            // Then simply get the companion position
            {
                // and update it to the target's position
                companion_position.0 = target_position.0;
                companion_position.1 = target_position.1;
            }
        }
    }
}

TODO:

  • Re use IDs of deleted entities

Dependencies

~1.8–2.4MB
~50K SLoC