Module Field_ops_lib.Bram_reduce

Module to reduce numbers modulo a fixed constant with just two subtraction stages.

Given a fixed modular base pp with nn bits and a multiplicative error EE with ee bits, this module can reduce any input N[0,Ep)N\in[0, Ep) to an equivalent value modulo pp in the range [0,p)[0, p) with just two subtraction stages.

The module accomplishes this by precomputing and loading a ROM with 2e2^e entries. We set R[0]=0R[0] = 0. Then, for i1i\geq1, entry R[i]R[i] holds the final nn bits of the largest multiple of pp that has i1i-1 as its ee-bit prefix when written with e+ne+n bits. In particular,

R[i]=(i1)2np(mod2n)R[i] = \left\lfloor \frac{(i-1)\cdot 2^n}{p} \right\rfloor\pmod {2^{n}}

Then, given an input N[0,Ep)N\in[0,Ep) with e+ne+n bits, we can lookup its ee-bit prefix in the ROM and subtract the resulting value from the nn-bit suffix of our input value:

N[n1:0]R[N[e1+n:n]] N[n-1:0] - R\left[N[e-1+n:n]\right]

(note that we only need to use the lower nn bits because we know what the higher bits are by construction). Then, when the lookup index is nonzero, we have to invert the MSB of the signed result to get an unsigned reduction nn^\prime to the range [0,3p)[0,3p).

After this, we do one more subtraction stage where we just multiplex between {n,np,n2p}\{n^\prime, n^\prime-p, n^\prime-2p\} to choose the final result.

This module can be generalized over any modular base pp (not necessarily prime), rather than just our given prime. However, this requires a bit of care to double check the bounds above - in our case, the first ROM reduction goes to the range [0,3p)[0,3p), but for some bases, it is possible that the reduction only goes to [0,4p)[0,4p) and requires one more subtractor in the subsequent stage.

module type Config = sig ... end
module Make (Config : Config) : sig ... end