/*
Copyright (C) 2011 Sebastian Pancratz
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 2.1 of the License, or
(at your option) any later version. See .
*/
#include
#include
#include "flint.h"
#include "nmod_vec.h"
#include "nmod_poly.h"
#include "ulong_extras.h"
void _nmod_poly_rem_q1(mp_ptr R,
mp_srcptr A, slong lenA, mp_srcptr B, slong lenB,
nmod_t mod)
{
slong i;
mp_limb_t invL, t, q0, q1, t1, t0, s1, s0;
FLINT_ASSERT(lenA == lenB + 1);
invL = (B[lenB-1] == 1) ? 1 : n_invmod(B[lenB-1], mod.n);
if (lenB < 2)
return;
q1 = n_mulmod2_preinv(A[lenA-1], invL, mod.n, mod.ninv);
t = n_mulmod2_preinv(q1, B[lenB-2], mod.n, mod.ninv);
t = n_submod(t, A[lenA-2], mod.n);
q0 = n_mulmod2_preinv(t, invL, mod.n, mod.ninv);
q1 = nmod_neg(q1, mod);
/* R = A + (q1*x + q0)*B */
t = A[0];
NMOD_ADDMUL(t, q0, B[0], mod);
R[0] = t;
if (mod.norm >= (FLINT_BITS + 1)/2 + 1)
{
for (i = 1; i < lenB - 1; i++)
{
NMOD_RED2(R[i], 0, A[i] + q1*B[i - 1] + q0*B[i], mod);
}
}
else if (mod.norm != 0)
{
for (i = 1; i < lenB - 1; i++)
{
umul_ppmm(t1, t0, q1, B[i - 1]);
umul_ppmm(s1, s0, q0, B[i]);
add_ssaaaa(t1, t0, t1, t0, 0, A[i]);
add_ssaaaa(t1, t0, t1, t0, s1, s0);
t1 = FLINT_MIN(t1, t1 - mod.n);
FLINT_ASSERT(t1 < mod.n);
NMOD_RED2(R[i], t1, t0, mod);
}
}
else
{
for (i = 1; i < lenB - 1; i++)
{
t = A[i];
NMOD_ADDMUL(t, q1, B[i - 1], mod);
NMOD_ADDMUL(t, q0, B[i], mod);
R[i] = t;
}
}
}