Skip to content

Commit

Permalink
Add contributing guide
Browse files Browse the repository at this point in the history
  • Loading branch information
chrysn committed Nov 30, 2024
1 parent fa66f7f commit fb7a921
Showing 1 changed file with 55 additions and 0 deletions.
55 changes: 55 additions & 0 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
**Contributions are welcome.**

## Contribution guidelines

General [RIOT contribution guidelines](https://github.com/RIOT-OS/RIOT/blob/master/CONTRIBUTING.md) apply,
with some

* focus points:

* When wrapping a new RIOT feature, **keep it small**.
Attempting to implement all features of a particular RIOT subsystem easily gets caught in long iterations of review.
Try wrapping a minimal viable version first, and let's take it from there.

* New features should come with tests, at least on a coarse granularity.
Tests in [`./tests/`](tests) are automatically picked up by CI.

* and alterations:

* RIOT has no coding conventions for Rust; instead, common Rust conventions apply.
Public items should be documented,
and code should be `cargo fmt` before each commit.

* Merging criteria are not as strict as in RIOT OS:
RIOT maintainers may merge PRs without a 2nd pair of eyes
(but are encouraged to solicit review on larger changes).

Before code from this crate gets used by RIOT OS,
it undergoes a secondary review step [similar to PR #20786](https://github.com/RIOT-OS/RIOT/pull/20786),
just like any external package code.

* The license of this crate is "[MIT](https://spdx.org/licenses/MIT.html) OR [Apache-2.0](https://spdx.org/licenses/Apache-2.0.html)" as stated in [Cargo.toml](Cargo.toml) and the [README](README.md).
This eases interoperability with the rest of the Rust ecosystem;
beware that built binaries include RIOT OS, and thus have LGPL-2.1 components.

## Common pitfalls

* Threads and interrupts in RIOT are modelled as is [common](https://onevariable.com/blog/interrupts-is-threads/) in Rust:
Data needs to be [`Send`](https://doc.rust-lang.org/std/marker/trait.Send.html) to be passed around.
Whenever generic data (in particular references) is passed into C functions,
trait bounds need to ensure that it is `Send`.

* Registering callbacks with RIOT OS typically means
that the OS may invoke the callback at any time.
Thus the callback either must be `'static`,
or it needs to be ensured that the callback is fully unregistered
before its lifetime ends.

Note that we can not rely on [`Drop`](https://doc.rust-lang.org/std/ops/trait.Drop.html) to be called
(for any user may safely [`forget()`](https://doc.rust-lang.org/std/mem/fn.forget.html) an item at any time).

Running user code in Rust closures is a common workaround
(eg. [in `thread::scope`](https://doc.riot-os.org/rustdoc/latest/riot_wrappers/thread/fn.scope.html)),
but beware that any items in that scope need to be [branded](http://plv.mpi-sws.org/rustbelt/ghostcell/paper.pdf) (like with the `'id` of the thread scope):
Otherwise, users might create nested scopes and switch around the provided types,
delaying their cleanup from the inner to the outer scope.

0 comments on commit fb7a921

Please sign in to comment.