/* Copyright (C) 2010 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 "ulong_extras.h" #include "nmod_vec.h" #include "nmod_mat.h" #include "nmod_poly.h" #include "perm.h" static mp_limb_t _nmod_mat_det_2x2(mp_limb_t a, mp_limb_t b, mp_limb_t c, mp_limb_t d, nmod_t mod) { b = nmod_neg(b, mod); return nmod_addmul(nmod_mul(a, d, mod), b, c, mod); } static mp_limb_t _nmod_mat_det_3x3(mp_limb_t a, mp_limb_t b, mp_limb_t c, mp_limb_t d, mp_limb_t e, mp_limb_t f, mp_limb_t g, mp_limb_t h, mp_limb_t i, nmod_t mod) { mp_limb_t s, t, u; s = _nmod_mat_det_2x2(e, f, h, i, mod); t = _nmod_mat_det_2x2(g, i, d, f, mod); u = _nmod_mat_det_2x2(d, e, g, h, mod); s = nmod_mul(a, s, mod); s = nmod_addmul(s, b, t, mod); s = nmod_addmul(s, c, u, mod); return s; } static mp_limb_t _nmod_mat_det_4x4(mp_limb_t ** const mat, nmod_t mod) { mp_limb_t s, t, u, v; s = _nmod_mat_det_3x3(mat[1][1], mat[1][2], mat[1][3], mat[2][1], mat[2][2], mat[2][3], mat[3][1], mat[3][2], mat[3][3], mod); t = _nmod_mat_det_3x3(mat[1][0], mat[1][2], mat[1][3], mat[2][0], mat[2][2], mat[2][3], mat[3][0], mat[3][2], mat[3][3], mod); u = _nmod_mat_det_3x3(mat[1][0], mat[1][1], mat[1][3], mat[2][0], mat[2][1], mat[2][3], mat[3][0], mat[3][1], mat[3][3], mod); v = _nmod_mat_det_3x3(mat[1][0], mat[1][1], mat[1][2], mat[2][0], mat[2][1], mat[2][2], mat[3][0], mat[3][1], mat[3][2], mod); t = nmod_neg(t, mod); v = nmod_neg(v, mod); s = nmod_mul(mat[0][0], s, mod); s = nmod_addmul(s, mat[0][1], t, mod); s = nmod_addmul(s, mat[0][2], u, mod); s = nmod_addmul(s, mat[0][3], v, mod); return s; } mp_limb_t _nmod_mat_det(nmod_mat_t A) { mp_limb_t det; slong * P; slong m = A->r; slong rank; slong i; P = flint_malloc(sizeof(slong) * m); rank = nmod_mat_lu(P, A, 1); det = UWORD(0); if (rank == m) { det = UWORD(1); for (i = 0; i < m; i++) det = n_mulmod2_preinv(det, nmod_mat_entry(A, i, i), A->mod.n, A->mod.ninv); } if (_perm_parity(P, m) == 1) det = nmod_neg(det, A->mod); flint_free(P); return det; } mp_limb_t nmod_mat_det(const nmod_mat_t A) { nmod_mat_t tmp; mp_limb_t det; slong dim = A->r; if (dim != A->c) { flint_printf("Exception (nmod_mat_det). Non-square matrix.\n"); flint_abort(); } if (dim == 0) return A->mod.n != 1; if (dim == 1) return nmod_mat_entry(A, 0, 0); if (dim == 2) return _nmod_mat_det_2x2( nmod_mat_entry(A, 0, 0), nmod_mat_entry(A, 0, 1), nmod_mat_entry(A, 1, 0), nmod_mat_entry(A, 1, 1), A->mod); if (dim == 3) return _nmod_mat_det_3x3( nmod_mat_entry(A, 0, 0), nmod_mat_entry(A, 0, 1), nmod_mat_entry(A, 0, 2), nmod_mat_entry(A, 1, 0), nmod_mat_entry(A, 1, 1), nmod_mat_entry(A, 1, 2), nmod_mat_entry(A, 2, 0), nmod_mat_entry(A, 2, 1), nmod_mat_entry(A, 2, 2), A->mod); if (dim == 4) return _nmod_mat_det_4x4(A->rows, A->mod); if (dim <= 8) { mp_limb_t cp[9]; _nmod_mat_charpoly_berkowitz(cp, A, A->mod); if (dim % 2) return nmod_neg(cp[0], A->mod); else return cp[0]; } nmod_mat_init_set(tmp, A); if (n_is_prime(A->mod.n)) det = _nmod_mat_det(tmp); else det = _nmod_mat_det_howell(tmp); nmod_mat_clear(tmp); return det; }