This crate provides macros that create bit fields and bit enums, which are useful in bit packing code (e.g. in drivers or networking code).
Some highlights: - Highly efficient and 100% safe code that is just as good and hand-writen shifts and masks, - Full compatibility with const contexts, - Useable in no-std environments, - Strong compile time guarantees (for example, taking 5 bits out of a bitfield and putting them into another won't even need to compile a bounds check), - Automatic creation of bitenums, which allow converting enums to/from numbers, - Array support within bitfields to represent repeating bit patterns.
A bit field is created similar to a regular Rust struct. Annotations define the layout of the structure. As an example, consider the following definition, which specifies a bit field:
```
struct GICD_TYPER { #[bits(11..=15, r)] lspi: u5,
#[bit(10, r)]
security_extn: bool,
#[bits(5..=7, r)]
cpu_number: u3,
#[bits(0..=4, r)]
itlines_number: u5,
} ```
It has the following parts: - #[bitfield(u32)] specifies that this is a bitfield in which u32 is the underlying data type. This means that all the bits inside of the bitfield have to fit within 32 bits. u8, u16, u32, u64 and u128 are supported as underlying data types. - Each field is annotated with the range of bits that are used by the field. The data type must match the number of bits: A range of 0..=8 with u8 would cause a compile error, as u9 is the data type that matches 0..=8. - bool fields are declared as "bit", all other fields as "bits" - Valid data types for fields are the basic types u8, u16, u32, u64, u128, bool as well as enums (see below) or types like u1, u2, u3 from arbitrary-int - Fields are declared as "r" for read-only, "w" for write-only or "rw" as read/write. In the example above, all fields are read-only as this specific register is only used to read values.
Very often, fields aren't just numbers but really enums. This is supported by first defining a bitenum and then using that inside of a bitfield:
```
pub enum NonExhaustiveEnum { Zero = 0b00, One = 0b01, Two = 0b10, }
enum ExhaustiveEnum { Zero = 0b00, One = 0b01, Two = 0b10, Three = 0b11, }
struct BitfieldWithEnum {
#[bits(2..=3, rw)]
e2: Option
#[bits(0..=1, rw)]
e1: ExhaustiveEnum,
} ```
Sometimes, bits inside of bitfields are repeated. To support this, this crate allows specifying bitwise arrays. For example, the following struct gives read/write access to each individual nibble (hex character) of a u64:
```
struct Nibble64 { #[bits(0..=3, rw)] nibble: [u4; 16], } ```
Arrays can also have a stride. This is useful in the case of multiple smaller values repeating. For example, the following definition provides access to each bit of each nibble:
```
struct NibbleBits64 { #[bit(0, rw, stride: 4)] nibble_bit0: [bool; 16],
#[bit(1, rw, stride: 4)]
nibble_bit1: [bool; 16],
#[bit(2, rw, stride: 4)]
nibble_bit2: [bool; 16],
#[bit(3, rw, stride: 4)]
nibble_bit3: [bool; 16],
} ```
Arbitrary bit widths like u5 or u67 do not exist in Rust at the moment. Therefore, the following dependency is required:
arbitrary-int = "1.1.0"
Eventhough bitfields feel somewhat like structs, they are internally implemented as simple data types like u32. Therefore, they provide an immutable interface: Instead of changing the value of a field, any change operation will return a new bitfield with that field modified.
let a = NibbleBits64::new_with_raw_value(0x12345678_ABCDEFFF);
// Read a value
assert_eq!(u4::new(0xE), a.nibble(3));
// Change a value
let b = a.with_nibble(0, u4::new(0x3))
assert_eq!(0x12345678_ABCDEFF3, nibble.raw_value());