Bit extraction
This document provides an overview of the bit extraction feature in Concrete, including usage examples, limitations, and performance considerations.
Overview
Bit extraction could be useful in some applications that require directly manipulating bits of integers. Bit extraction allows you to extract a specific slice of bits from an integer, where index 0 corresponds to the least significant bit (LSB). The cost of this operation increases with the index of the highest significant bit you wish to extract.
Bit extraction only works in the Native
encoding, which is usually selected when all table lookups in the circuit are less than or equal to 8 bits.
Extracting a specific bit
Extracting multiple bits with slices
You can use slices for indexing fhe.bits(value)
:
Bit extraction supports slices with negative steps:
Bit extraction with signed integers
Bit extraction supports signed integers:
Use case example
Here's a practical example that uses bit extraction to determine if a number is even:
It prints:
Limitations
Negative indexing is not supported: Bits extraction using negative indices is not supported, such as
fhe.bits(x)[-1]
.This is because the bit-width of
x
is unknown before inputset evaluation, making it impossible to determine the correct bit to extract.
Reverse slicing requires explicit starting bit: When extracting bits in reverse order (using a negative step), the start bit must be specified, for example,
fhe.bits(x)[::-1]
is not supported.Signed integer slicing requires explicit stopping bit: For signed integers, when using slices, the stop bit must be explicitly provided, for example,
fhe.bits(x)[1:]
is not supported.Float bit extraction is not supported: While Concrete supports floats to some extent, bit extraction is not possible on float types.
Performance considerations
A Chain of individual bit extractions
Extracting a specific bit requires clearing all the preceding lower bits. This involves extracting these previous bits as intermediate values and then subtracting them from the input.
Implications:
Bits are extracted sequentially, starting from the least significant bit to the more significant ones. The cost is proportional to the index of the highest extracted bit plus one.
No parallelization is possible. The computation time is proportional to the cost, independent of the number of CPUs.
Examples:
Extracting
fhe.bits(x)[4]
is approximately five times costlier than extractingfhe.bits(x)[0]
.Extracting
fhe.bits(x)[4]
takes around five times more wall clock time thanfhe.bits(x)[0]
.The cost of extracting
fhe.bits(x)[0:5]
is almost the same as that offhe.bits(x)[5]
.
Reuse of Intermediate Extracted Bits
Common sub-expression elimination is applied to intermediate extracted bits.
Implications:
The overall cost for a series of
fhe.bits(x)[m:n]
calls on the same inputx
is almost equivalent to the cost of the single most computationally expensive extraction in the series, i.e.fhe.bits(x)[n]
.The order of extraction in that series does not affect the overall cost.
Example:
The combined operation fhe.bit(x)[3] + fhe.bit(x)[2] + fhe.bit(x)[1]
has almost the same cost as fhe.bits(x)[3]
.
TLUs of 1b input precision
Each extracted bit incurs a cost of approximately one TLU of 1-bit input precision. Therefore, fhe.bits(x)[0]
is generally faster than any other TLU operation.
Last updated