Sure. I'm essentially following the same technique used in Seng-Kiat Chua
and San Ling's article "On the Rational Cuspidal Subgroup and the Rational
Torsion Points of J_0(pq)", and my thesis (arXiv:0707.0437).
Let L be the space of functions on the upper half plane generated
(multiplicatively) by eta function $\eta_d$ as d runs through divisor of
N. Then we can give an explicit isomorphism between (L \tensor \QQ) and
(C \tensor \QQ), where C is the group of divisors supported on the cusps
(call this isomorphism \Lambda). Furthermore, Ligozat has a proposition
on precisely when an element in L is a function on $X_0(N)$ (Proposition
2.2.3 in my thesis). Using this proposition, and the isomorphism Lambda, one
can generate the lattice of all principle divisors in C, and quotiening
out degree zero divisors by these principle divisors gives the cuspidal
subgroup.

I'm enclosing the code that does this in this email, although
as it is written is really hackish. The example is
sage: load "cuspidal_subgroup.py"
sage: CS=CuspidalGroup(15)
sage: CS.PrincipleDivisor

Free module of degree 4 and rank 3 over Integer Ring
Echelon basis matrix:
[ 1  3  1 -5]
[ 0  4  0 -4]
[ 0  0  2 -2]

I don't know how to quotient two lattices in sage (I'm afraid it might
not be implemented yet), but looking at the principledivisor, you can
see the cuspidal subgroup is Z/2 \times Z/4.

Soroosh

On Fri, Jul 13, 2007 at 10:31:21PM +0100, John Cremona wrote:
>
> And to me please!!
>
>
> On 7/13/07, William Stein <[EMAIL PROTECTED]> wrote:
> >
> > On 7/13/07, William Stein <[EMAIL PROTECTED]> wrote:
> > > > I was writing a function to calculate the structure of cuspidal
subgroup of
> > > > J_0(N) without using modular symbols. I came across the following
puzzling
> >
> > How are you doing that?  I didn't know there was an algorithm to compute
> > the structure of J_0(N)(Qbar)_tor without using modular
symbols.  Interesting.
> > Could you explain it to me?
> >
> > William
> >
> > >
> >
>
>
> --
> John Cremona
>
> 
def degree_zero_divisor(rank,den=1):
    amb=FreeModule(ZZ,rank)
    B=amb.basis()
    gens=[]
    for i in range(rank):
        gens.append((B[0]-B[i])*den)
    return amb.submodule(gens)

def binary_to_prod(n,facs):
    prod=1
    for p in facs:
        prod=prod*(p**(1-n%2))
        n=n.div(2)
    return prod

class CuspidalGroup():
    def __init__(self, N):
        """
        INPUT:
            N -- Level for the cuspidal group. Currently only defined for
                 N square free.
        This group is denoted stored as two lattice: degree zero cusps,
        and principle cusps.
        """
        if not N.is_squarefree():
            raise TypeError, "N (=%s) should be square free."%N

        self.__level=N
        self.__facs=N.factor()
        self.__den=prod([p[0]**2-1 for p in self.__facs])
        rank=(2**len(self.__facs))
        self.__rank=rank
        self.__divisor_lattice=degree_zero_divisor(rank)
        ambZ=self.__divisor_lattice.ambient_module()
        ambQ=self.__divisor_lattice.ambient_vector_space()
        """
        The divisor lattice is generated by P_{\prod pi_i^{a_i}}.
        Such elements are denoted by \sum 2^i a_i. So B[0]=P_1.
        """
        LambdaZ=self._generate_lambda()
        LambdaQ=LambdaZ*(24/self.__den)

        self.Cond1=phi_inverse(LambdaQ)
        self.Cond4=degree_zero_divisor(rank)
        self.Cond5=phi_inverse
(self._generate_cond5()*LambdaZ*(12/self.__den))
        Lat0=self.Cond1.intersection(self.Cond4).intersection(self.Cond5)
        self.PrincipleDivisor=Lat0


    def _generate_lambda(self):
        """
        Generates the lambda matrix, which maps cusps to eta functions.
        """
        rank=self.__rank
        Lambda=Matrix(QQ,rank)
        facs=[-p[0] for p in self.__facs]
        for i in range(rank):
            for j in range(rank):
                Lambda[i,j]=binary_to_prod(ZZ(i)._xor(ZZ(j)),facs)
        return Lambda

    def _generate_cond5(self):
        l=len(self.__facs)
        M=Matrix(QQ,l,self.__rank)
        for j in range(l):
            divpoint=2**j
            for i in range(2**(l-1)):
                x=2*divpoint*ZZ(i).div(divpoint)+divpoint+ZZ(i).mod(divpoint)
                M[j,x]=1

        return M

"""
This should go somewhere reasonable, but for now, let's just have it
as a random function.
"""
def row_inverse(phi):
    """
    Input:
        phi: \ZZ^n \rightarrow \QQ, given as an n-dimensional rational
vector.
    Output:
        phi^{-1}(\ZZ).
    This function has no error checking in it yet.
    Also, this algorithm is prone to coefficient explosion. I will worry
about
    that later. I think we can do everything modulo denomanitor. That would
    solve the coefficient explosion.
    """
    den=phi.denominator()
    phiZ=(den*phi).change_ring(ZZ)
    rank=phi.degree()
    M=Matrix(ZZ,rank)
    g=[den]
    p=[]
    q=[]
    for i in range(rank):
        (a,b,c)=XGCD(g[i],phiZ[i])
        g.append(a)
        p.append(b)
        q.append(c)
        M[i,i]=g[i].div(g[i+1])
        K=phiZ[i].div(g[i+1])
        for j in range(i-1,-1,-1):
            M[i,j]=-K*q[j]
            K=(K*p[j])%den
    return M.row_space()

def phi_inverse(phi):
    v=phi.rows()[0]
    M=row_inverse(v)
    for v in phi.rows():
        M=M.intersection(row_inverse(v))
    return M

--~--~---------~--~----~------------~-------~--~----~
To post to this group, send email to sage-devel@googlegroups.com
To unsubscribe from this group, send email to [EMAIL PROTECTED]
For more options, visit this group at http://groups.google.com/group/sage-devel
URLs: http://sage.scipy.org/sage/ and http://modular.math.washington.edu/sage/
-~----------~----~----~----~------~----~------~--~---

Reply via email to