/*
Copyright (C) 2021 Fredrik Johansson
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_mat.h"
#include "nmod_poly.h"
void
_nmod_mat_charpoly_berkowitz(mp_ptr cp, const nmod_mat_t mat, nmod_t mod)
{
const slong n = mat->r;
if (mod.n == 1)
{
_nmod_vec_zero(cp, n + 1);
}
else if (n == 0)
{
cp[0] = 1;
}
else if (n == 1)
{
cp[0] = nmod_neg(nmod_mat_entry(mat, 0, 0), mod);
cp[1] = 1;
}
else if (n == 2)
{
cp[0] = nmod_sub(nmod_mul(nmod_mat_entry(mat, 0, 0), nmod_mat_entry(mat, 1, 1), mod),
nmod_mul(nmod_mat_entry(mat, 0, 1), nmod_mat_entry(mat, 1, 0), mod), mod);
cp[1] = nmod_add(nmod_mat_entry(mat, 0, 0), nmod_mat_entry(mat, 1, 1), mod);
cp[1] = nmod_neg(cp[1], mod);
cp[2] = 1;
}
else
{
slong i, k, t;
mp_ptr a, A, s;
int nlimbs;
TMP_INIT;
TMP_START;
a = TMP_ALLOC(sizeof(mp_limb_t) * (n * n));
A = a + (n - 1) * n;
nlimbs = _nmod_vec_dot_bound_limbs(n, mod);
_nmod_vec_zero(cp, n + 1);
cp[0] = nmod_neg(nmod_mat_entry(mat, 0, 0), mod);
for (t = 1; t < n; t++)
{
for (i = 0; i <= t; i++)
{
a[0 * n + i] = nmod_mat_entry(mat, i, t);
}
A[0] = nmod_mat_entry(mat, t, t);
for (k = 1; k < t; k++)
{
for (i = 0; i <= t; i++)
{
s = a + k * n + i;
s[0] = _nmod_vec_dot(mat->rows[i], a + (k - 1) * n, t + 1, mod, nlimbs);
}
A[k] = a[k * n + t];
}
A[t] = _nmod_vec_dot(mat->rows[t], a + (t - 1) * n, t + 1, mod, nlimbs);
for (k = 0; k <= t; k++)
{
cp[k] = nmod_sub(cp[k], _nmod_vec_dot_rev(A, cp, k, mod, nlimbs), mod);
cp[k] = nmod_sub(cp[k], A[k], mod);
}
}
/* Shift all coefficients up by one */
for (i = n; i > 0; i--)
cp[i] = cp[i - 1];
cp[0] = 1;
_nmod_poly_reverse(cp, cp, n + 1, n + 1);
TMP_END;
}
}
void nmod_mat_charpoly_berkowitz(nmod_poly_t cp, const nmod_mat_t mat)
{
if (mat->r != mat->c)
{
flint_printf("Exception (nmod_mat_charpoly_berkowitz). Non-square matrix.\n");
flint_abort();
}
nmod_poly_fit_length(cp, mat->r + 1);
_nmod_poly_set_length(cp, mat->r + 1);
_nmod_mat_charpoly_berkowitz(cp->coeffs, mat, mat->mod);
}