All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.bouncycastle.pqc.math.ntru.polynomial.LongPolynomial5 Maven / Gradle / Ivy

package org.bouncycastle.pqc.math.ntru.polynomial;

import org.bouncycastle.util.Arrays;

/**
 * A polynomial class that combines five coefficients into one long value for
 * faster multiplication by a ternary polynomial.
* Coefficients can be between 0 and 2047 and are stored in bits 0..11, 12..23, ..., 48..59 of a long number. */ public class LongPolynomial5 { private long[] coeffs; // groups of 5 coefficients private int numCoeffs; /** * Constructs a LongPolynomial5 from a IntegerPolynomial. The two polynomials are independent of each other. * * @param p the original polynomial. Coefficients must be between 0 and 2047. */ public LongPolynomial5(IntegerPolynomial p) { numCoeffs = p.coeffs.length; coeffs = new long[(numCoeffs + 4) / 5]; int cIdx = 0; int shift = 0; for (int i = 0; i < numCoeffs; i++) { coeffs[cIdx] |= ((long)p.coeffs[i]) << shift; shift += 12; if (shift >= 60) { shift = 0; cIdx++; } } } private LongPolynomial5(long[] coeffs, int numCoeffs) { this.coeffs = coeffs; this.numCoeffs = numCoeffs; } /** * Multiplies the polynomial with a TernaryPolynomial, taking the indices mod N and the values mod 2048. */ public LongPolynomial5 mult(TernaryPolynomial poly2) { long[][] prod = new long[5][coeffs.length + (poly2.size() + 4) / 5 - 1]; // intermediate results, the subarrays are shifted by 0,...,4 coefficients // multiply ones int[] ones = poly2.getOnes(); for (int idx = 0; idx != ones.length; idx++) { int pIdx = ones[idx]; int cIdx = pIdx / 5; int m = pIdx - cIdx * 5; // m = pIdx % 5 for (int i = 0; i < coeffs.length; i++) { prod[m][cIdx] = (prod[m][cIdx] + coeffs[i]) & 0x7FF7FF7FF7FF7FFL; cIdx++; } } // multiply negative ones int[] negOnes = poly2.getNegOnes(); for (int idx = 0; idx != negOnes.length; idx++) { int pIdx = negOnes[idx]; int cIdx = pIdx / 5; int m = pIdx - cIdx * 5; // m = pIdx % 5 for (int i = 0; i < coeffs.length; i++) { prod[m][cIdx] = (0x800800800800800L + prod[m][cIdx] - coeffs[i]) & 0x7FF7FF7FF7FF7FFL; cIdx++; } } // combine shifted coefficients (5 arrays) into a single array of length prod[*].length+1 long[] cCoeffs = Arrays.copyOf(prod[0], prod[0].length + 1); for (int m = 1; m <= 4; m++) { int shift = m * 12; int shift60 = 60 - shift; long mask = (1L << shift60) - 1; int pLen = prod[m].length; for (int i = 0; i < pLen; i++) { long upper, lower; upper = prod[m][i] >> shift60; lower = prod[m][i] & mask; cCoeffs[i] = (cCoeffs[i] + (lower << shift)) & 0x7FF7FF7FF7FF7FFL; int nextIdx = i + 1; cCoeffs[nextIdx] = (cCoeffs[nextIdx] + upper) & 0x7FF7FF7FF7FF7FFL; } } // reduce indices of cCoeffs modulo numCoeffs int shift = 12 * (numCoeffs % 5); for (int cIdx = coeffs.length - 1; cIdx < cCoeffs.length; cIdx++) { long iCoeff; // coefficient to shift into the [0..numCoeffs-1] range int newIdx; if (cIdx == coeffs.length - 1) { iCoeff = numCoeffs == 5 ? 0 : cCoeffs[cIdx] >> shift; newIdx = 0; } else { iCoeff = cCoeffs[cIdx]; newIdx = cIdx * 5 - numCoeffs; } int base = newIdx / 5; int m = newIdx - base * 5; // m = newIdx % 5 long lower = iCoeff << (12 * m); long upper = iCoeff >> (12 * (5 - m)); cCoeffs[base] = (cCoeffs[base] + lower) & 0x7FF7FF7FF7FF7FFL; int base1 = base + 1; if (base1 < coeffs.length) { cCoeffs[base1] = (cCoeffs[base1] + upper) & 0x7FF7FF7FF7FF7FFL; } } return new LongPolynomial5(cCoeffs, numCoeffs); } public IntegerPolynomial toIntegerPolynomial() { int[] intCoeffs = new int[numCoeffs]; int cIdx = 0; int shift = 0; for (int i = 0; i < numCoeffs; i++) { intCoeffs[i] = (int)((coeffs[cIdx] >> shift) & 2047); shift += 12; if (shift >= 60) { shift = 0; cIdx++; } } return new IntegerPolynomial(intCoeffs); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy