Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
r""" Brandt Modules
Introduction ============
This tutorial outlines the construction of Brandt modules in Sage. The importance of this construction is that it provides us with a method to compute modular forms on `\Gamma_0(N)` as outlined in Pizer's paper [Piz1980]_. In fact there exists a non-canonical Hecke algebra isomorphism between the Brandt modules and a certain subspace of `S_{2}(\Gamma_0(pM))` which contains all the newforms.
The Brandt module is the free abelian group on right ideal classes of a quaternion order together with a natural Hecke action determined by Brandt matrices.
Quaternion Algebras -------------------
A quaternion algebra over `\QQ` is a central simple algebra of dimension 4 over `\QQ`. Such an algebra `A` is said to be ramified at a place `v` of `\QQ` if and only if `A_v=A\otimes \QQ_v` is a division algebra. Otherwise `A` is said to be split at `v`.
``A = QuaternionAlgebra(p)`` returns the quaternion algebra `A` over `\QQ` ramified precisely at the places `p` and `\infty`.
``A = QuaternionAlgebra(k,a,b)`` returns a quaternion algebra with basis `\{1,i,j,j\}` over `\mathbb{K}` such that `i^2=a`, `j^2=b` and `ij=k.`
An order `R` in a quaternion algebra is a 4-dimensional lattice on `A` which is also a subring containing the identity.
``R = A.maximal_order()`` returns a maximal order `R` in the quaternion algebra `A.`
An Eichler order `\mathcal{O}` in a quaternion algebra is the intersection of two maximal orders. The level of `\mathcal{O}` is its index in any maximal order containing it.
``O = A.order_of_level_N`` returns an Eichler order `\mathcal{O}` in `A` of level `N` where `p` does not divide `N`.
A right `\mathcal{O}`-ideal `I` is a lattice on `A` such that `I_p=a_p\mathcal{O}` (for some `a_p\in A_p^*`) for all `p<\infty`. Two right `\mathcal{O}`-ideals `I` and `J` are said to belong to the same class if `I=aJ` for some `a \in A^*`. (Left `\mathcal{O}`-ideals are defined in a similar fashion.)
The right order of `I` is defined to be the set of elements in `A` which fix `I` under right multiplication.
``right_order(R, basis)`` returns the right ideal of `I` in `R` given a basis for the right ideal `I` contained in the maximal order `R.`
``ideal_classes(self)`` returns a tuple of all right ideal classes in self which, for the purpose of constructing the Brandt module B(p,M), is taken to be an Eichler order of level M.
The implementation of this method is especially interesting. It depends on the construction of a Hecke module defined as a free abelian group on right ideal classes of a quaternion algebra with the following action
.. MATH::
T_n[I] = \sum_{\phi} [J]
where `(n,pM)=1` and the sum is over cyclic `\mathcal{O}`-module homomorphisms `\phi :I\rightarrow J` of degree `n` up to isomorphism of `J`. Equivalently one can sum over the inclusions of the submodules `J \rightarrow n^{-1}I`. The rough idea is to start with the trivial ideal class containing the order `\mathcal{O}` itself. Using the method ``cyclic_submodules(self, I, p)`` one computes `T_p([\mathcal{O}])` for some prime integer `p` not dividing the level of the order `\mathcal{O}`. Apply this method repeatedly and test for equivalence among resulting ideals. A theorem of Serre asserts that one gets a complete set of ideal class representatives after a finite number of repetitions.
One can prove that two ideals `I` and `J` are equivalent if and only if there exists an element `\alpha \in I \overline{J}` such `N(\alpha)=N(I)N(J)`.
``is_equivalent(I,J)`` returns true if `I` and `J` are equivalent. This method first compares the theta series of `I` and `J`. If they are the same, it computes the theta series of the lattice `I\overline(J)`. It returns true if the `n^{th}` coefficient of this series is nonzero where `n=N(J)N(I)`.
The theta series of a lattice `L` over the quaternion algebra `A` is defined as
.. MATH::
\theta_L(q)=\sum_{x \in L} q^{\frac{N(x)}{N(L)}}
``L.theta_series(T,q)`` returns a power series representing `\theta_L(q)` up to a precision of `\mathcal{O}(q^{T+1})`.
Hecke Structure ---------------
The Hecke structure defined on the Brandt module is given by the Brandt matrices which can be computed using the definition of the Hecke operators given earlier.
``hecke_matrix_from_defn(self,n)`` returns the matrix of the n-th Hecke operator `B_{0}(n)` acting on self, computed directly from the definition.
However, one can efficiently compute Brandt matrices using theta series. In fact, let `\{I_{1},.....,I_{h}\}` be a set of right `\mathcal{O}`-ideal class representatives. The (i,j) entry in the Brandt matrix `B_{0}(n)` is the product of the `n^{th}` coefficient in the theta series of the lattice `I_{i}\overline{I_{j}}` and the first coefficient in the theta series of the lattice `I_{i}\overline{I_{i}}`.
``compute_hecke_matrix_brandt(self,n)`` returns the n-th Hecke matrix, computed using theta series.
EXAMPLES::
sage: B = BrandtModule(23)
sage: B.maximal_order() Order of Quaternion Algebra (-1, -23) with base ring Rational Field with basis (1/2 + 1/2*j, 1/2*i + 1/2*k, j, k)
sage: B.right_ideals() (Fractional ideal (2 + 2*j, 2*i + 2*k, 4*j, 4*k), Fractional ideal (2 + 2*j, 2*i + 6*k, 8*j, 8*k), Fractional ideal (2 + 10*j + 8*k, 2*i + 8*j + 6*k, 16*j, 16*k))
sage: B.hecke_matrix(2) [1 2 0] [1 1 1] [0 3 0]
sage: B.brandt_series(3) [1/4 + q + q^2 + O(q^3) 1/4 + q^2 + O(q^3) 1/4 + O(q^3)] [ 1/2 + 2*q^2 + O(q^3) 1/2 + q + q^2 + O(q^3) 1/2 + 3*q^2 + O(q^3)] [ 1/6 + O(q^3) 1/6 + q^2 + O(q^3) 1/6 + q + O(q^3)]
REFERENCES:
- [Piz1980]_ - [Koh2000]_
Further Examples ----------------
We decompose a Brandt module over both `\ZZ` and `\QQ`. ::
sage: B = BrandtModule(43, base_ring=ZZ); B Brandt module of dimension 4 of level 43 of weight 2 over Integer Ring sage: D = B.decomposition() sage: D [ Subspace of dimension 1 of Brandt module of dimension 4 of level 43 of weight 2 over Integer Ring, Subspace of dimension 1 of Brandt module of dimension 4 of level 43 of weight 2 over Integer Ring, Subspace of dimension 2 of Brandt module of dimension 4 of level 43 of weight 2 over Integer Ring ] sage: D[0].basis() ((0, 0, 1, -1),) sage: D[1].basis() ((1, 2, 2, 2),) sage: D[2].basis() ((1, 1, -1, -1), (0, 2, -1, -1)) sage: B = BrandtModule(43, base_ring=QQ); B Brandt module of dimension 4 of level 43 of weight 2 over Rational Field sage: B.decomposition()[2].basis() ((1, 0, -1/2, -1/2), (0, 1, -1/2, -1/2))
AUTHORS:
- Jon Bober - Alia Hamieh - Victoria de Quehen - William Stein - Gonzalo Tornaria """
#***************************************************************************** # Copyright (C) 2009 William Stein <wstein@gmail.com> # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 2 of the License, or # (at your option) any later version. # http://www.gnu.org/licenses/ #***************************************************************************** from __future__ import print_function
# imports from sage.misc.all import prod, verbose from sage.rings.all import Integer, ZZ, QQ, PolynomialRing, GF, CommutativeRing
from sage.algebras.quatalg.quaternion_algebra import QuaternionAlgebra, basis_for_quaternion_lattice from sage.algebras.quatalg.quaternion_algebra_cython import rational_matrix_from_rational_quaternions
from sage.arith.all import gcd, factor, prime_divisors, kronecker, next_prime, lcm from sage.modular.hecke.all import (AmbientHeckeModule, HeckeSubmodule, HeckeModuleElement) from sage.modular.dirichlet import TrivialCharacter from sage.matrix.all import MatrixSpace, matrix from sage.misc.mrange import cartesian_product_iterator from sage.structure.richcmp import richcmp, richcmp_method from sage.misc.cachefunc import cached_method
from copy import copy
cache = {}
def BrandtModule(N, M=1, weight=2, base_ring=QQ, use_cache=True): """ Return the Brandt module of given weight associated to the prime power `p^r` and integer `M`, where `p` and `M` are coprime.
INPUT:
- `N` -- a product of primes with odd exponents - `M` -- an integer coprime to `q` (default: 1) - ``weight`` -- an integer that is at least 2 (default: 2) - ``base_ring`` -- the base ring (default: ``QQ``) - ``use_cache`` -- whether to use the cache (default: ``True``)
OUTPUT:
a Brandt module
EXAMPLES::
sage: BrandtModule(17) Brandt module of dimension 2 of level 17 of weight 2 over Rational Field sage: BrandtModule(17,15) Brandt module of dimension 32 of level 17*15 of weight 2 over Rational Field sage: BrandtModule(3,7) Brandt module of dimension 2 of level 3*7 of weight 2 over Rational Field sage: BrandtModule(3,weight=2) Brandt module of dimension 1 of level 3 of weight 2 over Rational Field sage: BrandtModule(11, base_ring=ZZ) Brandt module of dimension 2 of level 11 of weight 2 over Integer Ring sage: BrandtModule(11, base_ring=QQbar) Brandt module of dimension 2 of level 11 of weight 2 over Algebraic Field
The ``use_cache`` option determines whether the Brandt module returned by this function is cached::
sage: BrandtModule(37) is BrandtModule(37) True sage: BrandtModule(37,use_cache=False) is BrandtModule(37,use_cache=False) False
TESTS:
Note that `N` and `M` must be coprime::
sage: BrandtModule(3,15) Traceback (most recent call last): ... ValueError: M must be coprime to N
Only weight 2 is currently implemented::
sage: BrandtModule(3,weight=4) Traceback (most recent call last): ... NotImplementedError: weight != 2 not yet implemented
Brandt modules are cached::
sage: B = BrandtModule(3,5,2,ZZ) sage: B is BrandtModule(3,5,2,ZZ) True """ raise NotImplementedError("Brandt modules currently only implemented when N is a prime") raise ValueError("M must be positive") raise ValueError("weight must be at least 2") raise TypeError("base_ring must be a commutative ring")
def class_number(p, r, M): r""" Return the class number of an order of level `N = p^r M` in the quaternion algebra over `\QQ` ramified precisely at `p` and infinity.
This is an implementation of Theorem 1.12 of [Piz1980]_.
INPUT:
- `p` -- a prime - `r` -- an odd positive integer (default: 1) - `M` -- an integer coprime to `q` (default: 1)
OUTPUT:
Integer
EXAMPLES::
sage: sage.modular.quatalg.brandt.class_number(389,1,1) 33 sage: sage.modular.quatalg.brandt.class_number(389,1,2) # TODO -- right? 97 sage: sage.modular.quatalg.brandt.class_number(389,3,1) # TODO -- right? 4892713 """
def maximal_order(A): """ Return a maximal order in the quaternion algebra ramified at `p` and infinity.
This is an implementation of Proposition 5.2 of [Piz1980]_.
INPUT:
- `A` -- quaternion algebra ramified precisely at `p` and infinity
OUTPUT:
a maximal order in `A`
EXAMPLES::
sage: A = BrandtModule(17).quaternion_algebra() sage: sage.modular.quatalg.brandt.maximal_order(A) Order of Quaternion Algebra (-17, -3) with base ring Rational Field with basis (1/2 + 1/2*j, 1/2*i + 1/2*k, -1/3*j - 1/3*k, k)
sage: A = QuaternionAlgebra(17,names='i,j,k') sage: A.maximal_order() Order of Quaternion Algebra (-3, -17) with base ring Rational Field with basis (1/2 + 1/2*i, 1/2*j - 1/2*k, -1/3*i + 1/3*k, -k) """
def basis_for_left_ideal(R, gens): """ Return a basis for the left ideal of `R` with given generators.
INPUT:
- `R` -- quaternion order - ``gens`` -- list of elements of `R`
OUTPUT:
list of four elements of `R`
EXAMPLES::
sage: B = BrandtModule(17); A = B.quaternion_algebra(); i,j,k = A.gens() sage: sage.modular.quatalg.brandt.basis_for_left_ideal(B.maximal_order(), [i+j,i-j,2*k,A(3)]) [1/2 + 1/6*j + 2/3*k, 1/2*i + 1/2*k, 1/3*j + 1/3*k, k] sage: sage.modular.quatalg.brandt.basis_for_left_ideal(B.maximal_order(), [3*(i+j),3*(i-j),6*k,A(3)]) [3/2 + 1/2*j + 2*k, 3/2*i + 3/2*k, j + k, 3*k] """
def right_order(R, basis): """ Given a basis for a left ideal `I`, return the right order in the quaternion order `R` of elements `x` such that `I x` is contained in `I`.
INPUT:
- `R` -- order in quaternion algebra - ``basis`` -- basis for an ideal `I`
OUTPUT:
order in quaternion algebra
EXAMPLES:
We do a consistency check with the ideal equal to a maximal order::
sage: B = BrandtModule(17); basis = sage.modular.quatalg.brandt.basis_for_left_ideal(B.maximal_order(), B.maximal_order().basis()) sage: sage.modular.quatalg.brandt.right_order(B.maximal_order(), basis) Order of Quaternion Algebra (-17, -3) with base ring Rational Field with basis (1/2 + 1/6*j + 2/3*k, 1/2*i + 1/2*k, 1/3*j + 1/3*k, k) sage: basis [1/2 + 1/6*j + 2/3*k, 1/2*i + 1/2*k, 1/3*j + 1/3*k, k]
sage: B = BrandtModule(17); A = B.quaternion_algebra(); i,j,k = A.gens() sage: basis = sage.modular.quatalg.brandt.basis_for_left_ideal(B.maximal_order(), [i*j-j]) sage: sage.modular.quatalg.brandt.right_order(B.maximal_order(), basis) Order of Quaternion Algebra (-17, -3) with base ring Rational Field with basis (1/2 + 1/2*i + 1/2*j + 17/2*k, i, j + 8*k, 9*k) """ # Compute matrix of multiplication by each element of the basis.
# I = matrix with rows the given basis for I
# psi = matrix of right multiplication on each basis element
# invert them
# apply the four inverses to I
# The right order is the intersection of the row span of the W with the row span of B.
def quaternion_order_with_given_level(A, level): """ Return an order in the quaternion algebra A with given level. (Implemented only when the base field is the rational numbers.)
INPUT:
- ``level`` -- The level of the order to be returned. Currently this is only implemented when the level is divisible by at most one power of a prime that ramifies in this quaternion algebra.
EXAMPLES::
sage: from sage.modular.quatalg.brandt import quaternion_order_with_given_level, maximal_order sage: A.<i,j,k> = QuaternionAlgebra(5) sage: level = 2 * 5 * 17 sage: O = quaternion_order_with_given_level(A, level) sage: M = maximal_order(A) sage: L = O.free_module() sage: N = M.free_module() sage: L.index_in(N) == level/5 #check that the order has the right index in the maximal order True """ raise NotImplementedError("base field must be rational numbers")
raise NotImplementedError("Currently this algorithm only works when the quaternion algebra is only ramified at one finite prime.")
# (The algorithm we use is similar to that in Magma (by David Kohel).) # in the following magma code, M denotes is the level
for p in A.ramified_primes(): if level % p**2 == 0: raise NotImplementedError("Currently sage can only compute orders whose level is divisible by at most one power of any prime that ramifies in the quaternion algebra")
P = basis_for_left_ideal(O, [N1] + [x*y - y*x for x, y in cartesian_product_iterator([A.basis(), A.basis()]) ]) O = A.quaternion_order(P)
#x = O.random_element((-p/2).floor(), (p/2).ceil())
class BrandtSubmodule(HeckeSubmodule): def _repr_(self): """ Return string representation of this Brandt submodule.
EXAMPLES::
sage: BrandtModule(11)[0]._repr_() 'Subspace of dimension 1 of Brandt module of dimension 2 of level 11 of weight 2 over Rational Field' """
class BrandtModuleElement(HeckeModuleElement): def __init__(self, parent, x): """ EXAMPLES::
sage: B = BrandtModule(37) sage: x = B([1,2,3]); x (1, 2, 3) sage: parent(x) Brandt module of dimension 3 of level 37 of weight 2 over Rational Field """
def _richcmp_(self, other, op): """ EXAMPLES::
sage: B = BrandtModule(13,5) sage: B.0 (1, 0, 0, 0, 0, 0) sage: B.0 == B.1 False sage: B.0 == 0 False sage: B(0) == 0 True sage: B.0 + 2*B.1 == 2*B.1 + B.0 True sage: loads(dumps(B.0)) == B.0 True """
def monodromy_pairing(self, x): """ Return the monodromy pairing of ``self`` and ``x``.
EXAMPLES::
sage: B = BrandtModule(5,13) sage: B.monodromy_weights() (1, 3, 1, 1, 1, 3) sage: (B.0 + B.1).monodromy_pairing(B.0 + B.1) 4
TESTS:
One check for :trac:`12866`::
sage: Br = BrandtModule(2,7) sage: g1, g2 = Br.basis() sage: g = g1 - g2 sage: g.monodromy_pairing(g) 6 """
def __mul__(self, right): """ Return the monodromy pairing of ``self`` and ``right``.
EXAMPLES::
sage: B = BrandtModule(7,10) sage: B.monodromy_weights() (1, 1, 1, 2, 1, 1, 2, 1, 1, 1) sage: B.0 * B.0 1 sage: B.3 * B.3 2 sage: (B.0+B.3) * (B.0 + B.1 + 2*B.3) 5 """
def _add_(self, right): """ Return the sum of ``self`` and ``right``.
EXAMPLES::
sage: B = BrandtModule(11) sage: B.0 + B.1 # indirect doctest (1, 1) """
def _sub_(self, right): """ Return the difference of ``self`` and ``right``.
EXAMPLES::
sage: B = BrandtModule(11) sage: B.0 - B.1 # indirect doctest (1, -1) """
def _neg_(self): """ Return the opposite of ``self``.
EXAMPLES::
sage: B = BrandtModule(11) sage: -B.0 # indirect doctest (-1, 0) """
@richcmp_method class BrandtModule_class(AmbientHeckeModule): """ A Brandt module.
EXAMPLES::
sage: BrandtModule(3, 10) Brandt module of dimension 4 of level 3*10 of weight 2 over Rational Field """ def __init__(self, N, M, weight, base_ring): """ INPUT:
- N -- ramification number (coprime to M) - M -- auxiliary level - weight -- integer 2 - base_ring -- the base ring
EXAMPLES::
sage: BrandtModule(3, 5, weight=2, base_ring=ZZ) Brandt module of dimension 2 of level 3*5 of weight 2 over Integer Ring """ raise NotImplementedError("right now N must be prime")
Element = BrandtModuleElement
def _submodule_class(self): """ Return the Python class of submodules of this ambient Brandt module.
EXAMPLES::
sage: BrandtModule(37)._submodule_class() <class 'sage.modular.quatalg.brandt.BrandtSubmodule'> """
@cached_method def free_module(self): """ Return the underlying free module of the Brandt module.
EXAMPLES::
sage: B = BrandtModule(10007,389) sage: B.free_module() Vector space of dimension 325196 over Rational Field """
def N(self): """ Return ramification level `N`.
EXAMPLES::
sage: BrandtModule(7,5,2,ZZ).N() 7 """
def M(self): """ Return the auxiliary level (prime to `p` part) of the quaternion order used to compute this Brandt module.
EXAMPLES::
sage: BrandtModule(7,5,2,ZZ).M() 5 """
def character(self): r""" The character of this space.
Always trivial.
EXAMPLES::
sage: BrandtModule(11,5).character() Dirichlet character modulo 55 of conductor 1 mapping 12 |--> 1, 46 |--> 1 """
def _repr_(self): """ Return string representation of this Brandt module.
EXAMPLES::
sage: BrandtModule(7,5,2,ZZ)._repr_() 'Brandt module of dimension 4 of level 7*5 of weight 2 over Integer Ring' """ self.rank(), self.__N, aux, self.weight(), self.base_ring())
def __richcmp__(self, other, op): r""" Compare ``self`` to ``other``.
EXAMPLES::
sage: BrandtModule(37, 5, 2, ZZ) == BrandtModule(37, 5, 2, QQ) False sage: BrandtModule(37, 5, 2, ZZ) == BrandtModule(37, 5, 2, ZZ) True sage: BrandtModule(37, 5, 2, ZZ) == loads(dumps(BrandtModule(37, 5, 2, ZZ))) True """
(other.__M, other.__N, other.weight(), other.base_ring()), op)
@cached_method def quaternion_algebra(self): r""" Return the quaternion algebra `A` over `\QQ` ramified precisely at `p` and infinity used to compute this Brandt module.
EXAMPLES::
sage: BrandtModule(997).quaternion_algebra() Quaternion Algebra (-2, -997) with base ring Rational Field sage: BrandtModule(2).quaternion_algebra() Quaternion Algebra (-1, -1) with base ring Rational Field sage: BrandtModule(3).quaternion_algebra() Quaternion Algebra (-1, -3) with base ring Rational Field sage: BrandtModule(5).quaternion_algebra() Quaternion Algebra (-2, -5) with base ring Rational Field sage: BrandtModule(17).quaternion_algebra() Quaternion Algebra (-17, -3) with base ring Rational Field """
@cached_method def maximal_order(self): """ Return a maximal order in the quaternion algebra associated to this Brandt module.
EXAMPLES::
sage: BrandtModule(17).maximal_order() Order of Quaternion Algebra (-17, -3) with base ring Rational Field with basis (1/2 + 1/2*j, 1/2*i + 1/2*k, -1/3*j - 1/3*k, k) sage: BrandtModule(17).maximal_order() is BrandtModule(17).maximal_order() True """
@cached_method def order_of_level_N(self): """ Return Eichler order of level `N = p^{2 r + 1} M` in the quaternion algebra.
EXAMPLES::
sage: BrandtModule(7).order_of_level_N() Order of Quaternion Algebra (-1, -7) with base ring Rational Field with basis (1/2 + 1/2*j, 1/2*i + 1/2*k, j, k) sage: BrandtModule(7,13).order_of_level_N() Order of Quaternion Algebra (-1, -7) with base ring Rational Field with basis (1/2 + 1/2*j + 12*k, 1/2*i + 9/2*k, j + 11*k, 13*k) sage: BrandtModule(7,3*17).order_of_level_N() Order of Quaternion Algebra (-1, -7) with base ring Rational Field with basis (1/2 + 1/2*j + 35*k, 1/2*i + 65/2*k, j + 19*k, 51*k) """
def cyclic_submodules(self, I, p): """ Return a list of rescaled versions of the fractional right ideals `J` such that `J` contains `I` and the quotient has group structure the product of two cyclic groups of order `p`.
We emphasize again that `J` is rescaled to be integral.
INPUT:
- `I` -- ideal I in R = self.order_of_level_N() - `p` -- prime `p` coprime to self.level()
OUTPUT:
list of the `p+1` fractional right R-ideals that contain I such that J/I is GF(p) x GF(p).
EXAMPLES::
sage: B = BrandtModule(11) sage: I = B.order_of_level_N().unit_ideal() sage: B.cyclic_submodules(I, 2) [Fractional ideal (1/2 + 3/2*j + k, 1/2*i + j + 1/2*k, 2*j, 2*k), Fractional ideal (1/2 + 1/2*i + 1/2*j + 1/2*k, i + k, j + k, 2*k), Fractional ideal (1/2 + 1/2*j + k, 1/2*i + j + 3/2*k, 2*j, 2*k)] sage: B.cyclic_submodules(I, 3) [Fractional ideal (1/2 + 1/2*j, 1/2*i + 5/2*k, 3*j, 3*k), Fractional ideal (1/2 + 3/2*j + 2*k, 1/2*i + 2*j + 3/2*k, 3*j, 3*k), Fractional ideal (1/2 + 3/2*j + k, 1/2*i + j + 3/2*k, 3*j, 3*k), Fractional ideal (1/2 + 5/2*j, 1/2*i + 1/2*k, 3*j, 3*k)] sage: B.cyclic_submodules(I, 11) Traceback (most recent call last): ... ValueError: p must be coprime to the level """ raise ValueError("p must be a prime")
# step 1: Compute alpha, beta, and the matrix of their action on I/pI. # NOTE: Move this code to orders once we have it all working...
# If the quadratic polynomial over GF(p) given by # X^2 - alpha.reduced_trace() * X + alpha.reduced_norm() # is not irreducible, we try again with a new element. # special case p == 2, since there is a unique quadratic irreducible poly. else: # check if the discriminant is a square -- if so, poly is reducible
# right multiplication by X changes something to be written # in terms of the basis for I.
# Compute the matrix of right multiplication by alpha acting on # our fixed choice of basis for this ideal.
# step 2: Find j such that if f=I[j], then mod 2 we have span(I[0],alpha*I[i]) # has trivial intersection with span(I[j],alpha*I[j]). # # In terms of our matrices alpha, beta, we can now think of I/p*I # as being the GF(p)^4 that M_alpha and M_beta naturally act on, # and I[0], I[1], I[2], I[3] correspond to the standard basis. # # We try each of the standard basis vectors.
# step 3: Enumerate the elements of P^1(GF(p^2)), recording each # cyclic submodule of degree p. rational_quaternions_from_integral_matrix_and_denom
def hecke_matrix(self, n, algorithm='default', sparse=False, B=None): """ Return the matrix of the `n`-th Hecke operator.
INPUT:
- `n` -- integer
- ``algorithm`` -- string (default: 'default')
- 'default' -- let Sage guess which algorithm is best
- 'direct' -- use cyclic subideals (generally much better when you want few Hecke operators and the dimension is very large); uses 'theta' if n divides the level.
- 'brandt' -- use Brandt matrices (generally much better when you want many Hecke operators and the dimension is very small; bad when the dimension is large)
- ``sparse`` -- bool (default: ``False``)
- `B` -- integer or ``None`` (default: ``None``); in direct algorithm, use theta series to this precision as an initial check for equality of ideal classes.
EXAMPLES::
sage: B = BrandtModule(3,7); B.hecke_matrix(2) [0 3] [1 2] sage: B.hecke_matrix(5, algorithm='brandt') [0 6] [2 4] sage: t = B.hecke_matrix(11, algorithm='brandt', sparse=True); t [ 6 6] [ 2 10] sage: type(t) <type 'sage.matrix.matrix_rational_sparse.Matrix_rational_sparse'> sage: B.hecke_matrix(19, algorithm='direct', B=2) [ 8 12] [ 4 16] """ raise IndexError("n must be positive.") # already trivially know the Hecke operator in this case algorithm = 'brandt'
else: raise ValueError("unknown algorithm '%s'"%algorithm)
def _compute_hecke_matrix_prime(self, p, sparse=False, B=None): """ Return matrix of the `p`-th Hecke operator on self. The matrix is always computed using the direct algorithm.
INPUT:
- `p` -- prime number
- `B` -- integer or None (default: None); in direct algorithm, use theta series to this precision as an initial check for equality of ideal classes.
- ``sparse`` -- bool (default: False); whether matrix should be sparse
EXAMPLES::
sage: B = BrandtModule(37) sage: t = B._compute_hecke_matrix_prime(2); t [1 1 1] [1 0 2] [1 2 0] sage: type(t) <type 'sage.matrix.matrix_rational_dense.Matrix_rational_dense'> sage: type(B._compute_hecke_matrix_prime(2,sparse=True)) <type 'sage.matrix.matrix_rational_sparse.Matrix_rational_sparse'> """
def _compute_hecke_matrix_directly(self, n, B=None, sparse=False): """ Given an integer `n` coprime to the level, return the matrix of the n-th Hecke operator on self, computed on our fixed basis by directly using the definition of the Hecke action in terms of fractional ideals.
INPUT:
- `n` -- integer, coprime to level
- ``sparse`` -- bool (default: False); whether matrix should be sparse
EXAMPLES::
sage: B = BrandtModule(37) sage: t = B._compute_hecke_matrix_directly(2); t [1 1 1] [1 0 2] [1 2 0] sage: type(t) <type 'sage.matrix.matrix_rational_dense.Matrix_rational_dense'> sage: type(B._compute_hecke_matrix_directly(2,sparse=True)) <type 'sage.matrix.matrix_rational_sparse.Matrix_rational_sparse'>
You can't compute the Hecke operator for n not coprime to the level using this function::
sage: B._compute_hecke_matrix_directly(37) Traceback (most recent call last): ... ValueError: n must be coprime to the level
The generic function (which uses theta series) does work, though::
sage: B.hecke_matrix(37) [1 0 0] [0 0 1] [0 1 0]
An example where the Hecke operator isn't symmetric::
sage: B = BrandtModule(43) sage: B._compute_hecke_matrix_directly(2) [1 2 0 0] [1 0 1 1] [0 1 0 2] [0 1 2 0] sage: B._compute_hecke_matrix_brandt(2) [1 2 0 0] [1 0 1 1] [0 1 0 2] [0 1 2 0] """
# For rigor it does not matter at all what bound we chose. # This B is used only for the first phase of checking equality # of ideals modulo equivalence -- we always provably check # equivalence if the theta series are the same up to this # bound.
# I think the runtime of this algorithm is now dominated by # computing theta series of ideals. The computation of # cyclic submodules is a lower order term.
# TODO: temporary!! -- it's not sufficiently *optimized* to be # sure this is best in these cases. #if gcd(2*d*q,n) == 1: # use_fast_alg = True #else: # use_fast_alg = False
v = C[r].cyclic_right_subideals(n) else: else:
@cached_method def _theta_dict(self, B): """ Return a dictionary from theta series vectors of degree `B` to list of integers `i`, where the key is the vector of coefficients of the normalized theta series of the `i`th right ideal, as indexed by ``self.right_ideals()``.
INPUT:
- `B` -- positive integer, precision of theta series vectors
OUTPUT:
dictionary
EXAMPLES:
In this example the theta series determine the ideal classes::
sage: B = BrandtModule(5,11); B Brandt module of dimension 4 of level 5*11 of weight 2 over Rational Field sage: sorted(list(B._theta_dict(5).items())) [((1, 0, 0, 4, 0), [3]), ((1, 0, 0, 4, 2), [2]), ((1, 0, 2, 0, 6), [1]), ((1, 2, 4, 0, 6), [0])]
In this example, the theta series does not determine the ideal class::
sage: sorted(list(BrandtModule(37)._theta_dict(6).items())) [((1, 0, 2, 2, 6, 4), [1, 2]), ((1, 2, 2, 4, 2, 4), [0])] """ else:
def _compute_hecke_matrix_brandt(self, n, sparse=False): """ Return the n-th Hecke matrix, computed using Brandt matrices (theta series).
When the n-th Hecke operator is requested, we computed theta series to precision `2n+20`, since it only takes slightly longer, and this means that any Hecke operator `T_m` can quickly be computed, for `m<2n+20`.
INPUT:
- n -- integer, coprime to level - sparse -- bool (default: False); whether matrix should be sparse
EXAMPLES::
sage: B = BrandtModule(3,17) sage: B._compute_hecke_matrix_brandt(3) [0 1 0 0] [1 0 0 0] [0 0 0 1] [0 0 1 0] sage: B._compute_hecke_matrix_brandt(5) [4 1 1 0] [1 4 0 1] [2 0 2 2] [0 2 2 2] sage: B._compute_hecke_matrix_brandt(5).fcp() (x - 6) * (x - 3) * (x^2 - 3*x - 2)
""" # we go out to 2*n+20 for efficiency, since it takes only a # little longer, but saves a lot of time if one computes # successive Hecke operators, which is a very common thing to # do.
@cached_method def _smallest_good_prime(self): """ Return the smallest prime number that does not divide the level.
EXAMPLES::
sage: BrandtModule(17,6)._smallest_good_prime() 5 """
@cached_method def right_ideals(self, B=None): """ Return sorted tuple of representatives for the equivalence classes of right ideals in ``self``.
OUTPUT:
sorted tuple of fractional ideals
EXAMPLES::
sage: B = BrandtModule(23) sage: B.right_ideals() (Fractional ideal (2 + 2*j, 2*i + 2*k, 4*j, 4*k), Fractional ideal (2 + 2*j, 2*i + 6*k, 8*j, 8*k), Fractional ideal (2 + 10*j + 8*k, 2*i + 8*j + 6*k, 16*j, 16*k))
TESTS::
sage: B = BrandtModule(1009) sage: Is = B.right_ideals() sage: n = len(Is) sage: prod(not Is[i].is_equivalent(Is[j]) for i in range(n) for j in range(i)) 1 """
else:
@cached_method def _ideal_products(self, diagonal_only=False): """ Return all products of right ideals, which are used in computing the Brandt matrices.
This function is used internally by the Brandt matrices algorithms.
INPUT:
- ``diagonal_only`` -- bool (default: ``False``) if ``True`` returns only the diagonal ideal products
OUTPUT:
list of ideals
EXAMPLES::
sage: B = BrandtModule(37) sage: B._ideal_products() [[Fractional ideal (8 + 8*j + 8*k, 4*i + 8*j + 4*k, 16*j, 16*k)], [Fractional ideal (8 + 24*j + 8*k, 4*i + 8*j + 4*k, 32*j, 32*k), Fractional ideal (16 + 16*j + 48*k, 4*i + 8*j + 36*k, 32*j + 32*k, 64*k)], [Fractional ideal (8 + 24*j + 24*k, 4*i + 24*j + 4*k, 32*j, 32*k), Fractional ideal (8 + 4*i + 16*j + 28*k, 8*i + 16*j + 8*k, 32*j, 64*k), Fractional ideal (16 + 16*j + 16*k, 4*i + 24*j + 4*k, 32*j + 32*k, 64*k)]] sage: B._ideal_products(diagonal_only=True) [Fractional ideal (8 + 8*j + 8*k, 4*i + 8*j + 4*k, 16*j, 16*k), Fractional ideal (16 + 16*j + 48*k, 4*i + 8*j + 36*k, 32*j + 32*k, 64*k), Fractional ideal (16 + 16*j + 16*k, 4*i + 24*j + 4*k, 32*j + 32*k, 64*k)] """ return matrix(self.base_ring()[['q']], 0)
# 1. Compute the diagonal
# 2. Compute the rest of the products
def _brandt_series_vectors(self, prec=None): """ Return Brandt series coefficient vectors out to precision *at least* prec.
EXAMPLES::
sage: B = BrandtModule(37, use_cache=False) sage: B._brandt_series_vectors(5) [[(1/2, 1, 1, 2, 1), (1/2, 0, 1, 1, 3), (1/2, 0, 1, 1, 3)], [(1/2, 0, 1, 1, 3), (1/2, 1, 0, 0, 3), (1/2, 0, 2, 3, 1)], [(1/2, 0, 1, 1, 3), (1/2, 0, 2, 3, 1), (1/2, 1, 0, 0, 3)]]
If you have computed to higher precision and ask for a lower precision, the higher precision is still returned::
sage: B._brandt_series_vectors(2) [[(1/2, 1, 1, 2, 1), (1/2, 0, 1, 1, 3), (1/2, 0, 1, 1, 3)], [(1/2, 0, 1, 1, 3), (1/2, 1, 0, 0, 3), (1/2, 0, 2, 3, 1)], [(1/2, 0, 1, 1, 3), (1/2, 0, 2, 3, 1), (1/2, 1, 0, 0, 3)]] """ raise ValueError("prec must be at least 2") return [[]]
# 1. Compute the theta series for x in self._ideal_products()]
# 2. Compute the number e_j
# 3. Make the brandt matrix series
def brandt_series(self, prec, var='q'): r""" Return matrix of power series `\sum T_n q^n` to the given precision.
Note that the Hecke operators in this series are always over `\QQ`, even if the base ring of this Brandt module is not `\QQ`.
INPUT:
- ``prec`` -- positive integer - ``var`` -- string (default: `q`)
OUTPUT:
matrix of power series with coefficients in `\QQ`
EXAMPLES::
sage: B = BrandtModule(11) sage: B.brandt_series(2) [1/4 + q + O(q^2) 1/4 + O(q^2)] [ 1/6 + O(q^2) 1/6 + q + O(q^2)] sage: B.brandt_series(5) [1/4 + q + q^2 + 2*q^3 + 5*q^4 + O(q^5) 1/4 + 3*q^2 + 3*q^3 + 3*q^4 + O(q^5)] [ 1/6 + 2*q^2 + 2*q^3 + 2*q^4 + O(q^5) 1/6 + q + q^3 + 4*q^4 + O(q^5)]
Asking for a smaller precision works::
sage: B.brandt_series(3) [1/4 + q + q^2 + O(q^3) 1/4 + 3*q^2 + O(q^3)] [ 1/6 + 2*q^2 + O(q^3) 1/6 + q + O(q^3)] sage: B.brandt_series(3,var='t') [1/4 + t + t^2 + O(t^3) 1/4 + 3*t^2 + O(t^3)] [ 1/6 + 2*t^2 + O(t^3) 1/6 + t + O(t^3)] """
@cached_method def eisenstein_subspace(self): """ Return the 1-dimensional subspace of ``self`` on which the Hecke operators `T_p` act as `p+1` for `p` coprime to the level.
NOTE: This function assumes that the base field has characteristic 0.
EXAMPLES::
sage: B = BrandtModule(11); B.eisenstein_subspace() Subspace of dimension 1 of Brandt module of dimension 2 of level 11 of weight 2 over Rational Field sage: B.eisenstein_subspace() is B.eisenstein_subspace() True sage: BrandtModule(3,11).eisenstein_subspace().basis() ((1, 1),) sage: BrandtModule(7,10).eisenstein_subspace().basis() ((1, 1, 1, 1/2, 1, 1, 1/2, 1, 1, 1),) sage: BrandtModule(7,10,base_ring=ZZ).eisenstein_subspace().basis() ((2, 2, 2, 1, 2, 2, 1, 2, 2, 2),) """ raise ValueError("characteristic must be 0") # cut down until we get a 1-d space using Hecke operators T_p # with p coprime to the level.
def is_cuspidal(self): r""" Returns whether ``self`` is cuspidal, i.e. has no Eisenstein part.
EXAMPLES::
sage: B = BrandtModule(3, 4) sage: B.is_cuspidal() False sage: B.eisenstein_subspace() Brandt module of dimension 1 of level 3*4 of weight 2 over Rational Field """
@cached_method def monodromy_weights(self): r""" Return the weights for the monodromy pairing on this Brandt module.
The weights are associated to each ideal class in our fixed choice of basis. The weight of an ideal class `[I]` is half the number of units of the right order `I`.
NOTE: The base ring must be `\QQ` or `\ZZ`.
EXAMPLES::
sage: BrandtModule(11).monodromy_weights() (2, 3) sage: BrandtModule(37).monodromy_weights() (1, 1, 1) sage: BrandtModule(43).monodromy_weights() (2, 1, 1, 1) sage: BrandtModule(7,10).monodromy_weights() (1, 1, 1, 2, 1, 1, 2, 1, 1, 1) sage: BrandtModule(5,13).monodromy_weights() (1, 3, 1, 1, 1, 3) sage: BrandtModule(2).monodromy_weights() (12,) sage: BrandtModule(2,7).monodromy_weights() (3, 3) """ # Before normalization, # # theta(R) = 1 + e*q + .... # # where e is the number of units in the order R. # # Since the theta series may be normalized as # # c * theta(R) = a[0] + a[1]*q + ... # # we recover e = a[1]/a[0] regardless of normalization.
############################################################################# # Benchmarking ############################################################################# def benchmark_magma(levels, silent=False): """ INPUT:
- ``levels`` -- list of pairs `(p,M)` where `p` is a prime not dividing `M` - ``silent`` -- bool, default ``False``; if ``True`` suppress printing during computation
OUTPUT:
list of 4-tuples ('magma', p, M, tm), where tm is the CPU time in seconds to compute T2 using Magma
EXAMPLES::
sage: a = sage.modular.quatalg.brandt.benchmark_magma([(11,1), (37,1), (43,1), (97,1)]) # optional - magma ('magma', 11, 1, ...) ('magma', 37, 1, ...) ('magma', 43, 1, ...) ('magma', 97, 1, ...) sage: a = sage.modular.quatalg.brandt.benchmark_magma([(11,2), (37,2), (43,2), (97,2)]) # optional - magma ('magma', 11, 2, ...) ('magma', 37, 2, ...) ('magma', 43, 2, ...) ('magma', 97, 2, ...) """ ans = [] from sage.interfaces.all import magma for p, M in levels: t = magma.cputime() magma.eval('HeckeOperator(BrandtModule(%s, %s),2)'%(p,M)) tm = magma.cputime(t) v = ('magma', p, M, tm) if not silent: print(v) ans.append(v) return ans
def benchmark_sage(levels, silent=False): """ INPUT:
- ``levels`` -- list of pairs `(p,M)` where `p` is a prime not dividing `M` - ``silent`` -- bool, default ``False``; if ``True`` suppress printing during computation
OUTPUT:
list of 4-tuples ('sage', p, M, tm), where tm is the CPU time in seconds to compute T2 using Sage
EXAMPLES::
sage: a = sage.modular.quatalg.brandt.benchmark_sage([(11,1), (37,1), (43,1), (97,1)]) ('sage', 11, 1, ...) ('sage', 37, 1, ...) ('sage', 43, 1, ...) ('sage', 97, 1, ...) sage: a = sage.modular.quatalg.brandt.benchmark_sage([(11,2), (37,2), (43,2), (97,2)]) ('sage', 11, 2, ...) ('sage', 37, 2, ...) ('sage', 43, 2, ...) ('sage', 97, 2, ...) """ |