flint-sys 0.9.0

Bindings to the FLINT C library
Documentation
/*
    Copyright (C) 2025, Vincent Neiger, Éric Schost
    Copyright (C) 2025, Mael Hostettler

    This file is part of FLINT.

    FLINT is free software: you can redistribute it and/or modify it under
    the terms of the GNU Lesser General Public License (LGPL) as published
    by the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.  See <https://www.gnu.org/licenses/>.
*/

#include "nmod_vec.h"
#include "nmod_poly.h"

void 
nmod_geometric_progression_init(nmod_geometric_progression_t G, ulong r, slong len, nmod_t mod)
{
    ulong q, inv_r, inv_q, tmp, qk, inv_qk, qq, s;
    nn_ptr diff, inv_diff, prod_diff;
    slong i;
    
    G->len = len;
    G->mod = mod;

    nmod_poly_init2(G->f, mod.n, 2*len - 1); G->f->length = 2*len - 1;
    nmod_poly_init2(G->g1, mod.n, len); G->g1->length = len;
    nmod_poly_init2(G->g2, mod.n, len); G->g2->length = len;
    G->g1->coeffs[0] = 1;
    G->g2->coeffs[0] = 1;
    G->f->coeffs[0] = 1;

    G->x = _nmod_vec_init(len);
    G->w = _nmod_vec_init(len);
    G->z = _nmod_vec_init(len);
    G->y = _nmod_vec_init(len);

    G->x[0] = 1;
    G->y[0] = 1;
    G->w[0] = 1;
    G->z[0] = 1;

    q = nmod_mul(r, r, mod);
    inv_r = nmod_inv(r, mod);
    inv_q = nmod_mul(inv_r, inv_r, mod);

    tmp = r;
    for (i = 1; i < 2*len - 1; i++)
    {
        G->f->coeffs[i] = nmod_mul(G->f->coeffs[i - 1], tmp, mod);
        tmp = nmod_mul(tmp, q, mod);
    }
    // If we had to normalize G->f than that means r is of low order and following
    // inversion will fail

    tmp = inv_r;
    for (i = 1; i < len; i++)
    {
        G->x[i] = nmod_mul(G->x[i - 1], tmp, mod);
        tmp = nmod_mul(tmp, inv_q, mod);
    }
    
    inv_diff  = _nmod_vec_init(len);
    diff      = _nmod_vec_init(len);
    prod_diff = _nmod_vec_init(len);
    inv_diff[0] = 1;
    diff[0] = 1;
    prod_diff[0] = 1;
 
    qk = q;  // montgomery inversion
    for (i = 1; i < len; i++)
    {
        diff[i] = qk - 1;
        inv_diff[i] = diff[i];
        qk = nmod_mul(qk, q, mod);
        prod_diff[i] = nmod_mul(diff[i], prod_diff[i - 1], mod);
    }
    
    tmp = nmod_inv(prod_diff[len-1], mod);
    for (i = len - 1; i > 0; i--)
    {
        inv_diff[i] = nmod_mul(prod_diff[i - 1], tmp, mod);
        tmp = nmod_mul(tmp, diff[i], mod);
    }
    inv_diff[0] = tmp;
    // end montgomery inversion

    // sets sequences w, y, z and polynomials g1, g2
    qk = 1;
    inv_qk = 1;
    qq = 1;
    s = 1;

    for (i = 1; i < len; i++)
    {
        qq = nmod_mul(qq, qk, mod);   // prod q^i
        s = nmod_mul(s, inv_qk, mod); // prod 1/q^i
        G->w[i] = nmod_mul(G->w[i - 1], inv_diff[i], mod); // prod 1/(q^i-1)
        tmp = nmod_mul(qq, G->w[i], mod); // prod q^i/(q^i-1)
        G->g2->coeffs[i] = tmp;

        if ((i & 1) == 1)   /* i is odd */
        {
            G->g1->coeffs[i] = mod.n - tmp;
            G->y[i] = mod.n - prod_diff[i];
            G->z[i] = mod.n - G->w[i];
        }
        else   /* i is even */
        {
            G->g1->coeffs[i] = tmp;
            G->y[i] = prod_diff[i];
            G->z[i] = G->w[i];
        }
        G->y[i] = nmod_mul(G->y[i], s, mod);

        qk = nmod_mul(qk, q, mod);
        inv_qk = nmod_mul(inv_qk, inv_q, mod);
    }
    // similarly, if either g1 or g2 have leading 0 coefficient, something is wrong 

    _nmod_vec_clear(prod_diff);
    _nmod_vec_clear(inv_diff);
    _nmod_vec_clear(diff);
}

void 
nmod_geometric_progression_clear(nmod_geometric_progression_t G)
{
    nmod_poly_clear(G->f);
    nmod_poly_clear(G->g2);
    nmod_poly_clear(G->g1);
    _nmod_vec_clear(G->x);
    _nmod_vec_clear(G->z);
    _nmod_vec_clear(G->y);
    _nmod_vec_clear(G->w);
}