The following rules are updated constantly. When we update a rule, we do not update the entire codebase, so you can expect to find existing code that violates these rules. We are thankful if you take the time to update the existing code to the current set of rules. If you do that, it would be helpful for code reviewers if you did the code style changes to existing code in a separate commit at the top of your branch, titled "Cleanup: code style".
- User lowercase for top level namespaces:
mdl
,render
etc. - Use camel case for class names:
MyUsefulClass
- Functions, methods, variables and parameters use camel case and begin with a lowercase
character:
void myUsefulFunction(const int aHelpfulParameter);
- Private member variables are prefixed with
m_
- Constants use camel case and begin with an uppercase character:
static const int ThisIsAConstant = 1;
- We use
clang-format
to format the code. The formatting rules are in the.clang-format
file at the root of the repository. Correct formatting is enforced by CI. Check out the dev-tools repository to get the correct binaries ofclang-format
for your platform.
- We use clang-tidy to help with certain style rules. The style rules are in the
.clang-tidy
file at the root of the repository. These rules are enforced by CI. - Avoid header files that declare more than one class
- Class members are usually ordered as follows, but it is ok to deviate from this if there is a good reason.
- Type aliases and static const members.
- Member variables
- Constructors / destructors
- Operators
- Public member functions
- Protected member functions
- Private member functions
- Extension interface (private pure virtual member functions)
- We follow the "almost always auto" style, so we use
auto
wherever possible except doing so would make the code very awkward. These exceptions are very rare. - Use
auto*
when declaring a pointer variable, e.g.auto* entityNode = new EntityNode{...}
. - Use
const
wherever possible. - We follow a left-to-right style for declarations. For example, when declaring and
initializing a variable, we write
auto entity = Entity{...};
instead ofEntity entity{...}
. - Use private namespaces for implementation details in cpp files. If possible, avoid private member functions for helper functions - prefer free functions in an anonymous namespace in the cpp file for this.
- We prefer
std::optional
over magic constants such as usingNaN
to signal the absence of afloat
value. - We prefer using
std::variant
over inheritance. - We prefer
struct
s overclass
es for simplePOD
-like types. - We prefer using value semantics.
- We use C++17.
- The entire source code and test cases must compile without warnings.
- Everything that can be const should be const: methods, parameters and variables.
- Move semantics should be used if possible. Prefer passing objects by value if a function takes ownership -- the caller can decide whether to move or copy the object at the call site.
- Use RAII where possible.
- Avoid raw pointers unless they are confined to a method, class, or subsystem. Don't return raw pointers from public methods. Favor references and smart pointers over raw pointers.
- Don't use exceptions -- use
Result
andError
instead.
This section presents some guidelines to keep compilation times low.
Avoid including headers in other headers. Remember that including a header B in another header A includes B in every file that includes A, and so on.
- Use forward declarations wherever possible. Remember that you can forward declare classes, class templates, and scoped enums. Furthermore, type aliases and even template aliases can use forward declarations.
- If you need to refer to a type in a header file, you should use a forward declaration
except for the following reasons:
- It's a typed defined in namespace
std
. - The type is used by value in a member variable declaration, or a function declaration.
- The type is used in the implementation of a template function.
- It's a typed defined in namespace
- Note that you should also use forward declarations for types used in the following contexts:
- Type parameters for standard library containers - if you are declaring a member of
type
std::vector<Foo>
in a class, you need not includeFoo
in the header file where your class is declared. You may have to declare a destructor which you must implement in the corresponding cpp file, whereFoo
must be fully defined. Defaulting the constructor in the cpp file is sufficient.
- Type parameters for standard library containers - if you are declaring a member of
type