Skip to content

Latest commit

 

History

History
79 lines (68 loc) · 4 KB

README.md

File metadata and controls

79 lines (68 loc) · 4 KB

Purpose

Abstracted slices and slice-likes types in Rust.

API and Source principles

  • Ergonomic: Practical (rather than for exotic combinations),
  • Type-safe - hence complex, but robust,
  • Large, but organized.

Rules and Constraints

API

  • Explicit and clear. For example, std and no_std features are mutually exclusive, so users detect any conflicts as early as possible. -- TODO Hashed could be a value object (transparent wrapper). It would make it more "future-proof"/flexible - but not backwards-compatible with your/third party's existing API's. Hence, we implement it directly for std::collections::HashMap (when using std feature). That makes it work for any application that uses HashMap - easier to use. -- size_for_array_only, allow_empty_arrays, disable_empty_arrays are crate features, rather than per-struct or per-object fields. See more below at Crate Features.

API and source

  • Generic: Generics, constant generic parameters and associated generic types are essential to memory efficiency and type safety. The drawbacks are longer build times and larger binaries. We optimize it where worthwhile.

Requiring Nightly Rust

As of mid 2022, we need Rust nightly. Otherwise this couldn't be so ergonomic.

Supporting no_std

We test building for several of more supported (tier 2) no_std Rust targets. See no_std at https://doc.rust-lang.org/nightly/rustc/platform-support.html#tier-2.

Crate Features

std, no_std and no_std_vec

  • no_std (plus optionally no_std_vec to support vector-based implementations & functions)
  • std to indicate a need for full functionality (which is the same as no_std_vec plus HashMap-based).
  • no_std (and no_std_vec) are mutually exclusive with std
  • libraries can use no_std_vec feature for Vec-based functionality, and std for full functionality (including HashMap-based) with conditional compilation (controlled by their features). Then such libraries can be used in either std or no_std projects. You can also use macros with_heap, enum_cfg, match_cfg if your crate also defines those feature names.

Array size features

See features in Cargo.toml and their respective test_crates.

Module name (and hence full qualified name) conventions

Some full qualified trait/struct/module names repeat their parts. Better have more granular module paths, and some name repetition in file paths and module names (like slices::bool_slice), rather than long (source) files).

SliceStorage and similar

These data-carrying enums are: SliceStorage, SliceStorageClone, SliceStorageDefault, SliceStorageDefaultClone. They provide storage and abstracted access and transformations.

Const generic param N is used by Array invariant only. However, it makes all variants consume space. Hence N > 0 is suggested primarily for no-heap or frequent instantiation on stack and for small sizes (N). If you run with heap, and you have infrequent instantiation and/or large sizes, suggest passing 0 for N, and use Vec invariant instead.

Why don't we call this to SliceStorageCopy instead of SliceStorage and why don't we rename the existing SliceStorageClone to SliceStorage? It could lead to laziness/not noticing/forgetting to use SliceStorageCopy whenever possible. Especially so because then any Copy type could be stored in either SliceStorage or SliceStorageCopy. Storing Copy items in a Clone-friendly storage would work, but it would be less efficient than storing them in a specialized Copy storage. Even more so with default values (which, if primitive and if stored in specialized Copy storage, could be optimized away - so the virtual memory wouldn't be used until written to it later, if at all). However, it's unlikely that anyone would use SliceStorageClone for Copy items (even though they could).