With const_guards
you can express certain compile time constraints on rust's const_generics
using the unstable generic_const_exprs
feature.
For documentation visit docs.rs.
Consider the following usage of the first
method on slices from the standard library:
rust
let slice: [(); 1] = [(); 1];
let head: Option<&()> = slice.first();
Would it be nice if we could just write
rust
let head: &() = slice.first();
since the compiler should know this slice has length 1
at this point.
With const guards we can express such as follows:
```rust
fn first<'a, T, const N: usize>(slice: &'a [T; N]) -> &'a T {
&slice[0]
}
The index call on the slice `&slice[0]` cannot possible fail because we enforced the length of the slice to be `> 0` at compile time. We could now call it as follows
rust
let slice: [(); 1] = [(); 1];
let head: &() = first(&slice);
while the case where the slice is actually empty would _fail to compile_:
let slice: [(); 0] = [(); 0];
let head: &() = first(&slice);
Finally we could even express this as a trait to make it more accessable:
rust
trait SliceHead<'a, T, const N: usize> {
#[guard(
impl<'a, T, const N: usize> SliceHead<'a, T, N> for &'a [T; N] { fn head(&self) -> &'a T { &self[0] } }
fn main() { let slice: &[(); 1] = &[(); 1]; let head: &() = slice.head(); } ``` Though, as you can see, we need to introduce generics not introduced by the guarded item explicitly.
Consider this simple example of a const guard: ```rust fn main() { f::<0>() }
fn f trait Protect {}
impl Protect for Guard<{}> {} fn main() {
f::<0>()
} fn f
and have a look at the expanded form:
rust
struct GuardTodo