monado/doc/conventions.md
2023-06-20 13:30:19 +00:00

232 lines
12 KiB
Markdown

# Code Style and Conventions {#conventions}
<!--
Copyright 2021-2022, Collabora, Ltd. and the Monado contributors
SPDX-License-Identifier: BSL-1.0
-->
<!--
NOTE to editors: To avoid stale references, make sure to mention relevant names
using markup like @ref xrt_device so that Doxygen tries to parse and link names.
This will result in Doxygen warnings if we change the name of something
mentioned in these examples.
-->
Here are some general code style guidelines we follow.
Note that we aim to "code with respect", to avoid terminology that may limit our
community or hurt those in it, as well as to conform with emerging industry
standards. Good guidelines to look to include the
[Android Coding with Respect][] policy and the
[Write Inclusive Documentation][] page from the Google developer documentation
style guide. The latter also links to a word list for clear documentation,
which, while not binding on this project, is useful in making sure your code,
comments, and docs are understandable by the worldwide Monado community.
[Android Coding with Respect]: https://source.android.com/setup/contribute/respectful-code
[Write Inclusive Documentation]: https://developers.google.com/style/inclusive-documentation
## Changelog fragments
In Monado we strongly prefer if all MRs merged into the main Monado repository
also includes changelog fragments. A changelog fragment is a small file
detailing the changes in the MR on a per area basis. They are slightly more
detailed then a commit subject line, but usually just one or two lines, usually
providing a little bit of context to the change. Sometimes for "big" changes
they are more detailed and provide paragraph of description, such as for adding
of new drivers, or large refactors. The changelog fragments are used to generate
the @ref CHANGELOG file, updated on each release (and running on the CI). The
changelog is generated with [proclamation][].
The changelog fragments are located in the `doc/changes` folder, organised into
sub-categories in subfolders. There isn't a 1-to-1 mapping of changelog fragment
to commit, but instead they are per change. A changelog fragment file is named
`mr.` + MR number + `.md`, for MR 1234 it would be named `mr.1234.md`. If a MR
has multiple changes for one sub-category a number is added between the MR
number and file extension, example `mr.1234.1.md` and `mr.1234.2.md`. If a
change spans multiple MRs, such as fixing a feature introduced in a earlier MR
we imply the use if YAML headers to mark a changelog fragment applying to
multiple MRs, can also be used to link issues.
```
---
- mr.1234
- issue.42
- mr.2000
---
Add cool feature X, it gives the answer to life, the universe and everything
else.
```
Generally the last commit in a MR adds the changelog fragments, as unfortunately
MR numbers are allocated when opened, also provides a nice readable separation
between MRs in the git history. Examples for commits adding changelog fragments
can be seen [here][example1], [here][example2] and [here][example3].
[example1]: https://gitlab.freedesktop.org/monado/monado/-/commit/5e0f0866a6f74116acbc46c6e2447fdb8c716d02
[example2]: https://gitlab.freedesktop.org/monado/monado/-/commit/98a5b18e0f90dab9f2ea5c2bbfd4ccd4998121c4
[example3]: https://gitlab.freedesktop.org/monado/monado/-/commit/785e99f115df87dd4561fe6f88a7988b5834b650
[proclamation]: https://gitlab.com/ryanpavlik/proclamation
## APIs
Internal APIs, when it makes sense, should be C APIs. Headers that define
general communication interfaces between modules (not only use of utilities)
belong in the `xrt/include/xrt` directory, and should not depend on any other module outside
that directory. (As a historical note: this directory gets its name from a
compressed version of the phrase "XR RunTime", a generic term for Monado and an
early development codename. Also, it's shorter than `monado_` and so nicer to
use in code.)
What follows are some basic API usage rules. Note that all the module usage
relations must be expressed in the build system, so module usage should form a
directed-acyclic-graph.
- Any module can implement or use APIs declared in `xrt/include/xrt`
- Any module (except the `xrt` interface headers themselves) can (and should!)
use APIs declared in `xrt/auxiliary/util`.
- Any module except for `auxiliary/util` and the `xrt` interface headers
themselves can use APIs declared in other `xrt/auxiliary` modules.
## Naming
- C APIs:
- `lower_snake_case` for types and functions.
- `UPPER_SNAKE_CASE` for macros. e.g. @ref U_TYPED_CALLOC (which is how all
allocations in C code should be performed)
- Prefix names with a "namespace" - the library/module where they reside. e.g.
@ref u_var_add_root, @ref math_pose_validate
- Related: only things prefixed by `xrt_` belong in the `xrt/include/xrt`
directory, and nothing named starting with `xrt_` should be declared
anywhere else. (Interfaces *declared* in `xrt/include/xrt` are
*implemented* in other modules.)
- Generally, we do not declare typedefs for `struct` and `enum` types, but
instead refer to them in long form, saying `struct` or `enum` then the name.
The exception to not using typedefs is function pointers used as function
arguments as these become very hard to both read and type out.
- If a typedef is needed, it should be named ending with `_t`. Function
pointer typedefs should end with `_func_t`.
- Parameters: `lower_snake_case` or acronyms.
- Output parameters should begin with `out_`.
- Of special note: Structures/types that represent "objects" often have long
type names or "conceptual" names. When a pointer to them is passed to a
function or kept as a local variable, it is typically named by taking the
first letter of each (typically `_`-delimited) word in the structure type
name. Sometimes, it is an abbreviated form of that name instead. Relevant
examples:
- @ref xrt_comp_native_create_swapchain() is a member function of the
interface @ref xrt_compositor_native, and takes a pointer to that
interface named `xcn`. It creates an @ref xrt_swapchain, which it
populates in the parameter named `out_xscn`: `out_` because it's a
purely output parameter, `xscn` from @ref xrt_swapchain_native
specifically the letters `Xrt_SwapChain_Native`. @ref xrt_swapchain and
related types are a small exception to the rules - there are only 2
words if you go by the `_` delimiters, but for clarity we treat
swapchain as if it were two words when abbreviating. A few other places
in the `xrt` headers use `x` + an abbreviated name form, like `xinst`
for @ref xrt_instance, `xdev` for @ref xrt_device, `xsysd` sometimes
used for @ref xrt_system_devices.
- `create` and `destroy` are used when the functions actually perform
allocation and return the new object, or deallocation of the passed-in
object.
- If some initialization or cleanup is required but the type is not opaque and
is allocated by the caller, the names to use are `init` and, if needed, one
of `cleanup`/`fini`/`teardown`. (We are not yet consistent on these names.)
One common example is when there is some shared code and a structure
partially implementing an interface: a further-derived object may need to
call an `init` function on the shared structure, but it was allocated by the
derived object and held by value.
- C++:
- Where a C API is exposed, it should follow the C API naming schemes.
- If only a C++ API is exposed, a fairly conventional C++ naming scheme is used:
- Namespaces: nested to match directory structure, starting with `xrt::`.
- There are no C++ interfaces in the `xrt/include/xrt`, by design, so this
is not ambiguous.
- Place types that need to be exposed in a header for technical reasons,
but that are still considered implementation details, within a
further-nested `detail` namespace, as seen elsewhere in the C++
ecosystem.
- Types/classes: `CamelCase`
- Methods/functions: `lowerCamelCase`
- Constants/constexpr values: `kCamelCase`
- If a header is only usable from C++ code, it should be named with the
extension `.hpp` to signify this.
- Math:
- For different types of transforms `T` between two entities `A` and `B`, try
to use variable names like `T_A_B` to express the transform such that `B =
T_A_B * A`. This is equivalent to "`B` expressed w.r.t. `A`" and "the
transform that converts a point in `B` coordinates into `A` coordinates".
`T` can be used for 4x4 isometry matrices, but you can use others like
`P` for poses, `R` for 3x3 rotations, `Q` for quaternion rotations, `t` for
translations, etc.
## Patterns and Idioms
This is an incomplete list of conventional idioms used in the Monado codebase.
### C "Inheritance" through first struct member
Despite being in C, the design is fairly object-oriented. Types implement
interfaces and derive from other types typically by placing a field of that
parent type/interface as their first element, conventionally named `base`. This
means that a pointer to the derived type, and a pointer to the base type, have
the same value.
For example, consider @ref client_gl_swapchain
- Its first element is named @ref client_gl_swapchain::base and is of type
@ref xrt_swapchain_gl - meaning that it implements @ref xrt_swapchain_gl
- @ref xrt_swapchain_gl in turn starts with @ref xrt_swapchain_gl::base which is
@ref xrt_swapchain - meaning that @ref xrt_swapchain_gl **extends** @ref
xrt_swapchain. (Both @ref xrt_swapchain_gl and @ref xrt_swapchain are abstract
interfaces, as indicated by the `xrt_` prefix.)
Structures/types that represent "objects" are often passed as the first
parameter to many functions, which serve as their "member functions". Sometimes,
these types are opaque and not related to other types in the system in a
user-visible way: they should have a `_create` and `_destroy` function. See @ref
time_state, @ref time_state_create, @ref time_state_destroy
In other cases, an interface will have function pointers defined as fields in
the interface structure. (A type implementing these may be opaque, but would
begin with a member of the interface/base type.) These interface function
pointers must still take in a self pointer as their first parameter, because
there is no implied `this` pointer in C. This would result in awkward calls with
repeated, error-prone mentions of the object pointer, such as this example
calling the @ref xrt_device::update_inputs interface:
`xdev->update_inputs(xdev)`. These are typically wrapped by inline free
functions that make the call through the function pointer. Considering again the
@ref xrt_device example, the way you would call @ref xrt_device::update_inputs
is actually @ref xrt_device_update_inputs().
### Destroy takes a pointer to a pointer, nulls it out
Destroy free functions should take a pointer to a pointer, performing null checks
before destruction, and setting null. They always succeed (void return): a
failure when destroying an object has little meaning in most cases. For a
sample, see @ref xrt_images_destroy. It would be used like this:
```c
struct xrt_image_native_allocator *xina = /* created and initialized, or maybe NULL */;
/* ... */
xrt_images_destroy(&xina);
/* here, xina is NULL in all cases, and if it wasn't NULL before, it has been freed. */
```
Note that this pattern is used in most cases but not all in the codebase: we
are gradually migrating those that don't fit this pattern. If you call a
destroy function that does not take a pointer-to-a-pointer, make sure to do
null checks before calling and set it to null after it returns.
Also note: when an interface includes a "destroy" function pointer, it takes the
normal pointer to an object: The free function wrapper is the one that takes a
pointer-to-a-pointer and handles the null checks. See for example @ref
xrt_instance_destroy takes the pointer-to-a-pointer, while the interface method
@ref xrt_instance::destroy takes the single pointer.