Skip to content

Latest commit

 

History

History
133 lines (103 loc) · 6.59 KB

field.md

File metadata and controls

133 lines (103 loc) · 6.59 KB

Bit-Field Memory Slots

This module implements a load/store protocol for BitSlice regions that enables them to act as if they were a storage slot for integers. Implementations of the BitField trait provide behavior similar to C and C++ language bit-fields. While any BitSlice<T, O> instantiation is able to provide this behavior, the lack of specialization in the language means that it is instead only implemented for BitSlice<T, Lsb0> and BitSlice<T, Msb0> in order to gain a performance advantage.

Batched Behavior

Bit-field behavior can be simulated using BitSlice’s existing APIs; however, the inherent methods are all required to operate on each bit individually in sequence. In addition to the semantic load/store behavior this module describes, it also implements it in a way that takes advantage of the contiguity properties of the Lsb0 and Msb0 orderings in order to maximize how many bits are transferred in each cycle of the overall operation.

This is most efficient when using BitSlice<usize, O> as the storage bit-slice, or using .load::<usize>() or .store::<usize>() as the transfer type.

Bit-Slice Storage and Integer Value Relationships

BitField permits any type of integer, including signed integers, to be stored into or loaded out of a BitSlice<T, _> with any storage type T. While the examples in this module will largely use u8, just to keep the text concise, BitField is tested, and will work correctly, for any combination of types.

BitField implementations use the processor’s own concept of integer registers to operate. As such, the byte-wise memory access patters for types wider than u8 depends on your processor’s byte endianness, as well as which BitField method, and which BitOrder type parameter, you are using.

BitField only operates within processor registers; traffic of T elements between the memory bank and the processor register is controlled entirely by the processor.

If you do not want to introduce the processor’s byte endianness as a variable that affects the in-memory representation of stored integers, use BitSlice<u8, _> as the bit-field storage type. In particular, BitSlice<u8, Msb0> will fill memory in a way that intuitively matches what most debuggers show when inspecting memory.

On the other hand, if you do not care about memory representation and just need fast storage of less than an entire integer, BitSlice<Lsb0, usize> is likely your best bet. As always, the choice of type parameters is a trade-off with different advantages for each combination, which is why bitvec refuses to make the choice for you.

Signed Behavior

The length of the BitSlice that stores a value is considered to be the width of that value when it is loaded back out. As such, storing an i16 into a bit-slice of length 12 means that the stored value has type i12.

When calling .load::<i16>() on a 12-bit slice, the load will detect the sign bit of the i12 value and sign-extend it to i16. This means that storing 2048i16 into a 12-bit slice and then loading it back out into an i16 will produce -2048i16 (negative), not 2048i16 (positive), because 1 << 11 is the sign bit.

BitField does not record the true sign bit of an integer being stored, and will not attempt to set the sign bit of the narrowed value in storage. Storing -127i8 (0b1000_0001) into a 7-bit slice will load 1i8.

Register Bit Order Preservation

The implementations in this module assume that the bits within a value being transferred into or out of a bit-slice should not be re-ordered. While the implementations will segment a value in order to make it fit into bit-slice storage, and will order those segments in memory according to their type parameter and specific trait method called, each segment will remain individually unmodified.

If we consider the value 0b100_1011, segmented at the underscore, then the segments 0b100 and 0b1011 will be present somewhere in the bit-slice that stores them. They may be shifted within an element or re-ordered across elements, but each segment will not be changed.

Endianness

bitvec uses the BitOrder trait to describe the order of bits within a single memory element. This ordering is independent of, and does not consider, the ordering of memory elements in a sequence; bitvec is always “little-endian” in this regard: lower indices are in lower memory addresses, higher indices are in higher memory addresses.

However, BitField is explicitly aware of multiple storage elements in sequence. It is by design able to allow combinations such as <BitSlice<u8, Lsb0> as BitField>::store_be::<u32>. Even where the storage and value types are the same, or the value is narrower, the bit-slice may be spread across multiple elements and must segment the value across them.

The _be and _le orderings on BitField method names refer to the numeric significance of bit-slice storage elements.

In _be methods, lower-address storage elements will hold more-significant segments of the value, and higher-address storage will hold less-significant.

In _le methods, lower-address storage elements will hold less-significant segments of the value, and higher-address storage will hold more-significant.

Consider again the value 0b100_1011, segmented at the underscore. When used with .store_be(), it will be placed into memory as [0b…100…, 0b…1011…]; when used with .store_le(), it will be placed into memory as [0b…1011…, 0b…100…].

Bit-Ordering Behaviors

The _be and _le suffices select the ordering of storage elements in memory. The other critical aspect of the BitField memory behavior is selecting which bits in a storage element are used when a bit-slice has partial elements.

When BitSlice<_, Lsb0> produces a Domain::Region, its head is in the most-significant bits of its element and its tail is in the least-significant bits. When BitSlice<_, Msb0> produces a Region, its head is in the least-significant bits, and its tail is in the most-significant bits.

You can therefore use these combinations of BitOrder type parameter and BitField method suffix to select exactly the memory behavior you want for a storage region.

Each implementation of BitField has documentation showing exactly what its memory layout looks like, with code examples and visual inspections of memory. This documentation is likely collapsed by default when viewing the trait docs; be sure to use the [+] button to expand it!