My RustConf 2018 Closing Keynote
2018-09-14
- 
            Rough draft of RustConf 2018 talk
            
                - Basic thesis of the talk:
- Why should you listen to me, who am I?
- How do you EVEN make a game in Rust?
- How do you EVEN make a game at all
- How games were made in the past
- Wayyyyyyy too much object orientation
- Back to the beginning.
- Generational indexes are awesome.
- Dynamic typing is actually kind of nice in VERY controlled quantities.
- The "registry" pattern
- ECS is SQL for games
- The one place where it all goes wrong
- Summary
 
I promised two things when I did the closing keynote at RustConf 2018:
- 
I will put the slides for the talk online 
- 
I will post the long-form version of the talk on a my blog 2.a. I will actually make a blog for this purpose 
...and I have taken my damn sweet time doing so. Finally, I have done what I promised I would do!
If you haven't seen it and would like to, you can see the live talk that I did here.
The slides for the talk (with a small errata) are now hosted here.
So, I said in my talk that I would post a potentially more interesting long-form version of the talk that includes things that I couldn't fit into the 30-ish minute time slot. What I've included below is my original long-form version of this talk, but my original intention was not to post this as-is. Originally I wanted to clean this up a bit more and make this into something that wasn't so much a giant wall of text, but after I started doing this I realized that I was just rewriting it entirely, and at that rate I would never get around to releasing it, which I promised I would do in a timely manner.
I'm not super happy with this, and in retrospect I think many of the things I cut for the 30-ish minute talk were probably wise to cut. I think the parts about console development are wildly speculative and not that helpful, and the part at the end about language boundaries being particularly difficult is, while true, kind of beside the point. However, people seemed very interested and are (very rightly) reminding me that I haven't posted this yet, so rather than delay any further I'm going to give you the very lightly edited text version of the talk that I made. Keep in mind this was before I realized I would NEVER fit this into 30 minutes :). This is written in a very informal, conversational style because this was intended as the source material for my talk.
Without further ado...
Rough draft of RustConf 2018 talk
This is a very rough outline of the things I'd like to talk about in my RustConf 2018 keynote.
This talk assumes that you know a bit of C++ in addition to a bit of Rust, and there are quite a lot of C++ code samples, but I hope they're obvious enough not to be distracting to the people who might not be used to C++.
Basic thesis of the talk:
Rust, by design, makes certain programming patterns more painful than others. This is a GOOD thing! It turns out that, for game development, the patterns that end up being the easiest to deal with in Rust closely mirror the patterns that are easiest for game development in any language.
I unfortunately had to learn this the HARD way!
Rust highly rewards data-oriented design with simple, understandable ownership semantics, and this is great news because this is a really good fit for game development. I suspect this is also true generally, not just for game development! (but what do I know?)
Why should you listen to me, who am I?
I was the lead programmer for Starbound, I've I was one of the first members of Chucklefish back when we were all working over IRC. I was the technical lead at Chucklefish when we first decided that we should probably have somebody be called the "technical lead". I still work with Chucklefish peripherally, but I've since moved back to the US and I work on things more independently now.
But, honestly not sure you SHOULD listen to me! I've helped release ONE (1) whole commercial game, and some of this talk is a cautionary tale about what NOT to do based on the ONE (1) commercial game I've helped make.
I definitely feel like you shouldn't make any... real grown up business decisions based on my advice. I can't say for sure that Rust is a good fit for you, or whether you should or shouldn't make your game in Rust, certainly there are probably larger concerns depending on, if you're an indie, what your personal skill level and tolerances and preferences are, and if you're a company what tooling you need and what engines, middleware, etc you already use.
What I can say is that personally I will be using Rust for game development (and other things!) for the foreseeable future, and if you'd like to hear why, or if you've already decided that you'd like to use Rust and would like to know what you're in for, then this talk may be useful.
This talk is going to be more... vague than I'd prefer. I'm generally more comfortable giving people specific very technical advice where I can prove my point almost from first principles, but things like architecture advice or general advice are probably never going to be so clear cut. I definitely think it's still useful, which is why I'm doing it (and people seem interested), but remember: this is all just like, my opinion, man.
How do you EVEN make a game in Rust?
I've gotten a lot of questions over the past year or so that more or less ask:
How do you make a game from scratch in Rust? No, seriously... HOW. I mean, I can see how you can do it in theory, but for some reason when I try to apply patterns that I'm used to from other languages I just hit lots of problems? I've heard that this is called "fighting the borrow checker", but... that doesn't really seem to help me much. What am I doing wrong?
Or, maybe it's something like:
I can see how rust is great if you like very strict control, I can see it being used for small utilities or things where security is paramount, but it seems very restrictive! I don't really see how you could make something large like a game without running into these limitations. How do you structure something like a game without needing Rc<RefCell
> and Arc<Mutex > everywhere? 
These (straw man) questions are of course about games, but they also mirror sentiments I've seen about Rust in general. In case it's not obvious, I disagree with this sentiment (I am at RustConf after all), and I think the best way for me to talk about WHY I disagree with it is to talk about what I know, so I'm going to come at this from the perspective of games.
The talk could just be me coming up to a podium and saying "Jeez, it's not that difficult! Data oriented design! If you're making a game, use an ECS! ... Thank you." and then walking off stage, and from a certain perspective I do feel that it can be that simple, but I think there's a more important fundamental point here and that's what this talk is about. The very very short answer to this question is: if you forget OO design and instead just concentrate on the data representation of your game state (or whatever you're making), and you try hard not to make anything more complicated than it really has to be, things can actually be pretty simple! This talk will be working up slowly to a simple ECS implementation in Rust and trying to give justifications for each step, but not all the steps are necessary. I don't even think using some specific design pattern like an ECS is essential, I think the more important larger point is to let go of some habits that, at least I had that make things more difficult than they need to be, and in Rust more so.
For some people the things that I say in this talk might be blindingly obvious. If this is all obvious to you, that's wonderful, but this hasn't always been obvious to me. In fact, part of the reason that I'm such a fan of Rust is that I didn't actually learn these lessons from trying to build a game in Rust, I mostly learned them before I ever even tried Rust for the first time, from mistakes that I made early in Starbound. I need somebody to pick on to demonstrate what NOT to do, so it's really convenient that I have such a great example in my past self.
I'm definitely going to repeat some advice that other people have given. In fact, some of these things are definitely widely known and have almost become kind of truisms: Inheritance is bad, OO is mostly bad, ECS design is good, etc. Hopefully to the extent that those are true I provide more evidence for them and describe why these are especially important in the presence of the borrow checker, but hopefully I can also add some nuance to these and this will be helpful to other people who want to use Rust for game development.
How do you EVEN make a game at all
Certainly a lot of these questions only appear when you're trying to come up with a game architecture from scratch, and certainly many people aren't going to actually do this. In fact, generally the advice for indie developers is that you should NEVER make your own engine, and if you use something like Unity or Unreal to make your game, a lot of these decisions are going to be made for you.
I definitely did NOT follow this advice. In fact, I think Starbound ran directly against every piece of advice you commonly give a new indie dev:
- 
Don't make your own engine (Starbound is written basically in bare C++ on top of SDL / libogg / libvorbis / libfreetype etc) 
- 
Always make a prototype and plan to throw it away (The code that I used to demo two weeks in is the same lineage as the 1.0 release) 
- 
Seriously, don't make your own engine, just use Unity / Unreal / Godot etc (We didn't even use boost! In fact, at some point we added our own versions of c++17 classes like std::optional and std::variant inside starbound, because at the time it was easier than starting to depend on boost. We also came up with our own system for doing 2d texture atlasing because we couldn't find a good way to do offline texture atlasing with so many small textures that can't be easily grouped) 
- 
Your first game should be simple, you should make something simple that you can release in a short time. (In starbound, you can have an item that the player holds that runs its own Lua script which affects player physics, which can cooperate with other items and abilities that the player has that have their own scripts that also can affect the player's physics (at the same time), while also giving the player scripted status effects which can add their own custom stats to a generic stat database that most entities have, which can have stacks of generic stat modifiers also controllable from scripts. There is also an entire crappy sort of DSL for drawing material blocks that visually connect in arbitrarily complicated ways.) 
I'm definitely not trying to tell anybody to make ANYTHING like what I made as my first commercial game, don't follow in my footsteps, I don't think I would recommend that for anybody. However, there are actually quite a lot of indie devs who don't use Unreal / Unity style game engines and instead of using these "framework" style game engines instead go for more "library" style where they simply use individual libraries for specific features, or use small "frameworks" that provide mostly rendering, audio, and input, and create the basic structure of the games they make themselves. (I once had a porting house call this "old school" development style, MUCH to my dismay)
- Any game made with XNA or MonoGame. XNA / MonoGame definitely provide a lot in a single package (rendering, sound, input) but they don't dictate the architecture of the game itself. This includes Terraria and Stardew Valley, just to name two.
- Any game made by Jonathan Blow (Braid, The Witness)
- Any game made by Zach Barth (Zachtronics). He has stated that when he starts developing a game, he starts with a bare event loop (input / update / render) and goes from there. (NOTE: I couldn't find a source for this, so I hope I'm not misrepresenting him, I think he said it in an interview that I now can't find)
- Any game made Supergiant (Bastion, Transistor, Pyre)
- Chucklefish games that are made 100% in house: Starbound obviously, also Wargroove (made with the generic C++ engine halley, but the main dev for the engine is the main dev of Wargroove).
- LOTS more that I'm not listing, many many many 2d games are made this way, some 3d games as well. There are dozens of us, I tell you.
Obviously the advice I give is not as applicable if you decide to use something like Unity or Unreal, which if you do and that's great and you're happy, then I'm not trying to convince you to leave that. Obviously if you use one of those, Rust is probably out of the question except for very very peripheral things, but honestly I still think eventually Rust WILL find its way into these sorts commercial mainstream game engine products, it's just a matter of time. For the moment though, the most likely people to use Rust and find this talk useful are other start-from-scratch indie types, and maybe a few AAA studios that make a lot of their own technology in house (Ready at Dawn!, EA SEED!). Also, I'm convinced there's going to be a Unity or UE competitor that uses Rust sooner or later!
So okay, suppose you've decided that you're going to make a game, you're NOT going to be using one of the opinionated game engines that decide much of the game structure for you like Unreal or Unity, and you've decided to use Rust, what would such a thing look like?
How games were made in the past
So, in the past, games were mostly engineered in a "data-oriented" way out of sheer necessity. There is not much room for abstraction when your target console has 128KB of RAM (SNES). I'm going to call this the "Action Replay" era of video game development. If you aren't aware, these are cheating devices kind of like "Game Genies" or "Game Sharks" if you know those, where you can patch a game's RAM state every frame. The "Game Genie" and similar devices allowed you to patch the ROM of the game cartridge, but the "Action Replay" devices actually let you insert hooks into say, the VBlank handler, that overwrote actual memory values at constant locations every frame to say give you infinite lives etc.
The reason this worked is because obviously with 128KB of RAM, you're not exactly going to have an implementation of malloc!. Every byte of storage is precious, so mostly games from this era are designed with very predictable manually managed layouts for their entire game state in memory. In the NES / SNES era, there was so little memory that generally the graphical representation of your game (tiles, sprites) and the logical representation of your game are the same, and if you kind of squint you see "entity" structures, but generally the maximum number of "things" that the game can keep track of are so few that there's not enough room for any generality. This is sometimes also known as the "healing power of off-screen" era of game development, because the moment you go off screen, everything about that part of the game level is often forgotten (and enemies are magically healed).
Imagine if you were to try to write a game for the NES in Rust (probably REALLY hard, but not impossible? No harder than C at least, which is equally really hard. I'm pretty sure basically all games from the SNES era are written directly in assembler.), you might define a single static data structure something like this completely made up example:
pub type ProjectileType = u8;
pub type EnemyType = u8;
pub type EnemyBehavior = u8;
pub type Tile = u8;
You can learn an enormous amount about how software works just by looking at its data structures in this way, even if this is a pretend example.
One thing to note here is that since every SNES game was overwhelmingly likely to be made in assembler, there almost certainly was no such thing as "data hiding", the entire state of the game is available at any point to the update loop.
This era where every memory location was precious and objects were carefully laid out in it actually lasted a huge amount of time, at least up to the N64. Let's look at another example which I'm moderately familiar with, and is still solidly in this era, but whose design is imo much closer to modern game engines. One of my all-time favorite games in the world: Mario 64. Mario 64 is interesting because it's 3D, and except for the low-poly simple graphics, is it really THAT much different from a modern 3D platformer? Yet, it's still solidly in the "Action Replay" era of gaming, when you get into a "game world" (you can run and jump around as Mario) you can reliably tell (more or less) where all the level data is going to live in the N64's expansive, luxurious 4 entire MB of ram. There's a predictable block of memory for the level geometry, for lots of various level flags, etc, but most of the dynamic content of the game is in the form of generic "entities" (or "objects"), much like modern game engines. In Mario 64, the entity structures are all exactly 608 bytes long, and there is a hard limit to 240 of them (sometimes this is a bit less, like in Bowser in the Fire Sea, there is a limit of 232 objects).
We don't actually know what language Mario 64 was programmed in, but it was likely C, so this is probably in the era where, if you had a time machine and a burning desire for a better systems language from the future, you could just as easily have used Rust to make a commercial Nintendo 64 game. Let's take a lot of artistic liberty and imagine what the structure of such a game would look like translated as directly as possible to Rust:
;
;
pub type EntityIndex = u8;
;
;
So, the specific details of these representations aren't really super important, the important takeaways here are that:
- The representation of the game state (though it's dramatically over-simplified here) is still very simple even in the real game. It's very predictable and amenable to RAM value poking, and there isn't really much in the way of "abstraction" here, everything is quite literal and built for purpose.
- We can't know for sure about the languages used for all the games in the N64 era, but we know at least for some titles that it was definitely C (Shadows of the Empire), and probably it was C in the vast majority of cases. The game structures all tended to be simple and predictable like Mario 64, you can imagine the game state being represented mostly as global C structs or arrays of structs.
- I'm speculating here, but probably there was not much "data hiding", you can imagine the structure of these games more or less as a giant static global struct containing "all of the game state", visible to all of the game's code.
- There are occasional pointers to functions, but there doesn't seem to be a whole lot of OO going on. There don't seem to be vtables, or even many things like vtables, or anything like that. This is potentially wild speculation and is definitely just a non-expert opinion, but from my "extensive" research into the memory layout and design of SNES - N64 era titles, it usually feels like you can kind of imagine the very basic, boring, made-for-purpose C code that the engine is made of just from looking at the data format.
Okay, so imagine you were to write Mario 64 very similarly to how it appears to originally be designed, but in Rust. You would effectively be writing a giant, procedural, single purpose game engine in a much much nicer C, but still at its heart, pretty much like C. Effectively your game engine is something like (DRAMATIC OVERSIMPLIFICATION INCOMING):
;
;
pub type EntityIndex = u8;
;
;
So I'm NOT recommending that you make games this way!  However, even if you
think this is 100% a ridiculous over-simplification, or you think that even if
you could make a large complex game this way, it would turn into a procedural
ball of mud, let's look at this for a while and see if there are any advantages
before we totally write this off.  Clearly there is a huge, gaping, obvious
disadvantage in that the entire game state is visible and mutable to all systems
and if something changes unexpectedly, you might have no idea where to look to
find what is changing the game state improperly.  Nobody is really suggesting
in the modern era that you write software with all state as effectively global.
HOWEVER there is one advantage here, which is that with a little bit of care,
you could write a game this way in 100% safe rust and almost definitely not run
into issues with the borrow checker.  You might not be able to use pointers
safely since certainly if you used pointers your entire game state would have
internal pointers to itself, so the one modification from Mario 64 style that we
need to make is that everywhere where there would have been a pointer, instead
store an index into some array.  The canonical example of this is the
"mario_entity" field on our GameState.  This may seem overly limiting, but with
a bit of thought it becomes clear especially if you had constant limits on
things (you don't even have malloc, remember!), you can always add something
like all_the_textures: [TextureDescriptor; 256], and use integer indexes into
this static array to describe textures instead of things like pointers.  The
other reason this is helpful w.r.t. the borrow checker is that Rust is rather
good at letting you split borrows for different fields of a public struct, so
since everything in our pretend-engine is just a giant nested tree of entirely
public structs, you should always be able to read the state of the game and
change exactly as much of the game as necessary in each system.  Remember, most
of the games in the "Action Replay" era are written like this, there is not
usually any malloc-ing, there is at most something very limited like
"allocating" only from a fixed array, and most everything is more or less
carefully laid out in a static struct somewhere.
One more criticism of this style is that you might say that using indexes instead of pointers is "safe" in the strict sense, but possibly only technically, you're trading UB and potential crashes with pointers for "random but unspecified" behavior if you access the wrong or outdated index, and potentially panics. You'd be right, btw! We'll visit this some more later, but just accept for now that it's possible and safe and this design doesn't run directly against the borrow checker.
Okay, so now I've hopefully described an admittedly over-simplified way that you COULD write a game in Rust that morally sort of closely mirrors the oldest game architectures. I'm not advocating this for your next game or anything, but there ARE games written now that are little more than this, written in direct, single purpose, procedural style. I'm not going to call out specific games, but I've seen a bunch of games written in a style that is little more than this, often wrapped in a paper thin veneer of OO. I've seen game source code that has a single 12k line world generation function. This is not an insult to such games, even marginally, the people who do this are often absolute geniuses who are just very good at knowing what to care about and what not to care about. There is real wisdom here!
But still, I'm genuinely not advocating this. Let's refer to this as the "UR-game-architecture", and we'll come back to it. This is approaching the "data-driven! use an ECS!" answer from the bottom up, let's digress and approach this now from the top down.
Wayyyyyyy too much object orientation
Now that we've covered sort of the simplest possible messy procedural C-ish game engine design, let's try to apply the principles of OO design and see if this is an improvement. I'm going to use Starbound or some simplification of Starbound as an example here, because I'm lucky enough to know a lot about it and have its source on hand, so I'm not going to be lying too much when I describe it.
What are the principles of OO? Not everybody entirely agrees with what OO is, but I'm including some basic hopefully non-controversial points:
- Single responsibility principle - Objects should have a single logical set of responsibilities, and methods should perform one operation inside that set of responsibilities.
- Encapsulation - You should bind data together with the functions that operate on it, keeping both safe from outside interference and misuse. This allows for refactoring by changing the internal representation of a class without changing its behavior.
- Abstraction, or "Liskov Substitution Principle", or similar - You should be able to substitute one derived class for another, as long as they share the same base and are used through that base class (or interface, or whatever).
- Interface segregation, or principle of least coupling, etc - The dependency on one class to another should use the smallest possible interface.
On a practical level, OO languages generally have a few important features to support OO design, namely object methods, private object data, inheritance, virtual methods, etc. I'm mostly going to be talking about C++ because it has a lot of OO footguns, it's what I know, and it's a super popular language for the "I'm gonna make my own engine" gamedev crowd.
We're going to try these principles out and see how they can be misused for game development, and then talk about how even though they often fail for gamedev (and this is now a "well known" state of affairs), they fail way more spectacularly with Rust. It is possible to make a game with at least some of these principles, though, and they're not all actually universally terrible ideas. I'm not really a proponent of OO at all, but there are at least a few good ideas that are sort of related to OO or kind of came out of OO (all of which Rust has and can perform admirably).
- The dot operator or "postfix functions". If you're a haskeller this is "type directed name resolution" as opposed to the normal state of affairs which is "name directed type resolution". A "powerful" way to avoid having to scope 100 different functions with very short common names, or have C style prefixes for everything. Great for IDEs too!
- Interfaces with laws (contracts) - Rust traits! No inheritance in sight, and they're best when they're small and come with meaningful rules around them, but if you squint they're kind of C++ pure virtual classes. They're awesome in the same way Haskell typeclasses are awesome.
- Data hiding - Being able to hide data to maintain invariants is invaluable, making a safe interface to unsafe code would not be possible without it. Very useful in the small, and with "library" code.
So just for the record I'm not trying to pick on the above good parts, these are fine, great even. With that out of the way, lets look at ways the rest of OO design can fail in gamedev.
So, superficially games seem well suited to OO because when trying to come up with OO designs, there are obvious "objects" that jump out at you. Using starbound as our example, things like "Player", "NPC", "Monster" are pretty easily understood concepts and obvious candidates for objects in our game, so lets start with these. We'll also include a "World" class which, like the Mario 64 example, is the basic structure of some live play arena (we're purposefully skipping things like interfaces, menus, etc and focusing on the core part of a game engine). I'll also have to switch to C++ because some of this is going to be difficult to express in Rust.
typedef uint32_t EntityId;
;
;
;
;
;
;
;
;
;
Right away, we see repeated structure in our data types, and probably these should be sub-objects with their own methods, so let's sketch that out a bit:
typedef uint32_t EntityId;
;
;
;
;
;
;
;
;
;
;
;
So far this isn't going too bad, but this is still just describing the data structures in our game. Remember, one of the OO principles we talked about is encapsulation, you want to expose a minimal interface to each of these structs (we really should call them classes!) and methods that only expose what is necessary. Also, there's that void pointer in entities inside world, we probably shouldn't have that, we're going to need to keep our entities stored somehow so let's make an Entity interface while we're at it for things that are common to all entities.
typedef uint32_t EntityId;
// Forward declare World to pass it to Entity
;
;
;
// Pure virtual interface!
;
;
;
;
;
;
So one point before we go on, why am I bothering with this EntityId stuff
here?  This is C++, why can't we use pointers?  Well, it turns out that doing
this is very unsafe, so the pattern that all game engines I've ever (in
languages without a fancy garbage collector) seen adopt is actually to have
some kind of map from some form of "entity id" to an actual entity pointer and
keep that in a single location.  The reason for this is that, say in C++, let's
say every entity keeps a shared_ptr
(In game engines, EntityId is usually something like an int that counts up, possibly a uint64_t that doesn't round or a uint32_t that does round, and this is done so that an EntityId is never or very rarely re-used. This way, if an entity goes away, generally anything monitoring it will notice it gone rather than some other entity immediately taking its place. There's another pattern here called "generational indexes" which is important and I'll talk about later)
Okay so this seems like it... could actually work to make a complex game with? I can sort of imagine a very simple game you could make with these classes and interfaces as they are right now, but let's flesh it out a bit to see what sort of problems we'll run into.
Let's think about what a monster needs to track a player. They might go through all the entities in the world, filter by only the players, sort by distance (you'd use some kind of spatial hash / kd-tree in reality), and then track the closest one. That actually seems basically possible with this as it is! Okay, new requirement: monsters should track players with the lowest health first. Uh-oh, okay player health is private, so we better make a public accessor for this:
;
We could have just made the health public, but OO design! What if there's an invariant being held, where if the health drops below some value, some animation state is triggered? What about damage? Certainly the single responsibility principle states that a monster shouldn't be in charge of setting a player's health.
Okay, new requirement: Monsters should not go after players marked as "admin". Alright, add an accessor!
;
Okay, we've made a little prototype game but no damage is happening yet, so let's go ahead and get damage set up. Hmm, where should the damage system for players go? Probably a player should reduce their health since it's their health... or maybe monsters could do it since it's their damage region? I dunno, I'll say it's the player's job. I guess that means we need accessors on the monsters for their damage region:
;
This is turning into a lot of accessors. We could just start making lots of things public, but what do OO principles state? That we should expose as little and finite an interface as possible so that we can change our implementation without refactoring. Maybe this won't get so bad.
Okay, new requirements: A certain monster will only aggro the player when the
player touches the ground near them.  Okay, this is going to require yet more
accessors, except because we've been following good OO principles about code
re-use, we have that internal member m_physics of type Physics, and it is the
only thing that can know whether an entity is touching the ground.  So, I guess
Physics needs an accessor to its internal state Physics::onGround And then
the Player needs an accessor Player::onGround which in turn will use
Physics::onGround.  Okay, this is turning into LOTS of accessors.
It feels like every time a new requirement comes in, you have to take what may have once been sane interfaces and "poke more holes through them". 8 months later, most of a semi-game has been built this way and there are LOTS of these holes poked through. Lots of code has no clear place to be because it concerns multiple entities at once, and a lot of logically similar functionality is split across several files. "The problem with OO is that everything happens somewhere else".
A new requirement comes in: I have an idea for a special kind of item that when the player holds it and its near a specific kind of enemy, it will glow. The enemies that trigger this should be scared of the item and back away, but have a special animation where they're mesmerized by the glowing item. This should only work for players that have achieved some specific quest goal.
You throw up your hands in frustration, this would require 4 separate modules to know about all of the other's internals (Player, Items, Monsters, Animation). You add a lot more accessors and special interfaces. Things are a mess.
So, these are made up examples, but they are not much different than the real things that I went through a lot. This is why it is now considered common knowledge now that for games, OO mostly just gets in the way. Data hiding has very limited utility for a game outside of just maintaining invariants at the edges of the code, where things are smaller and more contained. A lot, possibly even the vast majority of the interesting behavior in your game ends up spanning many data types, and does not naturally "belong" to any specific entity. Lots of entity types are 80% or 60% similar to others and its hard to re-use code, and the more modules we add inside our entities to help reuse code, the more layers are added. What have we gained vs our UR-architecture? That not everything is visible to all our functions, even though we keep adding more and more and more accessors so it sort of is? Maybe some marginally better code organization, but also sometimes worse code organization?
Okay, so obviously this has a lot of downsides even in C++, and obviously you know where we're eventually going to end up because I told you at the beginning of the talk (Just use ECS!). You CAN make a game this way, but you'll have a lot of problems and poke a LOT of holes and probably end up with some mega-objects. Just for a nice visceral reaction, let's look at the actual, for-real Player class inside Starbound as of the current version:
;
Interesting fact, there are many more methods than there are fields in Player, somehow. If you handed me a game and asked me to add a feature to the engine, and asked me whether it would be easier to add in the "too much OO architecture" vs "UR-architecture", I'd say it's easier in the "UR-architecture" almost every time. BUT it IS doable with effort obviously, because lots of Starbound is like this :(.
So, this has downsides in C++, but not every game has as many crazy one-off features as Starbound. Maybe this is just an extreme example, and usually this is not quite as bad of a problem? Let's see what happens if you try this in Rust! Right off the bat, things start to become hard. Let's go back to the simplest OO version in C++:
// typedefs...
;
// entity definitions...
;
Partially translating into Rust:
Even in this tiny example you can already see this is going to be a huge pain. First of all, World owns each Entity, but each Entity has a set of mutable methods, one of which must take a mutable reference to World. That won't work, because you will mutably borrow an Entity, and then have to mutably borrow it again to pass in a World reference (which would presumably contain the self Entity). In order to do this, probably every Entity implementation would need internal mutation because otherwise they would have no way to take a World reference, and also World would probably need internal mutation because it too would need to be passed by immutable reference.
Okay, now we won't have borrow errors during entity updates, but now everything has to be inside a RefCell? Maybe we can place our entire state inside a RefCell and it's not so hard? We've discussed the tendency for complex behaviors to be very cross-cutting, so what if we have the following situation:
A monster damages a player, this triggers the player to play an audio as a hurt sound effect, but for gameplay reasons this also triggers a logical sound that other creatures can react to. Maybe there is a swarming behavior for some particular monster type? So, hurting a player triggers an external mutation (audio), internal mutation (health), which then maybe signals monsters which have their own internal mutations (target entity). Maybe this all was started from a Monster, so the control flow goes from Monster to Player and back into Monster. If you're debugging Monster, you now have triggered spooky action at a distance by indirectly triggering mutation inside your own struct method. Except, rust doesn't allow this, (and this wouldn't even type check probably without RefCell), so in rust the version of this is that you simply get a RefCell panic.
Let's keep going, say entities have some kind of tag or dynamic set of tags, and let's add that to our C++ Entity interface:
;
Say you profile your game and simply returning a copy of List<Tag> is eating
up huge CPU time, so you change the interface to this:
;
This is assuming such a change is possible and not a use after free bug waiting to happen, which it often is. Well, Rust is supposed to save us from use after free, so what happens if you translate this to the Rust version:
I've made the lifetimes un-elided to make it obvious, but what this says is that
this method returns a single Vec reference that borrows the entire entity.
That's.. helpful but also not at all helpful if you want to later call any other
method that might mutate.  If this has internal mutation and is inside a
RefCell, this is also impossible and would have to instead return
std::cell::Ref.  Everything is hard, much harder than it is in C++.  You go
onto IRC to ask for help, you get the well meaning but potentially unhelpful
answer: "You're just still in the phase where you're fighting the borrow
checker."
It can get even worse. Entity is pretty sparse, even the real Entity inside
Starbound, but say you apply these same principles to World:
If you have a big enough World structure, even in the absence of things like
pure virtual interfaces (or in Rust, traits), you can still run into problems.
Should accessing one tile borrow the entire world?  Things would be much
easier if you could somehow borrow only part of the world and thus allow you
to mutate some other part, and that's exactly what you could do were all the
fields of World simply public.  Rust is quite good at splitting borrows of a
struct, but by design, it cannot do that for such an opaque method.  This would
be vastly easier if simply World was a struct with plain public members.  This
becomes MORE important the larger the structure is, and data hiding becomes less
important the more into the "application" level you are.  With games this
becomes worse and worse, because your games become more and more complicated
over time as you add features.  We're not trying to write an engine, we're just
trying to write a simple game directly in Rust, all of that data has to go
somewhere!
If you're writing a method like this, it's more obvious that you can make fields of a large compound structure public and split borrows will help, but imagine that you didn't design things this way. Imagine instead that you have two kinds of Worlds, one for the Server and one for the Client. This is precisely how Starbound works, and there is a large complex interface for a "world" that is common between the client and server.
Now it's impossible to "just allow the fields to be public" regardless of "OO" principles or not, just like Entity. More "fighting the borrow checker". OO principles guide you to having the loosest coupling you can, and a very common strategy for this is to, rather than having objects depend on each other directly, that they would depend on each other through a pure interface. Starbound is chock full of these, Entity and World are only two. They amplify borrowing problems by amplifying the amount you are forced to borrow!
I don't think I can completely fully justify everything I'm saying inside a 30 minute talk, this is really only scratching the surface, but hopefully I've given you at least an idea of where I'm coming from. I know that for some of you this is going to sound VERY subjective and possibly only sound like useful advice for a specific type of software (games). Also, for a lot of you, this is maybe boring and you already knew all this, but it's still hopefully helpful to see this as it pertains to an industry which not everyone is part of.
But, that being said, here are some of my takeaways:
- 
For games, OO doesn't really help, at all. The parts of OO that are useful which I listed above are fine, but at the point where you're not writing "library" code and are actually writing a game, data hiding is NOT generally useful, and just wastes a lot of effort and time. The more data your game has, the more it changes because you're constantly experimenting, the worse this gets. This might be old news to you! 
- 
Thinking about "objects" vs data types in a game sounds superficially appealing, but it's actually actively harmful. Most behavior doesn't "attach" to any data, and if you start thinking about things that way it can be hard to stop. Stop conflating the data representation of your game with the systems that operate on it! 
- 
Sometimes I think I would rather deal with a single 12k line mega-procedure than a tangled ball of objects. 
- 
Carmack quote: "Sometimes, the elegant implementation is just a function. Not a method. Not a class. Not a framework. Just a function." 
- 
Sometimes if you have to write a lot of messy procedural state changes, just being honest about it with long (maybe not 12k lines) procedures is the best strategy. Just be honest about the messiness of what's happening, hiding it in other functions or methods doesn't help, it only makes it more confusing when it goes wrong. If you can pull things out that are simple and pure functions, do it, but leave the messy procedural truth as it is! Another Carmack thing 
- 
I find it really easy and enlightening to think about the types of structures that determine the state of a game. You can often learn almost everything you need to know from taking a code base and just looking at all the type definitions. If you're looking at C++, if you just look at the struct members and their relationships and ignore all the code, this will often tell you more than looking at say, the names of functions. I spent a lot of time reading Haskell code and they had this pattern that's often mimicked in Rust where they would have a module called "Types" that just had... all the types that were required in some library. I kind of like that, and I think Rust is really great at this kind of programming. I'm being a bit cheeky here, but what tells you more, this interface: or this struct definition: ;Obviously a lot of the information here comes from the names of the types, and clearly SpatialHash2Dis some struct of its own with its own sub-types, but if you name your types well the point still stands, you see an obvious trail of bread crumbs to follow to understand what an interface may hide from you (That your spatial entity queries are probably going to be fast and you shouldn't expect a query to take linear time in the number of entities).
I asked if we could use OO to improve our UR-architecture, and surprisingly we largely can't. Useful things like the dot operator and data hiding to maintain invariants in library-ish code aside, the large scale structure of our game is not really helped by OO. We need some way to deal with the downsides of a single giant mutable public nested struct without making an OO mess. We're now approaching the ECS answer from the top down.
Back to the beginning.
Our "OO-architecture" was a complete bust in Rust, so let's start over with something we know will work, our "UR-architecture" and try to improve it. This is a pretty standard ECS transformation, and you may have seen this before. I've read a bunch of ECS explanations before and I know of one fairly good one, and lots of really confusing bad ones, and hopefully I can explain this in a nice way that focuses on the data representation, and is useful to Rust specifically.
Let's start from the beginning with our simplistic "Starbound", writing down what the state representation of our game is again, but this time in Rust. No methods, no functions, only the data types. This will be very similar to the Mario 64 example "UR-architecture" above, with a few small additions:
type EntityIndex = usize;
There are a few changes from the Mario 64 example. For one, we're trying to use Rust features like enums and data types like Option, rather than trying so hard to mirror C. The great thing about this, is suddenly the structure of our game state has less possible invalid states! For example, here if we had tried to use a unified "Entity" with a type code, like Mario 64, there would be invariants that had to be kept manually stating that certain fields were in sensible states depending on the type of the entity. Now, many of these invariants just fall naturally out because sum types are so great. This is interesting, because it makes encapsulation less important, even with everything public, there are certain invariants that cannot be broken. There ARE invariants that aren't expressible in this way though, an example being that the array of player ids probably need to point to Player type entities only!
The entities: Vec<Option<Entity>> is interesting as well.  Since some of our
entities keep indexes to this array as "pointers" to other entities, it makes
sense that we should never move an entity in the array.  If we allocate a bunch
of entities then remove the first entity we allocated, instead of moving every
entity, we need to set it to None instead so that the rest of the array stays in
place.  When allocating, we probably would go through the array looking for the
first None empty slot, and if none are found, push a new entity onto the end.
This is interesting, because this is very similar to how the "Mario 64" example
worked with a static array.
Other than these changes, this is not really a huge departure. All of the game state is still morally global, and each system is still potentially large and procedural. Here's the thing though... I actually don't think this is very bad at all? In all honesty, if I were doing a game jam game, this might be how I would write my game!
I wouldn't write low level graphics or audio code, but if I were doing a simple 2D game or very simple 3D game, this would work for me. My game state might be a pretty complex struct full of other structs, and I might have many "systems" (which remember here are just regular functions) in different files, and this would be fine. I would choose a very simple graphics API with as little state as possible, to reduce as much as possible the headache of "loading" and "unloading" graphics data and the same with the audio API, and this would be the structure of my game. If I had to use a very messy and stateful graphics or audio system, I would pre-load everything when creating the game state initially, stick all of it inside some Assets structure, and that would be it. There are plenty of high level level graphics and sound APIs for Rust that actually make this really easy.
But, this pattern is obviously not perfect, even other than everything being
globally public.  For one, there's a lot of repeated data in each of our entity
types, e.g. with Physics being repeated across Player, Monster, and Npc.  In
our physics_system function, probably there is some commonality in this system
which is broken down further into another function, but the physics_system
definitely for sure has to understand that there are 3 separate entity types
that all have physics.  Whenever we add an entity type, probably this system has
to change its implementation, and honestly probably very many systems have to
change when you add entity types.
Also, as we add more entities, we'll probably start to find out that a large part of them will be repeated over and over. Maybe "Monster" is too specific, and you split this up between "FlyingMonster" and "GroundMonster", but they share 80% of the same fields, and in turn both of those share 50% of their fields with Player.
This is in some ways better than having all entities be one unified type, but in some ways actually it's kind of worse? It's not terrible, but let's see if we can do better. Let's go back to having a unified Entity type like the Mario 64 example:
type EntityIndex = usize;
// All the different types of fields that an Entity can have, grouped somewhat
// logically...
// An entity is a collection of all the possible entity fields, we let every one
// of them be optional.  In this case, we have lost some type safety, because
// this can express more invalid states than our previous example, some of these
// combinations probably don't make sense.  Also, maybe right now it doesn't
// make sense for an entity to be missing a position, so all entities have to
// have physics even though it's optional here.
Okay, so this is interesting.  It's clear that we lose some invariants here in
that more potentially invalid entities could be created, but there are some big
advantages here.  One that stands out is that our implementation of
physics_system is probably vastly simplified, just loop over all the entities
and mutate the physics field if they have one.  This is much simpler than
looping over the entities and having to match on the entity type.
There are still structures that are only valid for each of our "logical" entity types though, and clearly it probably wouldn't make sense for something to be both an NPC and a Monster at the same time, so this is I guess another invariant to keep. It's interesting though, because the amount of per-"type" data got smaller. Let's change this a bit more and separate out more fields:
// Just for symmetry, let's let Health be a struct type
;
So there are lots and lots of ways to express the fields that may go into each
entity, and they all have their advantages and disadvantages.  In this case,
after transforming our field types this way, a couple of things become
expressible that weren't before!  For example, since we've broken out the
food_level field from Player and called it Hunger, we can now express entity
types that aren't Players that are hungry, so now we can describe NPCs with
hunger.  Also, after breaking out Aggression, we can express hostile NPCs!
However, maybe we can express something with monster-type animation that also
can carry humanoid items, and maybe this is actually logically invalid, so there
is a give and take here.  Still, this is interesting, and solves the problem
where there is much commonality between our entity types that requires repeated
code to express or extract from monolithic enums.
Now, it may be very clear where I'm going with this already, but this pattern, where entities are composed of one or more of a set of named parts, and they are specified a la carte, is very common. In fact, these parts are normally called "components"! We now have all of Entities, Components, and Systems, so if you squint this is everything needed to be an "ECS" system, but this is actually very simple! By focusing on the data representation of our state, it is really not very many steps in between this and what we started with.
Let's make one more actually sort of inconsequential change, but this will help us in a second. Also, we'll start calling all of these pieces we've been defining "components" just to drive the point home:
type EntityIndex = usize;
This is the classic "array of structs" to "struct of arrays" transform. It should be clear that (nearly) exactly the same information is expressible between this and the previous representation. There are a few new invariants to maintain, namely that each component Vec is the same length, but there's not really new information. When you just write out types like this, these kinds of changes actually look pretty simple, instead of in our OO architecture where something like this might be considered an impossible refactor. Obviously we'd have to change the systems a lot here too potentially every time we made these changes, but there's value in being able to think about the data independently.
The transformation to "structs of arrays" is generally the thing that ECS introductions really focus on, and then they start talking about performance concerns and cache behavior and it can be a little confusing and overwhelming. They're not wrong, but I really don't think this is the important part. This is why I've introduced ECS this way, because I think a lot of explanations miss this point, that from a certain perspective, these changes are actually pretty boring and normal. You might have this structure or you might have the previous structure, maybe one is better performing than the other, but you have not radically changed programming paradigms by choosing one vs the other. From a data-oriented perspective, these things are not earth shattering, they're just performance optimizations. There are lots MORE potential optimizations you can do here as well, but I think we've reached the point where we have a bare-minimum that somebody might call an "ECS system".
There are some takeaways here for Rust users in general, not just game developers:
- Thinking about JUST the structure of state is really powerful, often times "methods" and "objects" get in the way. Nobody really thinks that having methods on your structs is "too much OO", but it DOES couple your procedures to your data, even if only organizationally. Both data structure design and simple module organization are important, but don't conflate them! I'm convinced that "ECS design" in games is so great because it forces you to think about your data rather than be stuck in OO mode, not because of the magic of "structs of arrays". This is a lesson that can be applied to Rust in general.
- You can do a LOT with indexes into Vecs.  This is way easier than self
borrowing or Rc<RefCell>. 
Okay, let's stop for a second and talk about EntityIndex.  I genuinely think
that most of the time when you find yourself running into self borrowing with
Rust, plain Vecs and indexes should first tool you reach for.  There are OTHER
tools, like various arena crates, rental etc, but I think that generally you
should try Vec first.  Generally these problems happen when trying to represent
some kind of graph structure (and an ECS is just a really flat graph... kind
of), and even in C++ this is the advice generally given!  (See Andrei
Alexandrescu's performance talks, his favorite data structure in the world is
std::vector, "just use vector!")  Things like Arena allocators are great and
actually quite performant, and there are versions that work with multiple types,
but they can't be 'static without the addition of self-borrowing.  Self
borrowing solutions like rental are tools of last resort.  It is not worth it,
move on with your life, and just use Vecs and indexes.
THAT BEING SAID, they have do have some problems. In our entity examples, we've been glossing over the question of how to find "free" EntityIndexes, and how "deletion" works, because it's not actually terribly great. The cost of allocating an "entity" is not constant because we have to scan through a Vec looking for free entries. Deleting an entity is cheaper, but it has bad properties. We can delete an entity which frees up a slot in a Vec, but then it's possible that the very next allocated entity will use the same index. This is fine if we make sure that there are no outstanding "index references" to this entity before deletion, but what if we mess that up? We'll get a "random other entity" in its place without being able to tell that it was in fact removed out from under us!. Also, both of these situations are now much worse after the "struct of arrays" transform, because we now have multiple arrays instead of one and how we manage "entity indexes" is kind of up in the air now. We're going to solve both of these problems at once.
Generational indexes are awesome.
So this is one of my favorite patterns that I'm not sure is widely known in the Rust community but I believe is widely known in the gamedev one.
We've been using EntityIndex to identify and look up entities stored inside a
Vec.  This is very similar to the pattern I used in the C++ mini-Starbound, I
used an EntityId which was an increasing integer id to store entities like
this:
HashMap<EntityId, shared_ptr<Entity>> entities;
Both are abstract mappings from integers to some value.  In the Vec case, it is
basically the fastest POSSIBLE data structure, indexing into an array, and in
the case of HashMap it is much slower, but the HashMap is much more
flexible!  For one thing, there is only an upper bound really on the total
number of entries, not on, say the largest size of the index.  HashMap works
just as well if you start the keys at value 1,000,000,000 or 1, certainly a
billion entry Vec is different than a 1 sized one.  This matters for the
deletion problem above.  If we indexed using a HashMap and a key type of u32
or even u64, we could just have new indexes always increment (with eventual
wrap-around).  In this way, you could more or less guarantee that no index would
ever be re-used, or at the VERY least it would be a "very long time" before an
index would be reused.  In the u64 case, if you ever re-used an index, your
game technology probably has "cloud" in the name, so you might have bigger
problems anyway :).  BUT HashMap is much slower than Vec :(
Is there a way to get this property with integer indexes into a Vec? There is, and it's called "generational indexes"!
Instead of using just an integer index, we make a "generational index" type like this:
// You can use other types that usize / u64 if these are too large
Then, we make something called an "GenerationalIndexAllocator":
(There are faster ways to implement this, but this one is actually pretty much fine. Also note that this is an obviously GOOD use of data hiding).
The basic idea is that you can "allocate" indexes for a vector just like if you
had Vec<Option<Entry>>, but you will instead never re-use indexes.  It works
like this, you allocate an index and get back a GenerationalIndex with real
index 0, it will also have "generation" 0.  If you delete that index, it will go
into a pool of free indexes, so next time you allocate an index you might get
back another generational index with real index 0, but crucially the generation
will now be 1.  Generational indexes are never re-used because the generation
will always increment, yet the "real indexes" will always be "small", on the
order of the largest total number of entries ever allocated.  This way, you can
use fast indexing into a Vec without many of the bad "pointer-like" properties
of simple indexes!
I said this pattern is not widely known inside the Rust community, but that's at least a bit of a lie because there is a recently released crate built on this idea called "slotmap" and it's great! However, it's missing a CRUCIAL feature for our example, which is that in "slotmap" you can only allocate indexes for a specific SlotMap, you can't allocate indexes and re-use them for different SlotMaps. Useful, but it would be vastly MORE useful if these concepts were separate, which is what we've sketched out here. Indexes into a Vec are already a great pattern when encountering "self borrowing", and generational indexes make it much much better. "slotmap" beat me to releasing a crate for it :(, so consider this a feature request :)
We'll go ahead and make a type that's a bit easier to use than Vec<Option<T>>
to store our actual data as well:
// An associative array from GenerationalIndex to some Value T.
;
So with this new abstraction, let's change our engine some more:
// We're dropping the index or id suffix, because there is no other "Entity"
// type to get confused with.  Don't forget though, this doesn't "contain"
// anything, it's just a sort of index or id or handle or whatever you want to
// call it.
type Entity = GenerationalIndex;
// Map of Entity to some type T
type EntityMap<T> = ;
Neat! We're in the home stretch now, this is almost a full blown ECS system.
Take away for users of Rust in general: obviously generational indexes are awesome but for some reason they're more popular only in C++, but they're potentially even MORE useful for Rust! To the author of "slotmap", please expose the allocator separately, it's hugely useful! If that doesn't seem in scope or the kind of API you want to provide, I can release a similar crate with the version I have that does do this.
Dynamic typing is actually kind of nice in VERY controlled quantities.
Okay, we're really close now, this is very close to how a "real" ECS system (like specs!) might work. The biggest problem we haven't addressed from earlier is still that everything is kind of global still. More so, every "system" (for us, this is still just a fancy name for plain functions) depends on all of the types that go into our game state, which may be quite large. Most of a game is going to live inside one crate and the dependency graph between modules such as this is really hardly anything to fret over, but still changing anything inside "GameState" potentially would affect every system at least theoretically. Let's see what we can do about this?
I want to emphasize before we go on that this, like many steps before, is optional. You may balk at this as unnecessary complication, and you may be right! This IS something that is in most ECS implementations though, so it's worth covering if only to understand them, and it's more or less unavoidable when trying to build a library to do this sort of thing.
For this we're going to need the anymap crate, but also the mopa crate would
do as well.  What we need is a container that can store exactly one of every
type that we put into it:
How might we use this to store our components?
type Entity = GenerationalIndex;
type EntityMap<T> = ;
Now instead of storing game-specific data, let's just keep going with dynamic
typing!  We'll say that our game state is a dynamic collection of entities with
components, and also a dynamic collection of other types, one of each type.
We'll call these "resources".  Also we'll change the name from GameState to
something more accurate.
type Entity = GenerationalIndex;
type EntityMap<T> = ;
And here we have arrived at what an actual ECS data structure might look like. I've added a "resources" AnyMap because this is also a very common pattern, and it means that more or less your entire game state can be expressed inside this "ECS" structure. We're calling it an "ECS", but importantly, really there aren't any mentions of "systems" in sight. I don't actually like describing ECSes in terms of systems, because I think while it can be important, it's really incidental. If our "systems" are pure functions in a loop or they're something much fancier, BOTH of these capture the important part of ECS design.
So, you may be balking now at the egregious and sudden introduction of dynamic
typing.  Let's stop and think though what this buys us.  Say you get a new
feature request for your game, say you need a new crazy special monster that has
some kind of counter inside it.  Every time you kill the monster, it copies itself
into two and decrements the counter, duplicating like the heads on a hydra.
This means you might need a new component type, say EnemyDuplicationLevel or
something.  With dynamic typing, you can add this component without "disturbing"
your other systems, because without importing the new module, they can't
possibly "see" that the ECS has such a component anyway.  The same is true for
resources, you can add new data types to your model without "disturbing"
existing systems.
The justification for this may seem pretty weak, and it kind of is. In order to get the full picture we need to go a bit further. I'm going to speed up a bit so we can get to the end and I can show a full-ish picture, something which I consider a "modern" game engine design that you might use for a medium or large project. It's actually not much further along than this, but the last few features are all related and are not as useful alone.
The "registry" pattern
Now that we've introduced dynamic typing, there's a design pattern that I like in Rust that I haven't really seen in practice yet (I'm sure it exists and I have just not seen it). I'm going to call it the "registry pattern".
In ECS implementations like specs, there's a step where you "register" a type with your ECS, which inserts an entry into some AnyMap or equivalent to an AnyMap. Using a component type that is unregistered is usually an error. Let's take this a bit further, and not tie "registering" to the ECS per se, let's make our own "registry".
We'll also make one for "resources"
Then, we'll tie them together in one big global constant with lazy_static!
// When we add a component to our project, there are two steps.  First, add the
// component somewhere as a Rust module, THEN add it to this list here.  For
// added convenience, this function could go in the lib.rs which contains the
// component modules themselves.  If you were very fancy, you could have some
// kind of "plugin architecture" for this as well, grouping related components /
// resources together into "plugins".
// Ditto 
lazy_static! 
I like this pattern, because this is, if you squint, almost like a sort of global type registry like you might find in Java. It feels a bit "enterprisey", but it's quite useful! It is also similar (again, if you squint) to very full-featured all encompassing engines with built-in editors, where you might say add a component type through some menu or GUI and it ends up being stored in some project config file. In this case, instead of a project config file it's simply a bit of auxiliary rust code.
This pattern ends up being very useful! Imagine you have some game state that can be loaded from a JSON config file. Each resource and component can be loaded from JSON, so you can just add a component or resource type and then add the entry to the data format, load the game and see it (except you need a new system that understands the new types, we'll get there in a sec).
So at this point, this is a sketch of something that is unmistakably turning into a "real" ECS game engine. I like this, because so far, except for some brief required library-ish functionality, I have barely talked about functions or systems! The reason for this is, I don't like introducing this concept by talking about behavior, I really think that thinking just about how we describe our state is a much more useful way to approach this.
With that out of the way, it's pretty easy to see how you could also just as
easily add a SystemRegistry to go along with this.  Our systems are just
functions, so all this would do is allow you to add functions to your main loop
in a slightly more complicated way.  One possible addition here is to allow
systems to have some kind of config, so that maybe you can configure tunable
parameters for them or even use the same function multiple times with different
parameters.  It's generally advised though that you try very hard not to give
your systems state, so you can limit your game state to just components and
resources.  If your game state is just components and resources, you can often
do cool things like clone them (save state like in an emulator!) or easily
serialize it, and this is harder with something like system closures.
Takeaways for regular Rust: The registry pattern is actually quite nice. In Starbound there is a "Root" object which is sort of like this, where every type is registered, but it's more stateful than what I've described here. It's read only, but it's constructed by reading assets. I actually like the idea of a "type registry", and something like this is necessary as soon as you want to use this sort of dynamic typing with AnyMap, and the two patterns go together well to limit the problem of "everything depending on everything else".
ECS is SQL for games
Let me pull the curtain back a bit. What have we described so far: A way of declaring a special kind of key (Entity), and a series of records that go with those keys (components), and also for good measure a way to define records that don't pair with any of those keys (resources). We've added dynamic typing into the mix, and shown a pattern so that you can define all the record types you have in one place (kind of like a schema?). In the real ECS I have built, I even have ways of performing primitive "queries" on components, saying "give me all entities with a position, optionally a velocity, but NOT a mass".
This probably sounds achingly familiar, and if it does there's a good reason for it. ECS is just SQL for games. A very, very, very limited form of SQL, but spiritually very similar. Define the schema for your data, load it, run queries on it, and update it. Very limited SQL where every query might have to run in no more than a few microseconds.
I said earlier that I've read a lot of bad introductions to ECS and one good one. I'm not the first to mention this parallel between ECS and SQL, all the way back in 2009, there was an article about this very point with a focus on MMOs.
So, if this is true, why can't we just use something like sqlite to store our
game state?  The funny thing is, you might actually be able to do this, but it's
tough to run queries fast enough in 16ms.  Let's think about what kind of
benefits this would bring, though!  Oh man, I need to make a save format for the
game.. nope it's SQL I'm already done.  Oh, I need to update the format of
everyone's save file.. write some external SQL update scripts and you're done!
Oh, I'd like to give the ability to save the game's state at any arbitrary
point?  Done.  I noticed a bug and I want to roll back the game state to 30
seconds ago and step through it frame by frame, running queries to see when a
bad state was produced?  done. All this careful introduction of the generational
indexes concept, SQL has this already, they're keys with auto increment (more
like HashMap<EntityId, T> really).
At that point, each of your systems would basically be a series of queries to get data out of SQL, some updates, and then writing it back out to SQL. We've turned sexy game development into boring old web development! (web development can be sexy and game development can be boring, just for the record)
The only issue is that I'm pretty sure it's about two orders of magnitude too slow to work, but still it's a neat idea, and it's useful to draw parallels!
I think this is important, because I see a lot of debate about "what makes something a pure ECS" that I think is ultimately pretty silly.
- Is it an ECS if entities can have multiples of a component? Well, the differences between multiples of a component and a component that contains just a Vec are very very few, but yes it's still ECS. Is it still SQL if you have two tables that are 1 to N instead of 1 to 1? Yes, of course it is.
- Is it still an ECS if my components have methods? Yes, adding a component method is not the end of the world, especially if there is a small local invariant to maintain. SQL is all about data, but it also has stored procedures to maintain invariants.
- Is it still an ECS if I need two different ECS sets, or I put lots of data into resources? Should I be using "singleton entities" or resources? Is it still SQL if I have two different databases, or split a top level table apart? Yes, of course!
... and so on
The analogy is not perfect, but I think this is really kind of illuminating. There is research currently about how to make more capable ECS systems that have things like component graphs with child relationships between components and entities and all kinds of crazy stuff, and the thing is that's of course fine. SQL has all of this already, it's an entire language and set of software for expressing all kinds of data relationships, it's just that games have generally simpler needs, will basically always be in-memory only, and have timing requirements on the order of nanoseconds or microseconds, so we compromise and make new tools.
This is the other reason I'm doing this talk, to help further de-mystify "ECS" and show how we ended up here, and to give perspective.
The one place where it all goes wrong
(NOTE: I don't know if I need this section at all, I'm just hawking the OTHER talk I wanted to do. Maybe this is unimportant? This talk is already potentially very very long)
I claimed at the beginning that the attitude I see of being overly skeptical of the borrow checker, of considering it too restrictive, is bunk. I still absolutely believe this, but I want to talk quickly about the one place where I kept running into issues, really the only place where I really had trouble, where I constantly felt limited. Certainly as a project grows in size, something must go wrong right? Well, I have one example of this, though this problem isn't really limited to Rust at all, it's just more immediately painful (noticing a pattern?).
Language boundaries are hard.
So, the issues here are all... pretty hard to explain succinctly. I'm not sure I can come up with small limited examples like before without getting way into the weeds and doubling the length of this already lengthy talk. I wrote an entire crate (rlua) just to try and solve this problem, and after a ridiculous amount of work I can confidently say that I've sort of half-solved it.
I have a representation of components where you choose to read or write each component (they're stored in RwLocks for system concurrency). I want Lua to be able to read a set of systems, so I have a lua script perform a "query" on the ECS store. All I need to do is to be able to hand Lua the "RwLockReadGuard" (really a structure containing this) that the query returns, except... RwLockReadGuard is not static. Oh I guess I'll just have the Lua query lock inside the query API rather than once at the beginning... well that would be very slow and also not thread safe, a query should lock for the whole time. I guess I could use the rental crate.. oh god the rental crate is HARD and this is TERRIBLE. (I was later able to solve this problem with rlua's "scope" system, but it's still bad).
It's best if you don't have your systems be stateful, right? Generally systems should store data either in the components they operate on if they're logically related, or maybe in a custom resource, or simply store some cache-like values but be prepared if the system is reloaded and the cache is reset. Don't rely on data that can't be saved / restored / serialized etc. This is great until you write a Lua system, because exfiltrating values outside of Lua is actually really hard. You can't store a Lua state inside a resource because it's not Sync, and you can't store Lua data separate from the state because the Lua internal state and external handles are REALLY not Sync. You can sort of solve this with magic Lua registry keys, but then if each system has its own Lua instance, you've made a footgun if you ever try to use Lua values in the wrong context, and they're not serializable anyway.
You can make a data type which is limited to just data and not an internal Lua type, but then you must write marshalling between it and the non-lua representation, so every time your script system reads or writes this storage it's dog slow. Some of the slowness is from copying data, but most of it is from the intrinsically slow Lua API.
Everything is hard, it works if you really try but you feel like it should be simpler than this. It's harder even than the equivalent in C++, but the equivalent in C++ is actually just as bad, you just notice way faster in Rust. At least there are not constant crashes and shared_ptr cycles now?
Seems a bit similar to our situation before, where we were trying the wrong approach with OO. Unfortunately, in the case of rlua, there are pretty hard limitations based on how the internal Lua C API works, and it might be close to the best I can offer.
Language boundaries are hard, especially between languages that have very different sets of limitations like Rust and Lua (or C++ and Lua). Especially in the presence of a garbage collector in the contained language, if the garbage collector in the contained language ever makes contact with one in the host language, nothing will work, nothing will get collected. (In C++ / Rust this is shared_ptr / Arc, yes these are a sort of garbage collector).
I mention this because I think Lua is very popular for games, and I get questions a lot about how to combine rlua with things like specs, and I think it's just really hard, and goes against the larger point I'm making with this talk. I think this is probably uniquely hard due to language boundaries being so tricky, and the situation is not vastly worse or better than the one in C++, it's slightly more difficult and vastly safer, just slightly more difficult is not a great answer when in C++ it's still very difficult.
I also mention this because I might have an answer for this soon! The OTHER talk I thought about doing was describing what I believe is a novel way of implementing language runtimes safely in rust that have garbage collection that is zero cost, and comes with the ability to have a fast, mostly pain-free (or as much as possible) bindings experience.
In the meantime, think really really hard before you add a scripting layer to your game engine. The problem is, I LOVE scripting layers in game engines (for modability and many other reasons), so I do this anyway, but it is not a decision to be taken lightly and it can eat up a lot of time and effort. I want to make this more painless though, and make a safe, fast Lua that feels at home in Rust in the same way that PUC-Rio's Lua feels at home in C. I will talk about this in the future!
Summary
I had multiple purposes with this talk:
- To walk through the design of a medium-scale rust project for those with only experience in small-scale ones.
- To show some examples of traps people might fall in that cause them to "fight the borrow checker", and to help make more concrete how to move past this phase, at least in part.
- To show how awesome data-oriented programming is and how well of a fit it is for games and for Rust
- To show how patterns in game dev that appeared long before Rust, when applied to Rust, still seem to be a good fit.
I talked a lot about patterns that in some cases were hard fought in other languages, only appearing after laboriously exploring the space of solutions that DIDN'T ultimately work. Interestingly, those solutions that ultimately don't work very well often times are painful more quickly in Rust, they're louder, MORE annoying. I think this is a good thing, and I like having tools that make bad patterns feel as bad as they are. Rust is great at this.
But could you take most of the lessons here and apply them just as easily in C++? In C? Yes, absolutely. Even if you're the sort of person that doesn't make these mistakes even when there isn't this sort of language pressure to avoid them (you're way better than me), there are tons of other great benefits. I didn't really get a chance to talk about all the benefits of Rust or talk about my experiences with Rust in general, because I only have 30 minutes and this is probably way longer than 30 minutes at this point.
