Elliptic curves over number fields¶
An elliptic curve \(E\) over a number field \(K\) can be given
by a Weierstrass equation whose coefficients lie in \(K\) or by
using base_extend
on an elliptic curve defined over a subfield.
One major difference to elliptic curves over \(\QQ\) is that there might not exist a global minimal equation over \(K\), when \(K\) does not have class number one. Another difference is the lack of understanding of modularity for general elliptic curves over general number fields.
Currently Sage can obtain local information about \(E/K_v\) for finite places \(v\), it has an interface to Denis Simon’s script for 2-descent, it can compute the torsion subgroup of the Mordell-Weil group \(E(K)\), and it can work with isogenies defined over \(K\).
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1)
sage: E = EllipticCurve([0, 4+i])
sage: E.discriminant()
-3456*i - 6480
sage: P= E([i,2])
sage: P+P
(-2*i + 9/16 : -9/4*i - 101/64 : 1)
>>> from sage.all import *
>>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1)
>>> E = EllipticCurve([Integer(0), Integer(4)+i])
>>> E.discriminant()
-3456*i - 6480
>>> P= E([i,Integer(2)])
>>> P+P
(-2*i + 9/16 : -9/4*i - 101/64 : 1)
sage: E.has_good_reduction(2 + i)
True
sage: E.local_data(4+i)
Local data at Fractional ideal (i + 4):
Reduction type: bad additive
Local minimal model: Elliptic Curve defined by y^2 = x^3 + (i+4)
over Number Field in i with defining polynomial x^2 + 1
Minimal discriminant valuation: 2
Conductor exponent: 2
Kodaira Symbol: II
Tamagawa Number: 1
sage: E.tamagawa_product_bsd()
1
>>> from sage.all import *
>>> E.has_good_reduction(Integer(2) + i)
True
>>> E.local_data(Integer(4)+i)
Local data at Fractional ideal (i + 4):
Reduction type: bad additive
Local minimal model: Elliptic Curve defined by y^2 = x^3 + (i+4)
over Number Field in i with defining polynomial x^2 + 1
Minimal discriminant valuation: 2
Conductor exponent: 2
Kodaira Symbol: II
Tamagawa Number: 1
>>> E.tamagawa_product_bsd()
1
sage: E.simon_two_descent()
(1, 1, [(i : 2 : 1)])
>>> from sage.all import *
>>> E.simon_two_descent()
(1, 1, [(i : 2 : 1)])
sage: E.torsion_order()
1
>>> from sage.all import *
>>> E.torsion_order()
1
sage: E.isogenies_prime_degree(3)
[Isogeny of degree 3
from Elliptic Curve defined by y^2 = x^3 + (i+4)
over Number Field in i with defining polynomial x^2 + 1
to Elliptic Curve defined by y^2 = x^3 + (-27*i-108)
over Number Field in i with defining polynomial x^2 + 1]
>>> from sage.all import *
>>> E.isogenies_prime_degree(Integer(3))
[Isogeny of degree 3
from Elliptic Curve defined by y^2 = x^3 + (i+4)
over Number Field in i with defining polynomial x^2 + 1
to Elliptic Curve defined by y^2 = x^3 + (-27*i-108)
over Number Field in i with defining polynomial x^2 + 1]
AUTHORS:
Robert Bradshaw 2007
John Cremona
Chris Wuthrich
REFERENCE:
[Sil] Silverman, Joseph H. The arithmetic of elliptic curves. Second edition. Graduate Texts in Mathematics, 106. Springer, 2009.
[Sil2] Silverman, Joseph H. Advanced topics in the arithmetic of elliptic curves. Graduate Texts in Mathematics, 151. Springer, 1994.
- class sage.schemes.elliptic_curves.ell_number_field.EllipticCurve_number_field(K, ainvs)[source]¶
Bases:
EllipticCurve_field
Elliptic curve over a number field.
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: EllipticCurve([i, i - 1, i + 1, 24*i + 15, 14*i + 35]) Elliptic Curve defined by y^2 + i*x*y + (i+1)*y = x^3 + (i-1)*x^2 + (24*i+15)*x + (14*i+35) over Number Field in i with defining polynomial x^2 + 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EllipticCurve([i, i - Integer(1), i + Integer(1), Integer(24)*i + Integer(15), Integer(14)*i + Integer(35)]) Elliptic Curve defined by y^2 + i*x*y + (i+1)*y = x^3 + (i-1)*x^2 + (24*i+15)*x + (14*i+35) over Number Field in i with defining polynomial x^2 + 1
- base_extend(R)[source]¶
Return the base extension of
self
to \(R\).EXAMPLES:
sage: E = EllipticCurve('11a3') sage: K = QuadraticField(-5, 'a') sage: E.base_extend(K) Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 over Number Field in a with defining polynomial x^2 + 5 with a = 2.236067977499790?*I
>>> from sage.all import * >>> E = EllipticCurve('11a3') >>> K = QuadraticField(-Integer(5), 'a') >>> E.base_extend(K) Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 over Number Field in a with defining polynomial x^2 + 5 with a = 2.236067977499790?*I
Check that non-torsion points are remembered when extending the base field (see Issue #16034):
sage: E = EllipticCurve([1, 0, 1, -1751, -31352]) sage: K.<d> = QuadraticField(5) sage: E.gens() [(52 : 111 : 1)] sage: EK = E.base_extend(K) sage: EK.gens() [(52 : 111 : 1)]
>>> from sage.all import * >>> E = EllipticCurve([Integer(1), Integer(0), Integer(1), -Integer(1751), -Integer(31352)]) >>> K = QuadraticField(Integer(5), names=('d',)); (d,) = K._first_ngens(1) >>> E.gens() [(52 : 111 : 1)] >>> EK = E.base_extend(K) >>> EK.gens() [(52 : 111 : 1)]
- cm_discriminant()[source]¶
Return the CM discriminant of the \(j\)-invariant of this curve, or 0.
OUTPUT:
An integer \(D\) which is either \(0\) if this curve \(E\) does not have Complex Multiplication) (CM), or an imaginary quadratic discriminant if \(j(E)\) is the \(j\)-invariant of the order with discriminant \(D\).
Note
If \(E\) has CM but the discriminant \(D\) is not a square in the base field \(K\) then the extra endomorphisms will not be defined over \(K\). See also
has_rational_cm()
.EXAMPLES:
sage: EllipticCurve(j=0).cm_discriminant() -3 sage: EllipticCurve(j=1).cm_discriminant() Traceback (most recent call last): ... ValueError: Elliptic Curve defined by y^2 + x*y = x^3 + 36*x + 3455 over Rational Field does not have CM sage: EllipticCurve(j=1728).cm_discriminant() -4 sage: EllipticCurve(j=8000).cm_discriminant() -8 sage: K.<a> = QuadraticField(5) sage: EllipticCurve(j=282880*a + 632000).cm_discriminant() -20 sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^3 - 2) sage: EllipticCurve(j=31710790944000*a^2 + 39953093016000*a + 50337742902000).cm_discriminant() -108
>>> from sage.all import * >>> EllipticCurve(j=Integer(0)).cm_discriminant() -3 >>> EllipticCurve(j=Integer(1)).cm_discriminant() Traceback (most recent call last): ... ValueError: Elliptic Curve defined by y^2 + x*y = x^3 + 36*x + 3455 over Rational Field does not have CM >>> EllipticCurve(j=Integer(1728)).cm_discriminant() -4 >>> EllipticCurve(j=Integer(8000)).cm_discriminant() -8 >>> K = QuadraticField(Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> EllipticCurve(j=Integer(282880)*a + Integer(632000)).cm_discriminant() -20 >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> EllipticCurve(j=Integer(31710790944000)*a**Integer(2) + Integer(39953093016000)*a + Integer(50337742902000)).cm_discriminant() -108
- conductor()[source]¶
Return the conductor of this elliptic curve as a fractional ideal of the base field.
OUTPUT: fractional ideal; the conductor of the curve
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: EllipticCurve([i, i - 1, i + 1, 24*i + 15, 14*i + 35]).conductor() Fractional ideal (21*i - 3) sage: K.<a> = NumberField(x^2 - x + 3) sage: EllipticCurve([1 + a, -1 + a, 1 + a, -11 + a, 5 - 9*a]).conductor() Fractional ideal (-6*a)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EllipticCurve([i, i - Integer(1), i + Integer(1), Integer(24)*i + Integer(15), Integer(14)*i + Integer(35)]).conductor() Fractional ideal (21*i - 3) >>> K = NumberField(x**Integer(2) - x + Integer(3), names=('a',)); (a,) = K._first_ngens(1) >>> EllipticCurve([Integer(1) + a, -Integer(1) + a, Integer(1) + a, -Integer(11) + a, Integer(5) - Integer(9)*a]).conductor() Fractional ideal (-6*a)
A not so well known curve with everywhere good reduction:
sage: K.<a> = NumberField(x^2 - 38) sage: E = EllipticCurve([0,0,0, 21796814856932765568243810*a - 134364590724198567128296995, 121774567239345229314269094644186997594*a - 750668847495706904791115375024037711300]) sage: E.conductor() Fractional ideal (1)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(38), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0), Integer(21796814856932765568243810)*a - Integer(134364590724198567128296995), Integer(121774567239345229314269094644186997594)*a - Integer(750668847495706904791115375024037711300)]) >>> E.conductor() Fractional ideal (1)
An example which used to fail (see Issue #5307):
sage: K.<w> = NumberField(x^2 + x + 6) sage: E = EllipticCurve([w, -1, 0, -w-6, 0]) sage: E.conductor() Fractional ideal (86304, w + 5898)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + x + Integer(6), names=('w',)); (w,) = K._first_ngens(1) >>> E = EllipticCurve([w, -Integer(1), Integer(0), -w-Integer(6), Integer(0)]) >>> E.conductor() Fractional ideal (86304, w + 5898)
An example raised in Issue #11346:
sage: K.<g> = NumberField(x^2 - x - 1) sage: E1 = EllipticCurve(K, [0, 0, 0, -1/48, -161/864]) sage: [(p.smallest_integer(), e) for p,e in E1.conductor().factor()] [(2, 4), (3, 1), (5, 1)]
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - x - Integer(1), names=('g',)); (g,) = K._first_ngens(1) >>> E1 = EllipticCurve(K, [Integer(0), Integer(0), Integer(0), -Integer(1)/Integer(48), -Integer(161)/Integer(864)]) >>> [(p.smallest_integer(), e) for p,e in E1.conductor().factor()] [(2, 4), (3, 1), (5, 1)]
- galois_representation()[source]¶
The compatible family of the Galois representation attached to this elliptic curve.
Given an elliptic curve \(E\) over a number field \(K\) and a rational prime number \(p\), the \(p^n\)-torsion \(E[p^n]\) points of \(E\) is a representation of the absolute Galois group of \(K\). As \(n\) varies we obtain the Tate module \(T_p E\) which is a a representation of \(G_K\) on a free \(\ZZ_p\)-module of rank \(2\). As \(p\) varies the representations are compatible.
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K = NumberField(x**2 + 1, 'a') sage: E = EllipticCurve('11a1').change_ring(K) sage: rho = E.galois_representation() sage: rho Compatible family of Galois representations associated to the Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 + (-10)*x + (-20) over Number Field in a with defining polynomial x^2 + 1 sage: rho.is_surjective(3) True sage: rho.is_surjective(5) # long time (4s on sage.math, 2014) False sage: rho.non_surjective() [5]
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) + Integer(1), 'a') >>> E = EllipticCurve('11a1').change_ring(K) >>> rho = E.galois_representation() >>> rho Compatible family of Galois representations associated to the Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 + (-10)*x + (-20) over Number Field in a with defining polynomial x^2 + 1 >>> rho.is_surjective(Integer(3)) True >>> rho.is_surjective(Integer(5)) # long time (4s on sage.math, 2014) False >>> rho.non_surjective() [5]
- gens(**kwds)[source]¶
Return some points of infinite order on this elliptic curve.
Contrary to what the name of this method suggests, the points it returns do not always generate a subgroup of full rank in the Mordell-Weil group, nor are they necessarily linearly independent. Moreover, the number of points can be smaller or larger than what one could expect after calling
rank()
orrank_bounds()
.Note
The optional parameters control the Simon two descent algorithm; see the documentation of
simon_two_descent()
for more details.INPUT:
verbose
– 0, 1, 2, or 3 (default: 0); the verbosity levellim1
– (default: 2) limit on trivial points on quarticslim3
– (default: 4) limit on points on ELS quarticslimtriv
– (default: 2) limit on trivial points on elliptic curvemaxprob
– (default: 20)limbigprime
– (default: 30) to distinguish between small and large prime numbers. Use probabilistic tests for large primes. If 0, do not use probabilistic tests.known_points
– (default:None
) list of known points on the curve
OUTPUT: a set of points of infinite order given by the Simon two-descent
Note
For non-quadratic number fields, this code does return, but it takes a long time.
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 + 23, 'a') sage: E = EllipticCurve(K,[0,0,0,101,0]) sage: E.gens() [(23831509/8669448*a - 2867471/8669448 : 76507317707/18049790736*a - 424166479633/18049790736 : 1), (-2031032029/969232392*a + 58813561/969232392 : -15575984630401/21336681877488*a + 451041199309/21336681877488 : 1), (-186948623/4656964 : 549438861195/10049728312*a : 1)]
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) + Integer(23), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K,[Integer(0),Integer(0),Integer(0),Integer(101),Integer(0)]) >>> E.gens() [(23831509/8669448*a - 2867471/8669448 : 76507317707/18049790736*a - 424166479633/18049790736 : 1), (-2031032029/969232392*a + 58813561/969232392 : -15575984630401/21336681877488*a + 451041199309/21336681877488 : 1), (-186948623/4656964 : 549438861195/10049728312*a : 1)]
It can happen that no points are found if the height bounds used in the search are too small (see Issue #10745):
sage: K.<t> = NumberField(x^4 + x^2 - 7) sage: E = EllipticCurve(K, [1, 0, 5*t^2 + 16, 0, 0]) sage: E.gens(lim1=1, lim3=1) [] sage: E.rank() 1 sage: gg=E.gens(lim3=13); gg # long time (about 4s) [(... : 1)]
>>> from sage.all import * >>> K = NumberField(x**Integer(4) + x**Integer(2) - Integer(7), names=('t',)); (t,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(1), Integer(0), Integer(5)*t**Integer(2) + Integer(16), Integer(0), Integer(0)]) >>> E.gens(lim1=Integer(1), lim3=Integer(1)) [] >>> E.rank() 1 >>> gg=E.gens(lim3=Integer(13)); gg # long time (about 4s) [(... : 1)]
Check that the the point found has infinite order, and that it is on the curve:
sage: P=gg[0]; P.order() # long time +Infinity sage: E.defining_polynomial()(*P) # long time 0
>>> from sage.all import * >>> P=gg[Integer(0)]; P.order() # long time +Infinity >>> E.defining_polynomial()(*P) # long time 0
Here is a curve of rank 2:
sage: K.<t> = NumberField(x^2 - 17) sage: E = EllipticCurve(K, [-4, 0]) sage: E.gens() [(-1/2*t + 1/2 : -1/2*t + 1/2 : 1), (-t + 3 : -2*t + 10 : 1)] sage: E.rank() 2
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(17), names=('t',)); (t,) = K._first_ngens(1) >>> E = EllipticCurve(K, [-Integer(4), Integer(0)]) >>> E.gens() [(-1/2*t + 1/2 : -1/2*t + 1/2 : 1), (-t + 3 : -2*t + 10 : 1)] >>> E.rank() 2
Test that points of finite order are not included (see Issue #13593):
sage: E = EllipticCurve("17a3") sage: K.<t> = NumberField(x^2 + 3) sage: EK = E.base_extend(K) sage: EK.rank() 0 sage: EK.gens() []
>>> from sage.all import * >>> E = EllipticCurve("17a3") >>> K = NumberField(x**Integer(2) + Integer(3), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.rank() 0 >>> EK.gens() []
IMPLEMENTATION:
For curves over quadratic fields which are base-changes from \(\QQ\), we delegate the work to
gens_quadratic()
where methods over \(\QQ\) suffice. Otherwise, we use Denis Simon’s PARI/GP scripts from http://www.math.unicaen.fr/~simon/.
- gens_quadratic(**kwds)[source]¶
Return generators for the Mordell-Weil group modulo torsion, for a curve which is a base change from \(\QQ\) to a quadratic field.
EXAMPLES:
sage: E = EllipticCurve([1,2,3,40,50]) sage: E.conductor() 2123582 sage: E.gens() [(5 : 17 : 1)] sage: K.<i> = QuadraticField(-1) sage: EK = E.change_ring(K) sage: EK.gens_quadratic() [(5 : 17 : 1), (-13 : 48*i + 5 : 1)] sage: E.change_ring(QuadraticField(3, 'a')).gens_quadratic() [(5 : 17 : 1), (-1 : 2*a - 1 : 1), (11/4 : 33/4*a - 23/8 : 1)] sage: K.<a> = QuadraticField(-7) sage: E = EllipticCurve([0,0,0,197,0]) sage: E.conductor() 2483776 sage: E.gens() [(47995604297578081/7389879786648100 : -25038161802544048018837479/635266655830129794121000 : 1)] sage: K.<a> = QuadraticField(7) sage: E.change_ring(K).gens_quadratic() [(-1209642055/59583566*a + 1639995844/29791783 : -377240626321899/1720892553212*a + 138577803462855/245841793316 : 1), (1/28 : 393/392*a : 1), (-61*a + 162 : 1098*a - 2916 : 1)] sage: E = EllipticCurve([1, a]) sage: E.gens_quadratic() Traceback (most recent call last): ... ValueError: gens_quadratic() requires the elliptic curve to be a base change from Q
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(2),Integer(3),Integer(40),Integer(50)]) >>> E.conductor() 2123582 >>> E.gens() [(5 : 17 : 1)] >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EK = E.change_ring(K) >>> EK.gens_quadratic() [(5 : 17 : 1), (-13 : 48*i + 5 : 1)] >>> E.change_ring(QuadraticField(Integer(3), 'a')).gens_quadratic() [(5 : 17 : 1), (-1 : 2*a - 1 : 1), (11/4 : 33/4*a - 23/8 : 1)] >>> K = QuadraticField(-Integer(7), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(197),Integer(0)]) >>> E.conductor() 2483776 >>> E.gens() [(47995604297578081/7389879786648100 : -25038161802544048018837479/635266655830129794121000 : 1)] >>> K = QuadraticField(Integer(7), names=('a',)); (a,) = K._first_ngens(1) >>> E.change_ring(K).gens_quadratic() [(-1209642055/59583566*a + 1639995844/29791783 : -377240626321899/1720892553212*a + 138577803462855/245841793316 : 1), (1/28 : 393/392*a : 1), (-61*a + 162 : 1098*a - 2916 : 1)] >>> E = EllipticCurve([Integer(1), a]) >>> E.gens_quadratic() Traceback (most recent call last): ... ValueError: gens_quadratic() requires the elliptic curve to be a base change from Q
- global_integral_model()[source]¶
Return a model of
self
which is integral at all primes.EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([i/5,i/5,i/5,i/5,i/5]) sage: P1, P2 = K.primes_above(5) sage: E.global_integral_model() Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5)]) >>> P1, P2 = K.primes_above(Integer(5)) >>> E.global_integral_model() Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
sage: K.<a> = NumberField(x^2 - 38) sage: E = EllipticCurve([a,1/2]) sage: E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 1444*a*x + 27436 over Number Field in a with defining polynomial x^2 - 38
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(38), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([a,Integer(1)/Integer(2)]) >>> E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 1444*a*x + 27436 over Number Field in a with defining polynomial x^2 - 38
sage: K.<s> = NumberField(x^2 - 5) sage: w = (1+s)/2 sage: E = EllipticCurve(K, [2,w]) sage: E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 2*x + (1/2*s+1/2) over Number Field in s with defining polynomial x^2 - 5
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('s',)); (s,) = K._first_ngens(1) >>> w = (Integer(1)+s)/Integer(2) >>> E = EllipticCurve(K, [Integer(2),w]) >>> E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 2*x + (1/2*s+1/2) over Number Field in s with defining polynomial x^2 - 5
sage: K.<v> = NumberField(x^2 + 161*x - 150) sage: E = EllipticCurve([25105/216*v - 3839/36, 634768555/7776*v - 98002625/1296, 634768555/7776*v - 98002625/1296, 0, 0]) sage: M = E.global_integral_model(); M # choice varies, not tested Elliptic Curve defined by y^2 + (2094779518028859*v-1940492905300351)*x*y + (477997268472544193101178234454165304071127500*v-442791377441346852919930773849502871958097500)*y = x^3 + (26519784690047674853185542622500*v-24566525306469707225840460652500)*x^2 over Number Field in v with defining polynomial x^2 + 161*x - 150
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(161)*x - Integer(150), names=('v',)); (v,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(25105)/Integer(216)*v - Integer(3839)/Integer(36), Integer(634768555)/Integer(7776)*v - Integer(98002625)/Integer(1296), Integer(634768555)/Integer(7776)*v - Integer(98002625)/Integer(1296), Integer(0), Integer(0)]) >>> M = E.global_integral_model(); M # choice varies, not tested Elliptic Curve defined by y^2 + (2094779518028859*v-1940492905300351)*x*y + (477997268472544193101178234454165304071127500*v-442791377441346852919930773849502871958097500)*y = x^3 + (26519784690047674853185542622500*v-24566525306469707225840460652500)*x^2 over Number Field in v with defining polynomial x^2 + 161*x - 150
sage: R.<t> = QQ[] sage: K.<g> = NumberField(t^4 - t^3 - 3*t^2 - t + 1) sage: E = EllipticCurve([ -43/625*g^3 + 14/625*g^2 - 4/625*g + 706/625, -4862/78125*g^3 - 4074/78125*g^2 - 711/78125*g + 10304/78125, -4862/78125*g^3 - 4074/78125*g^2 - 711/78125*g + 10304/78125, 0,0]) sage: E.global_integral_model() Elliptic Curve defined by y^2 + (15*g^3-48*g-42)*x*y + (-111510*g^3-162162*g^2-44145*g+37638)*y = x^3 + (-954*g^3-1134*g^2+81*g+576)*x^2 over Number Field in g with defining polynomial t^4 - t^3 - 3*t^2 - t + 1
>>> from sage.all import * >>> R = QQ['t']; (t,) = R._first_ngens(1) >>> K = NumberField(t**Integer(4) - t**Integer(3) - Integer(3)*t**Integer(2) - t + Integer(1), names=('g',)); (g,) = K._first_ngens(1) >>> E = EllipticCurve([ -Integer(43)/Integer(625)*g**Integer(3) + Integer(14)/Integer(625)*g**Integer(2) - Integer(4)/Integer(625)*g + Integer(706)/Integer(625), -Integer(4862)/Integer(78125)*g**Integer(3) - Integer(4074)/Integer(78125)*g**Integer(2) - Integer(711)/Integer(78125)*g + Integer(10304)/Integer(78125), -Integer(4862)/Integer(78125)*g**Integer(3) - Integer(4074)/Integer(78125)*g**Integer(2) - Integer(711)/Integer(78125)*g + Integer(10304)/Integer(78125), Integer(0),Integer(0)]) >>> E.global_integral_model() Elliptic Curve defined by y^2 + (15*g^3-48*g-42)*x*y + (-111510*g^3-162162*g^2-44145*g+37638)*y = x^3 + (-954*g^3-1134*g^2+81*g+576)*x^2 over Number Field in g with defining polynomial t^4 - t^3 - 3*t^2 - t + 1
- global_minimal_model(proof=None, semi_global=False)[source]¶
Return a model of
self
that is integral, and minimal.Note
Over fields of class number greater than 1, a global minimal model may not exist. If it does not, set the parameter
semi_global
toTrue
to obtain a model minimal at all but one prime.INPUT:
proof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.semi_global
– boolean (default:False
); if there is no global minimal mode, return a semi-global minimal model (minimal at all but one prime) instead, if True; raise an error if False. No effect if a global minimal model exists.
OUTPUT:
A global integral and minimal model, or an integral model minimal at all but one prime of there is no global minimal model and the flag
semi_global
is True.EXAMPLES:
sage: K.<a> = NumberField(x^2 - 38) sage: E = EllipticCurve([0,0,0, 21796814856932765568243810*a - 134364590724198567128296995, 121774567239345229314269094644186997594*a - 750668847495706904791115375024037711300]) sage: E2 = E.global_minimal_model() sage: E2 Elliptic Curve defined by y^2 + a*x*y + (a+1)*y = x^3 + (a+1)*x^2 + (4*a+15)*x + (4*a+21) over Number Field in a with defining polynomial x^2 - 38 sage: E2.local_data() []
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(38), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0), Integer(21796814856932765568243810)*a - Integer(134364590724198567128296995), Integer(121774567239345229314269094644186997594)*a - Integer(750668847495706904791115375024037711300)]) >>> E2 = E.global_minimal_model() >>> E2 Elliptic Curve defined by y^2 + a*x*y + (a+1)*y = x^3 + (a+1)*x^2 + (4*a+15)*x + (4*a+21) over Number Field in a with defining polynomial x^2 - 38 >>> E2.local_data() []
See Issue #11347:
sage: K.<g> = NumberField(x^2 - x - 1) sage: E = EllipticCurve(K, [0, 0, 0, -1/48, 161/864]) sage: E2 = E.integral_model().global_minimal_model(); E2 Elliptic Curve defined by y^2 + x*y + y = x^3 + x^2 over Number Field in g with defining polynomial x^2 - x - 1 sage: [(p.norm(), e) for p, e in E2.conductor().factor()] [(9, 1), (5, 1)] sage: [(p.norm(), e) for p, e in E2.discriminant().factor()] [(-5, 2), (9, 1)]
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - x - Integer(1), names=('g',)); (g,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0), Integer(0), Integer(0), -Integer(1)/Integer(48), Integer(161)/Integer(864)]) >>> E2 = E.integral_model().global_minimal_model(); E2 Elliptic Curve defined by y^2 + x*y + y = x^3 + x^2 over Number Field in g with defining polynomial x^2 - x - 1 >>> [(p.norm(), e) for p, e in E2.conductor().factor()] [(9, 1), (5, 1)] >>> [(p.norm(), e) for p, e in E2.discriminant().factor()] [(-5, 2), (9, 1)]
See Issue #14472, this used not to work over a relative extension:
sage: K1.<w> = NumberField(x^2 + x + 1) sage: m = polygen(K1) sage: K2.<v> = K1.extension(m^2 - w + 1) sage: E = EllipticCurve([0*v, -432]) sage: E.global_minimal_model() Elliptic Curve defined by y^2 + y = x^3 over Number Field in v with defining polynomial x^2 - w + 1 over its base field
>>> from sage.all import * >>> K1 = NumberField(x**Integer(2) + x + Integer(1), names=('w',)); (w,) = K1._first_ngens(1) >>> m = polygen(K1) >>> K2 = K1.extension(m**Integer(2) - w + Integer(1), names=('v',)); (v,) = K2._first_ngens(1) >>> E = EllipticCurve([Integer(0)*v, -Integer(432)]) >>> E.global_minimal_model() Elliptic Curve defined by y^2 + y = x^3 over Number Field in v with defining polynomial x^2 - w + 1 over its base field
See Issue #18662: for fields of class number greater than 1, even when global minimal models did exist, their computation was not implemented. Now it is:
sage: K.<a> = NumberField(x^2 - 10) sage: K.class_number() 2 sage: E = EllipticCurve([0, 0, 0, -186408*a - 589491, 78055704*a + 246833838]) sage: E.discriminant().norm() 16375845905239507992576 sage: E.discriminant().norm().factor() 2^31 * 3^27 sage: E.has_global_minimal_model() True sage: Emin = E.global_minimal_model(); Emin Elliptic Curve defined by y^2 + (a+1)*x*y + (a+1)*y = x^3 + (-a)*x^2 + (a-12)*x + (-2*a+2) over Number Field in a with defining polynomial x^2 - 10 sage: Emin.discriminant().norm() 3456 sage: Emin.discriminant().norm().factor() 2^7 * 3^3
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> K.class_number() 2 >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(186408)*a - Integer(589491), Integer(78055704)*a + Integer(246833838)]) >>> E.discriminant().norm() 16375845905239507992576 >>> E.discriminant().norm().factor() 2^31 * 3^27 >>> E.has_global_minimal_model() True >>> Emin = E.global_minimal_model(); Emin Elliptic Curve defined by y^2 + (a+1)*x*y + (a+1)*y = x^3 + (-a)*x^2 + (a-12)*x + (-2*a+2) over Number Field in a with defining polynomial x^2 - 10 >>> Emin.discriminant().norm() 3456 >>> Emin.discriminant().norm().factor() 2^7 * 3^3
If there is no global minimal model, this method will raise an error unless you set the parameter
semi_global
toTrue
:sage: K.<a> = NumberField(x^2 - 10) sage: K.class_number() 2 sage: E = EllipticCurve([a, a, 0, 3*a+8, 4*a+3]) sage: E.has_global_minimal_model() False sage: E.global_minimal_model() Traceback (most recent call last): ... ValueError: Elliptic Curve defined by y^2 + a*x*y = x^3 + a*x^2 + (3*a+8)*x + (4*a+3) over Number Field in a with defining polynomial x^2 - 10 has no global minimal model! For a semi-global minimal model use semi_global=True sage: E.global_minimal_model(semi_global=True) Elliptic Curve defined by y^2 + a*x*y = x^3 + a*x^2 + (3*a+8)*x + (4*a+3) over Number Field in a with defining polynomial x^2 - 10
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> K.class_number() 2 >>> E = EllipticCurve([a, a, Integer(0), Integer(3)*a+Integer(8), Integer(4)*a+Integer(3)]) >>> E.has_global_minimal_model() False >>> E.global_minimal_model() Traceback (most recent call last): ... ValueError: Elliptic Curve defined by y^2 + a*x*y = x^3 + a*x^2 + (3*a+8)*x + (4*a+3) over Number Field in a with defining polynomial x^2 - 10 has no global minimal model! For a semi-global minimal model use semi_global=True >>> E.global_minimal_model(semi_global=True) Elliptic Curve defined by y^2 + a*x*y = x^3 + a*x^2 + (3*a+8)*x + (4*a+3) over Number Field in a with defining polynomial x^2 - 10
An example of a curve with everywhere good reduction but which has no model with unit discriminant:
sage: K.<a> = NumberField(x^2 - x - 16) sage: K.class_number() 2 sage: E = EllipticCurve([0,0,0,-15221331*a - 53748576, -79617688290*a - 281140318368]) sage: Emin = E.global_minimal_model(semi_global=True) sage: Emin.ainvs() (a, a - 1, a, 605*a - 2728, 15887*a - 71972) sage: Emin.discriminant() -17*a - 16 sage: Emin.discriminant().norm() -4096 sage: Emin.minimal_discriminant_ideal() Fractional ideal (1) sage: E.conductor() Fractional ideal (1)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - x - Integer(16), names=('a',)); (a,) = K._first_ngens(1) >>> K.class_number() 2 >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),-Integer(15221331)*a - Integer(53748576), -Integer(79617688290)*a - Integer(281140318368)]) >>> Emin = E.global_minimal_model(semi_global=True) >>> Emin.ainvs() (a, a - 1, a, 605*a - 2728, 15887*a - 71972) >>> Emin.discriminant() -17*a - 16 >>> Emin.discriminant().norm() -4096 >>> Emin.minimal_discriminant_ideal() Fractional ideal (1) >>> E.conductor() Fractional ideal (1)
- global_minimality_class()[source]¶
Return the obstruction to this curve having a global minimal model.
OUTPUT:
An ideal class of the base number field, which is trivial if and only if the elliptic curve has a global minimal model, and which can be used to find global and semi-global minimal models.
EXAMPLES:
A curve defined over a field of class number 2 with no global minimal model was a nontrivial minimality class:
sage: K.<a> = NumberField(x^2 - 10) sage: K.class_number() 2 sage: E = EllipticCurve([0, 0, 0, -22500, 750000*a]) sage: E.global_minimality_class() Fractional ideal class (10, 5*a) sage: E.global_minimality_class().order() 2
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> K.class_number() 2 >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(22500), Integer(750000)*a]) >>> E.global_minimality_class() Fractional ideal class (10, 5*a) >>> E.global_minimality_class().order() 2
Over the same field, a curve defined by a non-minimal model has trivial class, showing that a global minimal model does exist:
sage: K.<a> = NumberField(x^2 - 10) sage: E = EllipticCurve([0, 0, 0, 4536*a+14148, -163728*a-474336]) sage: E.is_global_minimal_model() False sage: E.global_minimality_class() Trivial principal fractional ideal class
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), Integer(4536)*a+Integer(14148), -Integer(163728)*a-Integer(474336)]) >>> E.is_global_minimal_model() False >>> E.global_minimality_class() Trivial principal fractional ideal class
Over a field of class number 1 the result is always the trivial class:
sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve([0, 0, 0, K(16), K(64)]) sage: E.global_minimality_class() Trivial principal fractional ideal class sage: E = EllipticCurve([0, 0, 0, 16, 64]) sage: E.base_field() Rational Field sage: E.global_minimality_class() 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), K(Integer(16)), K(Integer(64))]) >>> E.global_minimality_class() Trivial principal fractional ideal class >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), Integer(16), Integer(64)]) >>> E.base_field() Rational Field >>> E.global_minimality_class() 1
- has_additive_reduction(P)[source]¶
Return
True
if this elliptic curve has (bad) additive reduction at the prime \(P\).INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has additive reduction at \(P\), elseFalse
.EXAMPLES:
sage: E = EllipticCurve('27a1') sage: [(p, E.has_additive_reduction(p)) for p in prime_range(15)] [(2, False), (3, True), (5, False), (7, False), (11, False), (13, False)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_additive_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), True)]
>>> from sage.all import * >>> E = EllipticCurve('27a1') >>> [(p, E.has_additive_reduction(p)) for p in prime_range(Integer(15))] [(2, False), (3, True), (5, False), (7, False), (11, False), (13, False)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_additive_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), True)]
- has_bad_reduction(P)[source]¶
Return
True
if this elliptic curve has bad reduction at the prime \(P\).INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has bad reduction at \(P\), elseFalse
.Note
This requires determining a local integral minimal model; we do not just check that the discriminant of the current model has valuation zero.
EXAMPLES:
sage: E = EllipticCurve('14a1') sage: [(p, E.has_bad_reduction(p)) for p in prime_range(15)] [(2, True), (3, False), (5, False), (7, True), (11, False), (13, False)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_bad_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), True)]
>>> from sage.all import * >>> E = EllipticCurve('14a1') >>> [(p, E.has_bad_reduction(p)) for p in prime_range(Integer(15))] [(2, True), (3, False), (5, False), (7, True), (11, False), (13, False)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_bad_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), True)]
- has_cm()[source]¶
Return whether or not this curve has a CM \(j\)-invariant.
OUTPUT:
True
if this curve has CM over the algebraic closure of the base field, otherwiseFalse
. See alsocm_discriminant()
andhas_rational_cm()
.Note
Even if \(E\) has CM in this sense (that its \(j\)-invariant is a CM \(j\)-invariant), if the associated negative discriminant \(D\) is not a square in the base field \(K\), the extra endomorphisms will not be defined over \(K\). See also the method
has_rational_cm()
which tests whether \(E\) has extra endomorphisms defined over \(K\) or a given extension of \(K\).EXAMPLES:
sage: EllipticCurve(j=0).has_cm() True sage: EllipticCurve(j=1).has_cm() False sage: EllipticCurve(j=1728).has_cm() True sage: EllipticCurve(j=8000).has_cm() True sage: K.<a> = QuadraticField(5) sage: EllipticCurve(j=282880*a + 632000).has_cm() True sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^3 - 2) sage: EllipticCurve(j=31710790944000*a^2 + 39953093016000*a + 50337742902000).has_cm() True
>>> from sage.all import * >>> EllipticCurve(j=Integer(0)).has_cm() True >>> EllipticCurve(j=Integer(1)).has_cm() False >>> EllipticCurve(j=Integer(1728)).has_cm() True >>> EllipticCurve(j=Integer(8000)).has_cm() True >>> K = QuadraticField(Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> EllipticCurve(j=Integer(282880)*a + Integer(632000)).has_cm() True >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> EllipticCurve(j=Integer(31710790944000)*a**Integer(2) + Integer(39953093016000)*a + Integer(50337742902000)).has_cm() True
- has_global_minimal_model()[source]¶
Return whether this elliptic curve has a global minimal model.
OUTPUT:
boolean;
True
iff a global minimal model exists, i.e. an integral model which is minimal at every prime.EXAMPLES:
sage: K.<a> = NumberField(x^2 - 10) sage: E = EllipticCurve([0, 0, 0, 4536*a+14148, -163728*a-474336]) sage: E.is_global_minimal_model() False sage: E.has_global_minimal_model() True
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), Integer(4536)*a+Integer(14148), -Integer(163728)*a-Integer(474336)]) >>> E.is_global_minimal_model() False >>> E.has_global_minimal_model() True
- has_good_reduction(P)[source]¶
Return
True
if this elliptic curve has good reduction at the prime \(P\).INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has good reduction at \(P\), elseFalse
.Note
This requires determining a local integral minimal model; we do not just check that the discriminant of the current model has valuation zero.
EXAMPLES:
sage: E = EllipticCurve('14a1') sage: [(p, E.has_good_reduction(p)) for p in prime_range(15)] [(2, False), (3, True), (5, True), (7, False), (11, True), (13, True)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_good_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), True), (Fractional ideal (2*a + 1), False)]
>>> from sage.all import * >>> E = EllipticCurve('14a1') >>> [(p, E.has_good_reduction(p)) for p in prime_range(Integer(15))] [(2, False), (3, True), (5, True), (7, False), (11, True), (13, True)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_good_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), True), (Fractional ideal (2*a + 1), False)]
- has_multiplicative_reduction(P)[source]¶
Return
True
if this elliptic curve has (bad) multiplicative reduction at the prime \(P\).Note
See also
has_split_multiplicative_reduction()
andhas_nonsplit_multiplicative_reduction()
.INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has multiplicative reduction at \(P\), elseFalse
.EXAMPLES:
sage: E = EllipticCurve('14a1') sage: [(p, E.has_multiplicative_reduction(p)) for p in prime_range(15)] [(2, True), (3, False), (5, False), (7, True), (11, False), (13, False)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
>>> from sage.all import * >>> E = EllipticCurve('14a1') >>> [(p, E.has_multiplicative_reduction(p)) for p in prime_range(Integer(15))] [(2, True), (3, False), (5, False), (7, True), (11, False), (13, False)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
- has_nonsplit_multiplicative_reduction(P)[source]¶
Return
True
if this elliptic curve has (bad) non-split multiplicative reduction at the prime \(P\).INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has non-split multiplicative reduction at \(P\), elseFalse
.EXAMPLES:
sage: E = EllipticCurve('14a1') sage: [(p, E.has_nonsplit_multiplicative_reduction(p)) for p in prime_range(15)] [(2, True), (3, False), (5, False), (7, False), (11, False), (13, False)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_nonsplit_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
>>> from sage.all import * >>> E = EllipticCurve('14a1') >>> [(p, E.has_nonsplit_multiplicative_reduction(p)) for p in prime_range(Integer(15))] [(2, True), (3, False), (5, False), (7, False), (11, False), (13, False)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_nonsplit_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
- has_rational_cm(field=None)[source]¶
Return whether or not this curve has CM defined over its base field or a given extension.
INPUT:
field
– a field, which should be an extension of the base field of the curve. Iffield
isNone
(the default), it is taken to be the base field of the curve.
OUTPUT:
True
if the ring of endomorphisms of this curve over the given field is larger than \(\ZZ\); otherwiseFalse
. See alsocm_discriminant()
andhas_cm()
.Note
If \(E\) has CM but the discriminant \(D\) is not a square in the given field \(K\) then the extra endomorphisms will not be defined over \(K\), and this function will return
False
. See alsohas_cm()
. To obtain the CM discriminant, usecm_discriminant()
.EXAMPLES:
sage: E = EllipticCurve(j=0) sage: E.has_cm() True sage: E.has_rational_cm() False sage: D = E.cm_discriminant(); D -3 sage: E.has_rational_cm(QuadraticField(D)) True sage: E = EllipticCurve(j=1728) sage: E.has_cm() True sage: E.has_rational_cm() False sage: D = E.cm_discriminant(); D -4 sage: E.has_rational_cm(QuadraticField(D)) True
>>> from sage.all import * >>> E = EllipticCurve(j=Integer(0)) >>> E.has_cm() True >>> E.has_rational_cm() False >>> D = E.cm_discriminant(); D -3 >>> E.has_rational_cm(QuadraticField(D)) True >>> E = EllipticCurve(j=Integer(1728)) >>> E.has_cm() True >>> E.has_rational_cm() False >>> D = E.cm_discriminant(); D -4 >>> E.has_rational_cm(QuadraticField(D)) True
Higher degree examples:
sage: K.<a> = QuadraticField(5) sage: E = EllipticCurve(j=282880*a + 632000) sage: E.has_cm() True sage: E.has_rational_cm() False sage: E.cm_discriminant() -20 sage: E.has_rational_cm(K.extension(x^2 + 5, 'b')) True
>>> from sage.all import * >>> K = QuadraticField(Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(j=Integer(282880)*a + Integer(632000)) >>> E.has_cm() True >>> E.has_rational_cm() False >>> E.cm_discriminant() -20 >>> E.has_rational_cm(K.extension(x**Integer(2) + Integer(5), 'b')) True
An error is raised if a field is given which is not an extension of the base field:
sage: E.has_rational_cm(QuadraticField(-20)) Traceback (most recent call last): ... ValueError: Error in has_rational_cm: Number Field in a with defining polynomial x^2 + 20 with a = 4.472135954999579?*I is not an extension field of Number Field in a with defining polynomial x^2 - 5 with a = 2.236067977499790? sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^3 - 2) sage: E = EllipticCurve(j=31710790944000*a^2 + 39953093016000*a + 50337742902000) sage: E.has_cm() True sage: E.has_rational_cm() False sage: D = E.cm_discriminant(); D -108 sage: E.has_rational_cm(K.extension(x^2 + 108,'b')) True
>>> from sage.all import * >>> E.has_rational_cm(QuadraticField(-Integer(20))) Traceback (most recent call last): ... ValueError: Error in has_rational_cm: Number Field in a with defining polynomial x^2 + 20 with a = 4.472135954999579?*I is not an extension field of Number Field in a with defining polynomial x^2 - 5 with a = 2.236067977499790? >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(j=Integer(31710790944000)*a**Integer(2) + Integer(39953093016000)*a + Integer(50337742902000)) >>> E.has_cm() True >>> E.has_rational_cm() False >>> D = E.cm_discriminant(); D -108 >>> E.has_rational_cm(K.extension(x**Integer(2) + Integer(108),'b')) True
- has_split_multiplicative_reduction(P)[source]¶
Return
True
if this elliptic curve has (bad) split multiplicative reduction at the prime \(P\).INPUT:
P
– a prime ideal of the base field ofself
, or a field element generating such an ideal
OUTPUT:
boolean;
True
if the curve has split multiplicative reduction at \(P\), elseFalse
.EXAMPLES:
sage: E = EllipticCurve('14a1') sage: [(p, E.has_split_multiplicative_reduction(p)) for p in prime_range(15)] [(2, False), (3, False), (5, False), (7, True), (11, False), (13, False)] sage: K.<a> = NumberField(x^3 - 2) sage: P17a, P17b = [P for P,e in K.factor(17)] sage: E = EllipticCurve([0,0,0,0,2*a+1]) sage: [(p, E.has_split_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
>>> from sage.all import * >>> E = EllipticCurve('14a1') >>> [(p, E.has_split_multiplicative_reduction(p)) for p in prime_range(Integer(15))] [(2, False), (3, False), (5, False), (7, True), (11, False), (13, False)] >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> P17a, P17b = [P for P,e in K.factor(Integer(17))] >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(0),Integer(2)*a+Integer(1)]) >>> [(p, E.has_split_multiplicative_reduction(p)) for p in [P17a,P17b]] [(Fractional ideal (4*a^2 - 2*a + 1), False), (Fractional ideal (2*a + 1), False)]
- height_function()[source]¶
Return the canonical height function attached to
self
.EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve(K, '11a3') sage: E.height_function() EllipticCurveCanonicalHeight object associated to Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 over Number Field in a with defining polynomial x^2 - 5
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, '11a3') >>> E.height_function() EllipticCurveCanonicalHeight object associated to Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 over Number Field in a with defining polynomial x^2 - 5
- height_pairing_matrix(points=None, precision=None, normalised=True)[source]¶
Return the height pairing matrix of the given points.
INPUT:
points
– list orNone
(default); list of points on this curve, orNone
, in which caseself.gens()
will be usedprecision
– integer orNone
(default); number of bits of precision of result, orNone
, for default RealField precisionnormalised
– boolean (default:True
); ifTrue
, use normalised heights which are independent of base change. Otherwise use the non-normalised Néron-Tate height, as required for the regulator in the BSD conjecture.
EXAMPLES:
sage: E = EllipticCurve([0, 0, 1, -1, 0]) sage: E.height_pairing_matrix() [0.0511114082399688]
>>> from sage.all import * >>> E = EllipticCurve([Integer(0), Integer(0), Integer(1), -Integer(1), Integer(0)]) >>> E.height_pairing_matrix() [0.0511114082399688]
For rank 0 curves, the result is a valid 0x0 matrix:
sage: EllipticCurve('11a').height_pairing_matrix() [] sage: E = EllipticCurve('5077a1') sage: E.height_pairing_matrix([E.lift_x(x) for x in [-2,-7/4,1]], precision=100) [ 1.3685725053539301120518194471 -1.3095767070865761992624519454 -0.63486715783715592064475542573] [ -1.3095767070865761992624519454 2.7173593928122930896610589220 1.0998184305667292139777571432] [-0.63486715783715592064475542573 1.0998184305667292139777571432 0.66820516565192793503314205089] sage: E = EllipticCurve('389a1') sage: E = EllipticCurve('389a1') sage: P, Q = E.point([-1,1,1]), E.point([0,-1,1]) sage: E.height_pairing_matrix([P,Q]) [0.686667083305587 0.268478098806726] [0.268478098806726 0.327000773651605]
>>> from sage.all import * >>> EllipticCurve('11a').height_pairing_matrix() [] >>> E = EllipticCurve('5077a1') >>> E.height_pairing_matrix([E.lift_x(x) for x in [-Integer(2),-Integer(7)/Integer(4),Integer(1)]], precision=Integer(100)) [ 1.3685725053539301120518194471 -1.3095767070865761992624519454 -0.63486715783715592064475542573] [ -1.3095767070865761992624519454 2.7173593928122930896610589220 1.0998184305667292139777571432] [-0.63486715783715592064475542573 1.0998184305667292139777571432 0.66820516565192793503314205089] >>> E = EllipticCurve('389a1') >>> E = EllipticCurve('389a1') >>> P, Q = E.point([-Integer(1),Integer(1),Integer(1)]), E.point([Integer(0),-Integer(1),Integer(1)]) >>> E.height_pairing_matrix([P,Q]) [0.686667083305587 0.268478098806726] [0.268478098806726 0.327000773651605]
Over a number field:
sage: x = polygen(QQ) sage: K.<t> = NumberField(x^2 + 47) sage: EK = E.base_extend(K) sage: EK.height_pairing_matrix([EK(P),EK(Q)]) [0.686667083305587 0.268478098806726] [0.268478098806726 0.327000773651605]
>>> from sage.all import * >>> x = polygen(QQ) >>> K = NumberField(x**Integer(2) + Integer(47), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.height_pairing_matrix([EK(P),EK(Q)]) [0.686667083305587 0.268478098806726] [0.268478098806726 0.327000773651605]
sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve([0,0,0,i,i]) sage: P = E(-9+4*i, -18-25*i) sage: Q = E(i,-i) sage: E.height_pairing_matrix([P,Q]) [ 2.16941934493768 -0.870059380421505] [-0.870059380421505 0.424585837470709] sage: E.regulator_of_points([P,Q]) 0.164101403936070
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),i,i]) >>> P = E(-Integer(9)+Integer(4)*i, -Integer(18)-Integer(25)*i) >>> Q = E(i,-i) >>> E.height_pairing_matrix([P,Q]) [ 2.16941934493768 -0.870059380421505] [-0.870059380421505 0.424585837470709] >>> E.regulator_of_points([P,Q]) 0.164101403936070
When the parameter
normalised
is set toFalse
, each height is multiplied by the degree \(d\) of the base field, and the regulator of \(r\) points is multiplied by \(d^r\):sage: E.height_pairing_matrix([P,Q], normalised=False) [ 4.33883868987537 -1.74011876084301] [-1.74011876084301 0.849171674941418] sage: E.regulator_of_points([P,Q], normalised=False) 0.656405615744281
>>> from sage.all import * >>> E.height_pairing_matrix([P,Q], normalised=False) [ 4.33883868987537 -1.74011876084301] [-1.74011876084301 0.849171674941418] >>> E.regulator_of_points([P,Q], normalised=False) 0.656405615744281
- integral_model()[source]¶
Return a model of
self
which is integral at all primes.EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([i/5,i/5,i/5,i/5,i/5]) sage: P1, P2 = K.primes_above(5) sage: E.global_integral_model() Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5)]) >>> P1, P2 = K.primes_above(Integer(5)) >>> E.global_integral_model() Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
sage: K.<a> = NumberField(x^2 - 38) sage: E = EllipticCurve([a,1/2]) sage: E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 1444*a*x + 27436 over Number Field in a with defining polynomial x^2 - 38
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(38), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([a,Integer(1)/Integer(2)]) >>> E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 1444*a*x + 27436 over Number Field in a with defining polynomial x^2 - 38
sage: K.<s> = NumberField(x^2 - 5) sage: w = (1+s)/2 sage: E = EllipticCurve(K, [2,w]) sage: E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 2*x + (1/2*s+1/2) over Number Field in s with defining polynomial x^2 - 5
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('s',)); (s,) = K._first_ngens(1) >>> w = (Integer(1)+s)/Integer(2) >>> E = EllipticCurve(K, [Integer(2),w]) >>> E.global_integral_model() Elliptic Curve defined by y^2 = x^3 + 2*x + (1/2*s+1/2) over Number Field in s with defining polynomial x^2 - 5
sage: K.<v> = NumberField(x^2 + 161*x - 150) sage: E = EllipticCurve([25105/216*v - 3839/36, 634768555/7776*v - 98002625/1296, 634768555/7776*v - 98002625/1296, 0, 0]) sage: M = E.global_integral_model(); M # choice varies, not tested Elliptic Curve defined by y^2 + (2094779518028859*v-1940492905300351)*x*y + (477997268472544193101178234454165304071127500*v-442791377441346852919930773849502871958097500)*y = x^3 + (26519784690047674853185542622500*v-24566525306469707225840460652500)*x^2 over Number Field in v with defining polynomial x^2 + 161*x - 150
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(161)*x - Integer(150), names=('v',)); (v,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(25105)/Integer(216)*v - Integer(3839)/Integer(36), Integer(634768555)/Integer(7776)*v - Integer(98002625)/Integer(1296), Integer(634768555)/Integer(7776)*v - Integer(98002625)/Integer(1296), Integer(0), Integer(0)]) >>> M = E.global_integral_model(); M # choice varies, not tested Elliptic Curve defined by y^2 + (2094779518028859*v-1940492905300351)*x*y + (477997268472544193101178234454165304071127500*v-442791377441346852919930773849502871958097500)*y = x^3 + (26519784690047674853185542622500*v-24566525306469707225840460652500)*x^2 over Number Field in v with defining polynomial x^2 + 161*x - 150
sage: R.<t> = QQ[] sage: K.<g> = NumberField(t^4 - t^3 - 3*t^2 - t + 1) sage: E = EllipticCurve([ -43/625*g^3 + 14/625*g^2 - 4/625*g + 706/625, -4862/78125*g^3 - 4074/78125*g^2 - 711/78125*g + 10304/78125, -4862/78125*g^3 - 4074/78125*g^2 - 711/78125*g + 10304/78125, 0,0]) sage: E.global_integral_model() Elliptic Curve defined by y^2 + (15*g^3-48*g-42)*x*y + (-111510*g^3-162162*g^2-44145*g+37638)*y = x^3 + (-954*g^3-1134*g^2+81*g+576)*x^2 over Number Field in g with defining polynomial t^4 - t^3 - 3*t^2 - t + 1
>>> from sage.all import * >>> R = QQ['t']; (t,) = R._first_ngens(1) >>> K = NumberField(t**Integer(4) - t**Integer(3) - Integer(3)*t**Integer(2) - t + Integer(1), names=('g',)); (g,) = K._first_ngens(1) >>> E = EllipticCurve([ -Integer(43)/Integer(625)*g**Integer(3) + Integer(14)/Integer(625)*g**Integer(2) - Integer(4)/Integer(625)*g + Integer(706)/Integer(625), -Integer(4862)/Integer(78125)*g**Integer(3) - Integer(4074)/Integer(78125)*g**Integer(2) - Integer(711)/Integer(78125)*g + Integer(10304)/Integer(78125), -Integer(4862)/Integer(78125)*g**Integer(3) - Integer(4074)/Integer(78125)*g**Integer(2) - Integer(711)/Integer(78125)*g + Integer(10304)/Integer(78125), Integer(0),Integer(0)]) >>> E.global_integral_model() Elliptic Curve defined by y^2 + (15*g^3-48*g-42)*x*y + (-111510*g^3-162162*g^2-44145*g+37638)*y = x^3 + (-954*g^3-1134*g^2+81*g+576)*x^2 over Number Field in g with defining polynomial t^4 - t^3 - 3*t^2 - t + 1
- is_Q_curve(maxp=100, certificate=False, verbose=False)[source]¶
Return
True
if this is a \(\QQ\)-curve, with optional certificate.INPUT:
maxp
– integer (default: 100); bound on primes used for checking necessary local conditions. The result will not depend on this, but using a larger value may returnFalse
faster.certificate
– boolean (default:False
); ifTrue
then a second value is returned giving a certificate for the \(\QQ\)-curve property
OUTPUT:
If
certificate
isFalse
: eitherTrue
(if \(E\) is a \(\QQ\)-curve), orFalse
.If
certificate
isTrue
: a tuple consisting of a boolean flag as before and a certificate, defined as follows:when the flag is
True
, so \(E\) is a \(\QQ\)-curve:either {‘CM’:\(D\)} where \(D\) is a negative discriminant, when \(E\) has potential CM with discriminant \(D\);
otherwise {‘CM’: \(0\), ‘core_poly’: \(f\), ‘rho’: \(\rho\), ‘r’: \(r\), ‘N’: \(N\)}, when \(E\) is a non-CM \(\QQ\)-curve, where the core polynomial \(f\) is an irreducible monic polynomial over \(QQ\) of degree \(2^\rho\), all of whose roots are \(j\)-invariants of curves isogenous to \(E\), the core level \(N\) is a square-free integer with \(r\) prime factors which is the LCM of the degrees of the isogenies between these conjugates. For example, if there exists a curve \(E'\) isogenous to \(E\) with \(j(E')=j\in\QQ\), then the certificate is {‘CM’:0, ‘r’:0, ‘rho’:0, ‘core_poly’: x-j, ‘N’:1}.
when the flag is
False
, so \(E\) is not a \(\QQ\)-curve, the certificate is a prime \(p\) such that the reductions of \(E\) at the primes dividing \(p\) are inconsistent with the property of being a \(\QQ\)-curve. See the documentation forsage.src.schemes.elliptic_curves.Qcurves.is_Q_curve()
for details.
ALGORITHM:
See the documentation for
sage.src.schemes.elliptic_curves.Qcurves.is_Q_curve()
, and [CrNa2020] for details.EXAMPLES:
A non-CM curve over \(\QQ\) and a CM curve over \(\QQ\) are both trivially \(\QQ\)-curves:
sage: E = EllipticCurve([1,2,3,4,5]) sage: flag, cert = E.is_Q_curve(certificate=True) sage: flag True sage: cert {'CM': 0, 'N': 1, 'core_poly': x, 'r': 0, 'rho': 0} sage: E = EllipticCurve(j=8000) sage: flag, cert = E.is_Q_curve(certificate=True) sage: flag True sage: cert {'CM': -8}
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(2),Integer(3),Integer(4),Integer(5)]) >>> flag, cert = E.is_Q_curve(certificate=True) >>> flag True >>> cert {'CM': 0, 'N': 1, 'core_poly': x, 'r': 0, 'rho': 0} >>> E = EllipticCurve(j=Integer(8000)) >>> flag, cert = E.is_Q_curve(certificate=True) >>> flag True >>> cert {'CM': -8}
A non-\(\QQ\)-curve over a quartic field. The local data at bad primes above \(3\) is inconsistent:
sage: R.<x> = PolynomialRing(QQ) sage: K.<a> = NumberField(R([3, 0, -5, 0, 1])) sage: E = EllipticCurve([K([-3,-4,1,1]), K([4,-1,-1,0]), K([-2,0,1,0]), K([-621,778,138,-178]), K([9509,2046,-24728,10380])]) sage: E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + (a^3+a^2-4*a-3)*x*y + (a^2-2)*y = x^3 + (-a^2-a+4)*x^2 + (-178*a^3+138*a^2+778*a-621)*x + (10380*a^3-24728*a^2+2046*a+9509) over Number Field in a with defining polynomial x^4 - 5*x^2 + 3 is a Q-curve No: inconsistency at the 2 primes dividing 3 - potentially multiplicative: [True, False] (False, 3)
>>> from sage.all import * >>> R = PolynomialRing(QQ, names=('x',)); (x,) = R._first_ngens(1) >>> K = NumberField(R([Integer(3), Integer(0), -Integer(5), Integer(0), Integer(1)]), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([K([-Integer(3),-Integer(4),Integer(1),Integer(1)]), K([Integer(4),-Integer(1),-Integer(1),Integer(0)]), K([-Integer(2),Integer(0),Integer(1),Integer(0)]), K([-Integer(621),Integer(778),Integer(138),-Integer(178)]), K([Integer(9509),Integer(2046),-Integer(24728),Integer(10380)])]) >>> E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + (a^3+a^2-4*a-3)*x*y + (a^2-2)*y = x^3 + (-a^2-a+4)*x^2 + (-178*a^3+138*a^2+778*a-621)*x + (10380*a^3-24728*a^2+2046*a+9509) over Number Field in a with defining polynomial x^4 - 5*x^2 + 3 is a Q-curve No: inconsistency at the 2 primes dividing 3 - potentially multiplicative: [True, False] (False, 3)
A non-\(\QQ\)-curve over a quadratic field. The local data at bad primes is consistent, but the local test at good primes above \(13\) is not:
sage: K.<a> = NumberField(R([-10, 0, 1])) sage: E = EllipticCurve([K([0,1]), K([-1,-1]), K([0,0]), K([-236,40]), K([-1840,464])]) sage: E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + a*x*y = x^3 + (-a-1)*x^2 + (40*a-236)*x + (464*a-1840) over Number Field in a with defining polynomial x^2 - 10 is a Q-curve Applying local tests at good primes above p<=100 No: inconsistency at the 2 ordinary primes dividing 13 - Frobenius discriminants mod squares: [-1, -3] No: local test at p=13 failed (False, 13)
>>> from sage.all import * >>> K = NumberField(R([-Integer(10), Integer(0), Integer(1)]), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([K([Integer(0),Integer(1)]), K([-Integer(1),-Integer(1)]), K([Integer(0),Integer(0)]), K([-Integer(236),Integer(40)]), K([-Integer(1840),Integer(464)])]) >>> E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + a*x*y = x^3 + (-a-1)*x^2 + (40*a-236)*x + (464*a-1840) over Number Field in a with defining polynomial x^2 - 10 is a Q-curve Applying local tests at good primes above p<=100 No: inconsistency at the 2 ordinary primes dividing 13 - Frobenius discriminants mod squares: [-1, -3] No: local test at p=13 failed (False, 13)
A quadratic \(\QQ\)-curve with CM discriminant \(-15\) (so the \(j\)-invariant is not in \(\QQ\)):
sage: R.<x> = PolynomialRing(QQ) sage: K.<a> = NumberField(R([-1, -1, 1])) sage: E = EllipticCurve([K([1,0]), K([-1,0]), K([0,1]), K([0,-2]), K([0,1])]) sage: E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + x*y + a*y = x^3 + (-1)*x^2 + (-2*a)*x + a over Number Field in a with defining polynomial x^2 - x - 1 is a Q-curve Yes: E is CM (discriminant -15) (True, {'CM': -15})
>>> from sage.all import * >>> R = PolynomialRing(QQ, names=('x',)); (x,) = R._first_ngens(1) >>> K = NumberField(R([-Integer(1), -Integer(1), Integer(1)]), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([K([Integer(1),Integer(0)]), K([-Integer(1),Integer(0)]), K([Integer(0),Integer(1)]), K([Integer(0),-Integer(2)]), K([Integer(0),Integer(1)])]) >>> E.is_Q_curve(certificate=True, verbose=True) Checking whether Elliptic Curve defined by y^2 + x*y + a*y = x^3 + (-1)*x^2 + (-2*a)*x + a over Number Field in a with defining polynomial x^2 - x - 1 is a Q-curve Yes: E is CM (discriminant -15) (True, {'CM': -15})
An example over \(\QQ(\sqrt{2},\sqrt{3})\). The \(j\)-invariant is in \(\QQ(\sqrt{6})\), so computations will be done over that field, and in fact there is an isogenous curve with rational \(j\), so we have a so-called rational \(\QQ\)-curve:
sage: K.<a> = NumberField(R([1, 0, -4, 0, 1])) sage: E = EllipticCurve([K([-2,-4,1,1]), K([0,1,0,0]), K([0,1,0,0]), K([-4780,9170,1265,-2463]), K([163923,-316598,-43876,84852])]) sage: flag, cert = E.is_Q_curve(certificate=True) # long time sage: flag # long time True sage: cert # long time {'CM': 0, 'N': 1, 'core_degs': [1], 'core_poly': x - 85184/3, 'r': 0, 'rho': 0}
>>> from sage.all import * >>> K = NumberField(R([Integer(1), Integer(0), -Integer(4), Integer(0), Integer(1)]), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([K([-Integer(2),-Integer(4),Integer(1),Integer(1)]), K([Integer(0),Integer(1),Integer(0),Integer(0)]), K([Integer(0),Integer(1),Integer(0),Integer(0)]), K([-Integer(4780),Integer(9170),Integer(1265),-Integer(2463)]), K([Integer(163923),-Integer(316598),-Integer(43876),Integer(84852)])]) >>> flag, cert = E.is_Q_curve(certificate=True) # long time >>> flag # long time True >>> cert # long time {'CM': 0, 'N': 1, 'core_degs': [1], 'core_poly': x - 85184/3, 'r': 0, 'rho': 0}
Over the same field, a so-called strict \(\QQ\)-curve which is not isogenous to one with rational \(j\), but whose core field is quadratic. In fact the isogeny class over \(K\) consists of \(6\) curves, four with conjugate quartic \(j\)-invariants and \(2\) with quadratic conjugate \(j\)-invariants in \(\QQ(\sqrt{3})\) (but which are not base-changes from the quadratic subfield):
sage: E = EllipticCurve([K([0,-3,0,1]), K([1,4,0,-1]), K([0,0,0,0]), K([-2,-16,0,4]), K([-19,-32,4,8])]) sage: flag, cert = E.is_Q_curve(certificate=True) # long time sage: flag # long time True sage: cert # long time {'CM': 0, 'N': 2, 'core_degs': [1, 2], 'core_poly': x^2 - 840064*x + 1593413632, 'r': 1, 'rho': 1}
>>> from sage.all import * >>> E = EllipticCurve([K([Integer(0),-Integer(3),Integer(0),Integer(1)]), K([Integer(1),Integer(4),Integer(0),-Integer(1)]), K([Integer(0),Integer(0),Integer(0),Integer(0)]), K([-Integer(2),-Integer(16),Integer(0),Integer(4)]), K([-Integer(19),-Integer(32),Integer(4),Integer(8)])]) >>> flag, cert = E.is_Q_curve(certificate=True) # long time >>> flag # long time True >>> cert # long time {'CM': 0, 'N': 2, 'core_degs': [1, 2], 'core_poly': x^2 - 840064*x + 1593413632, 'r': 1, 'rho': 1}
- is_global_integral_model()[source]¶
Return whether
self
is integral at all primes.EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([i/5,i/5,i/5,i/5,i/5]) sage: P1, P2 = K.primes_above(5) sage: Emin = E.global_integral_model() sage: Emin.is_global_integral_model() True
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5)]) >>> P1, P2 = K.primes_above(Integer(5)) >>> Emin = E.global_integral_model() >>> Emin.is_global_integral_model() True
- is_global_minimal_model()[source]¶
Return whether this elliptic curve is a global minimal model.
OUTPUT:
Boolean, False if E is not integral, or if E is non-minimal at some prime, else True.
EXAMPLES:
sage: K.<a> = NumberField(x^2 - 10) sage: E = EllipticCurve([0, 0, 0, -22500, 750000*a]) sage: E.is_global_minimal_model() False sage: E.non_minimal_primes() [Fractional ideal (2, a), Fractional ideal (5, a)] sage: E = EllipticCurve([0, 0, 0, -3024, 46224]) sage: E.is_global_minimal_model() False sage: E.non_minimal_primes() [2, 3] sage: Emin = E.global_minimal_model() sage: Emin.is_global_minimal_model() True
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(22500), Integer(750000)*a]) >>> E.is_global_minimal_model() False >>> E.non_minimal_primes() [Fractional ideal (2, a), Fractional ideal (5, a)] >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(3024), Integer(46224)]) >>> E.is_global_minimal_model() False >>> E.non_minimal_primes() [2, 3] >>> Emin = E.global_minimal_model() >>> Emin.is_global_minimal_model() True
A necessary condition to be a global minimal model is that the model must be globally integral:
sage: E = EllipticCurve([0,0,0,1/2,1/3]) sage: E.is_global_minimal_model() False sage: Emin.is_global_minimal_model() True sage: Emin.ainvs() (0, 1, 1, -2, 0)
>>> from sage.all import * >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),Integer(1)/Integer(2),Integer(1)/Integer(3)]) >>> E.is_global_minimal_model() False >>> Emin.is_global_minimal_model() True >>> Emin.ainvs() (0, 1, 1, -2, 0)
- is_isogenous(other, proof=True, maxnorm=100)[source]¶
Return whether or not
self
is isogenous toother
.INPUT:
other
– another elliptic curveproof
– boolean (default:True
); ifFalse
, the function will returnTrue
whenever the two curves have the same conductor and are isogenous modulo \(p\) for all primes \(p\) of norm up tomaxnorm
. IfTrue
, the function returns False when the previous condition does not hold, and if it does hold we compute the complete isogeny class to see if the curves are indeed isogenous.maxnorm
– integer (default: 100); the maximum norm of primes \(p\) for which isogeny modulo \(p\) will be checked
OUTPUT:
boolean;
True
if there is an isogeny from curveself
to curveother
.EXAMPLES:
sage: x = polygen(QQ, 'x') sage: F = NumberField(x^2 - 2, 's'); F Number Field in s with defining polynomial x^2 - 2 sage: E1 = EllipticCurve(F, [7,8]) sage: E2 = EllipticCurve(F, [0,5,0,1,0]) sage: E3 = EllipticCurve(F, [0,-10,0,21,0]) sage: E1.is_isogenous(E2) False sage: E1.is_isogenous(E1) True sage: E2.is_isogenous(E2) True sage: E2.is_isogenous(E1) False sage: E2.is_isogenous(E3) True
>>> from sage.all import * >>> x = polygen(QQ, 'x') >>> F = NumberField(x**Integer(2) - Integer(2), 's'); F Number Field in s with defining polynomial x^2 - 2 >>> E1 = EllipticCurve(F, [Integer(7),Integer(8)]) >>> E2 = EllipticCurve(F, [Integer(0),Integer(5),Integer(0),Integer(1),Integer(0)]) >>> E3 = EllipticCurve(F, [Integer(0),-Integer(10),Integer(0),Integer(21),Integer(0)]) >>> E1.is_isogenous(E2) False >>> E1.is_isogenous(E1) True >>> E2.is_isogenous(E2) True >>> E2.is_isogenous(E1) False >>> E2.is_isogenous(E3) True
sage: x = polygen(QQ, 'x') sage: F = NumberField(x^2 - 2, 's'); F Number Field in s with defining polynomial x^2 - 2 sage: E = EllipticCurve('14a1') sage: EE = EllipticCurve('14a2') sage: E1 = E.change_ring(F) sage: E2 = EE.change_ring(F) sage: E1.is_isogenous(E2) True
>>> from sage.all import * >>> x = polygen(QQ, 'x') >>> F = NumberField(x**Integer(2) - Integer(2), 's'); F Number Field in s with defining polynomial x^2 - 2 >>> E = EllipticCurve('14a1') >>> EE = EllipticCurve('14a2') >>> E1 = E.change_ring(F) >>> E2 = EE.change_ring(F) >>> E1.is_isogenous(E2) True
sage: x = polygen(QQ, 'x') sage: F = NumberField(x^2 - 2, 's'); F Number Field in s with defining polynomial x^2 - 2 sage: k.<a> = NumberField(x^3 + 7) sage: E = EllipticCurve(F, [7,8]) sage: EE = EllipticCurve(k, [2, 2]) sage: E.is_isogenous(EE) Traceback (most recent call last): ... ValueError: Second argument must be defined over the same number field.
>>> from sage.all import * >>> x = polygen(QQ, 'x') >>> F = NumberField(x**Integer(2) - Integer(2), 's'); F Number Field in s with defining polynomial x^2 - 2 >>> k = NumberField(x**Integer(3) + Integer(7), names=('a',)); (a,) = k._first_ngens(1) >>> E = EllipticCurve(F, [Integer(7),Integer(8)]) >>> EE = EllipticCurve(k, [Integer(2), Integer(2)]) >>> E.is_isogenous(EE) Traceback (most recent call last): ... ValueError: Second argument must be defined over the same number field.
Some examples from Cremona’s 1981 tables:
sage: K.<i> = QuadraticField(-1) sage: E1 = EllipticCurve([i + 1, 0, 1, -240*i - 400, -2869*i - 2627]) sage: E1.conductor() Fractional ideal (-4*i - 7) sage: E2 = EllipticCurve([1+i,0,1,0,0]) sage: E2.conductor() Fractional ideal (-4*i - 7) sage: E1.is_isogenous(E2) # long time True sage: E1.is_isogenous(E2, proof=False) # faster (~170ms) True
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E1 = EllipticCurve([i + Integer(1), Integer(0), Integer(1), -Integer(240)*i - Integer(400), -Integer(2869)*i - Integer(2627)]) >>> E1.conductor() Fractional ideal (-4*i - 7) >>> E2 = EllipticCurve([Integer(1)+i,Integer(0),Integer(1),Integer(0),Integer(0)]) >>> E2.conductor() Fractional ideal (-4*i - 7) >>> E1.is_isogenous(E2) # long time True >>> E1.is_isogenous(E2, proof=False) # faster (~170ms) True
In this case E1 and E2 are in fact 9-isogenous, as may be deduced from the following:
sage: E3 = EllipticCurve([i + 1, 0, 1, -5*i - 5, -2*i - 5]) sage: E3.is_isogenous(E1) True sage: E3.is_isogenous(E2) True sage: E1.isogeny_degree(E2) # long time 9
>>> from sage.all import * >>> E3 = EllipticCurve([i + Integer(1), Integer(0), Integer(1), -Integer(5)*i - Integer(5), -Integer(2)*i - Integer(5)]) >>> E3.is_isogenous(E1) True >>> E3.is_isogenous(E2) True >>> E1.isogeny_degree(E2) # long time 9
- is_local_integral_model(*P)[source]¶
Test if
self
is integral at the prime ideal \(P\), or at all the primes if \(P\) is a list or tuple.INPUT:
*P
– a prime ideal, or a list or tuple of primes
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: P1, P2 = K.primes_above(5) sage: E = EllipticCurve([i/5,i/5,i/5,i/5,i/5]) sage: E.is_local_integral_model(P1, P2) False sage: Emin = E.local_integral_model(P1, P2) sage: Emin.is_local_integral_model(P1, P2) True
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> P1, P2 = K.primes_above(Integer(5)) >>> E = EllipticCurve([i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5)]) >>> E.is_local_integral_model(P1, P2) False >>> Emin = E.local_integral_model(P1, P2) >>> Emin.is_local_integral_model(P1, P2) True
- isogenies_prime_degree(l=None, algorithm='Billerey', minimal_models=True)[source]¶
Return a list of \(\ell\)-isogenies from
self
, where \(\ell\) is a prime.INPUT:
l
– eitherNone
or a prime or a list of primesalgorithm
– string (default:'Billerey'
); the algorithm to use to compute the reducible primes whenl
isNone
. Ignored for CM curves or ifl
is provided. Values are ‘Billerey’ (default), ‘Larson’, and ‘heuristic’.minimal_models
– boolean (default:True
); ifTrue
, all curves computed will be minimal or semi-minimal models. Over fields of larger degree it can be expensive to compute these so set toFalse
.
OUTPUT: list; \(\ell\)-isogenies for the given \(\ell\) or if \(\ell\) is
None
, all isogenies of prime degree (see below for the CM case)Note
Over \(\QQ\), the codomains of the isogenies returned are standard minimal models. Over other number fields they are global minimal models if these exist, otherwise models which are minimal at all but one prime.
Note
For curves with rational CM, isogenies of primes degree exist for infinitely many primes \(\ell\), though there are only finitely many isogenous curves up to isomorphism. The list returned only includes one isogeny of prime degree for each codomain.
EXAMPLES:
sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve(K, [0,0,0,0,1]) sage: isogs = E.isogenies_prime_degree() sage: [phi.degree() for phi in isogs] [2, 3] sage: pol = PolynomialRing(QQ,'x')([1,-3,5,-5,5,-3,1]) sage: L.<a> = NumberField(pol) sage: js = hilbert_class_polynomial(-23).roots(L, multiplicities=False); len(js) 3 sage: E = EllipticCurve(j=js[0]) sage: len(E.isogenies_prime_degree()) # long time 3
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),Integer(0),Integer(0),Integer(0),Integer(1)]) >>> isogs = E.isogenies_prime_degree() >>> [phi.degree() for phi in isogs] [2, 3] >>> pol = PolynomialRing(QQ,'x')([Integer(1),-Integer(3),Integer(5),-Integer(5),Integer(5),-Integer(3),Integer(1)]) >>> L = NumberField(pol, names=('a',)); (a,) = L._first_ngens(1) >>> js = hilbert_class_polynomial(-Integer(23)).roots(L, multiplicities=False); len(js) 3 >>> E = EllipticCurve(j=js[Integer(0)]) >>> len(E.isogenies_prime_degree()) # long time 3
Set
minimal_models
to False to avoid computing minimal models of the isogenous curves, since that can be time-consuming since it requires computation of the class group:sage: proof.number_field(False) sage: K.<z> = CyclotomicField(53) sage: E = EllipticCurve(K, [0,6,0,2,0]) sage: E.isogenies_prime_degree(2, minimal_models=False) [Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 to Elliptic Curve defined by y^2 = x^3 + 6*x^2 + (-8)*x + (-48) over Cyclotomic Field of order 53 and degree 52] sage: E.isogenies_prime_degree(2, minimal_models=True) # not tested (10s) [Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 to Elliptic Curve defined by y^2 = x^3 + (-20)*x + (-16) over Cyclotomic Field of order 53 and degree 52]
>>> from sage.all import * >>> proof.number_field(False) >>> K = CyclotomicField(Integer(53), names=('z',)); (z,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),Integer(6),Integer(0),Integer(2),Integer(0)]) >>> E.isogenies_prime_degree(Integer(2), minimal_models=False) [Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 to Elliptic Curve defined by y^2 = x^3 + 6*x^2 + (-8)*x + (-48) over Cyclotomic Field of order 53 and degree 52] >>> E.isogenies_prime_degree(Integer(2), minimal_models=True) # not tested (10s) [Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 to Elliptic Curve defined by y^2 = x^3 + (-20)*x + (-16) over Cyclotomic Field of order 53 and degree 52]
- isogeny_class(reducible_primes=None, algorithm='Billerey', minimal_models=True)[source]¶
Return the isogeny class of this elliptic curve.
INPUT:
reducible_primes
– list of integers orNone
(default)); if notNone
then this should be a list of primes; in computing the isogeny class, only composites isogenies of these degrees will be used.algorithm
– string (default:'Billerey'
); the algorithm to use to compute the reducible primes. Ignored for CM curves or ifreducible_primes
is provided. Values are'Billerey'
(default),'Larson'
, and'heuristic'
.minimal_models
– boolean (default:True
); ifTrue
, all curves in the class will be minimal or semi-minimal models. Over fields of larger degree it can be expensive to compute these so set toFalse
.
OUTPUT:
An instance of the class
sage.schemes.elliptic_curves.isogeny_class.IsogenyClass_EC_NumberField
. From this object may be obtained a list of curves in the class, a matrix of the degrees of the isogenies between them, and the isogenies themselves.Note
If using the algorithm
'heuristic'
for non-CM curves, the result is not guaranteed to be the complete isogeny class, since only reducible primes up to the default bound inreducible_primes_naive()
(currently 1000) are tested. However, no examples of non-CM elliptic curves with reducible primes greater than 100 have yet been computed so the output is likely to be correct.Note
By default, the curves in the isogeny class will all be minimal models if these exist (for example, when the class number is \(1\)); otherwise they will be minimal at all but one prime. This behaviour can be switched off if desired, for example over fields where the computation of the class group would be too expensive.
EXAMPLES:
sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve(K, [0,0,0,0,1]) sage: C = E.isogeny_class(); C Isogeny class of Elliptic Curve defined by y^2 = x^3 + 1 over Number Field in i with defining polynomial x^2 + 1 with i = 1*I
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),Integer(0),Integer(0),Integer(0),Integer(1)]) >>> C = E.isogeny_class(); C Isogeny class of Elliptic Curve defined by y^2 = x^3 + 1 over Number Field in i with defining polynomial x^2 + 1 with i = 1*I
The curves in the class (sorted):
sage: [E1.ainvs() for E1 in C] [(0, 0, 0, 0, -27), (0, 0, 0, 0, 1), (i + 1, i, i + 1, -i + 3, 4*i), (i + 1, i, i + 1, -i + 33, -58*i)]
>>> from sage.all import * >>> [E1.ainvs() for E1 in C] [(0, 0, 0, 0, -27), (0, 0, 0, 0, 1), (i + 1, i, i + 1, -i + 3, 4*i), (i + 1, i, i + 1, -i + 33, -58*i)]
The matrix of degrees of cyclic isogenies between curves:
sage: C.matrix() [1 3 6 2] [3 1 2 6] [6 2 1 3] [2 6 3 1]
>>> from sage.all import * >>> C.matrix() [1 3 6 2] [3 1 2 6] [6 2 1 3] [2 6 3 1]
The array of isogenies themselves is not filled out but only contains those used to construct the class, the other entries containing the integer 0. This will be changed when the class
EllipticCurveIsogeny
allowed composition. In this case we used \(2\)-isogenies to go from 0 to 2 and from 1 to 3, and \(3\)-isogenies to go from 0 to 1 and from 2 to 3:sage: isogs = C.isogenies() sage: [((i,j), isogs[i][j].degree()) ....: for i in range(4) for j in range(4) if isogs[i][j] != 0] [((0, 1), 3), ((0, 3), 2), ((1, 0), 3), ((1, 2), 2), ((2, 1), 2), ((2, 3), 3), ((3, 0), 2), ((3, 2), 3)] sage: [((i,j), isogs[i][j].x_rational_map()) ....: for i in range(4) for j in range(4) if isogs[i][j] != 0] [((0, 1), (1/9*x^3 - 12)/x^2), ((0, 3), (-1/2*i*x^2 + i*x - 12*i)/(x - 3)), ((1, 0), (x^3 + 4)/x^2), ((1, 2), (-1/2*i*x^2 - i*x - 2*i)/(x + 1)), ((2, 1), (1/2*i*x^2 - x)/(x + 3/2*i)), ((2, 3), (x^3 + 4*i*x^2 - 10*x - 10*i)/(x^2 + 4*i*x - 4)), ((3, 0), (1/2*i*x^2 + x + 4*i)/(x - 5/2*i)), ((3, 2), (1/9*x^3 - 4/3*i*x^2 - 34/3*x + 226/9*i)/(x^2 - 8*i*x - 16))]
>>> from sage.all import * >>> isogs = C.isogenies() >>> [((i,j), isogs[i][j].degree()) ... for i in range(Integer(4)) for j in range(Integer(4)) if isogs[i][j] != Integer(0)] [((0, 1), 3), ((0, 3), 2), ((1, 0), 3), ((1, 2), 2), ((2, 1), 2), ((2, 3), 3), ((3, 0), 2), ((3, 2), 3)] >>> [((i,j), isogs[i][j].x_rational_map()) ... for i in range(Integer(4)) for j in range(Integer(4)) if isogs[i][j] != Integer(0)] [((0, 1), (1/9*x^3 - 12)/x^2), ((0, 3), (-1/2*i*x^2 + i*x - 12*i)/(x - 3)), ((1, 0), (x^3 + 4)/x^2), ((1, 2), (-1/2*i*x^2 - i*x - 2*i)/(x + 1)), ((2, 1), (1/2*i*x^2 - x)/(x + 3/2*i)), ((2, 3), (x^3 + 4*i*x^2 - 10*x - 10*i)/(x^2 + 4*i*x - 4)), ((3, 0), (1/2*i*x^2 + x + 4*i)/(x - 5/2*i)), ((3, 2), (1/9*x^3 - 4/3*i*x^2 - 34/3*x + 226/9*i)/(x^2 - 8*i*x - 16))]
The isogeny class may be visualized by obtaining its graph and plotting it:
sage: G = C.graph() sage: G.show(edge_labels=True) # long time sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve([1+i, -i, i, 1, 0]) sage: C = E.isogeny_class(); C # long time Isogeny class of Elliptic Curve defined by y^2 + (i+1)*x*y + i*y = x^3 + (-i)*x^2 + x over Number Field in i with defining polynomial x^2 + 1 with i = 1*I sage: len(C) # long time 6 sage: C.matrix() # long time [ 1 3 9 18 6 2] [ 3 1 3 6 2 6] [ 9 3 1 2 6 18] [18 6 2 1 3 9] [ 6 2 6 3 1 3] [ 2 6 18 9 3 1] sage: [E1.ainvs() for E1 in C] # long time [(i + 1, i - 1, i, -i - 1, -i + 1), (i + 1, i - 1, i, 14*i + 4, 7*i + 14), (i + 1, i - 1, i, 59*i + 99, 372*i - 410), (i + 1, -i, i, -240*i - 399, 2869*i + 2627), (i + 1, -i, i, -5*i - 4, 2*i + 5), (i + 1, -i, i, 1, 0)]
>>> from sage.all import * >>> G = C.graph() >>> G.show(edge_labels=True) # long time >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(1)+i, -i, i, Integer(1), Integer(0)]) >>> C = E.isogeny_class(); C # long time Isogeny class of Elliptic Curve defined by y^2 + (i+1)*x*y + i*y = x^3 + (-i)*x^2 + x over Number Field in i with defining polynomial x^2 + 1 with i = 1*I >>> len(C) # long time 6 >>> C.matrix() # long time [ 1 3 9 18 6 2] [ 3 1 3 6 2 6] [ 9 3 1 2 6 18] [18 6 2 1 3 9] [ 6 2 6 3 1 3] [ 2 6 18 9 3 1] >>> [E1.ainvs() for E1 in C] # long time [(i + 1, i - 1, i, -i - 1, -i + 1), (i + 1, i - 1, i, 14*i + 4, 7*i + 14), (i + 1, i - 1, i, 59*i + 99, 372*i - 410), (i + 1, -i, i, -240*i - 399, 2869*i + 2627), (i + 1, -i, i, -5*i - 4, 2*i + 5), (i + 1, -i, i, 1, 0)]
An example with CM by \(\sqrt{-5}\):
sage: pol = PolynomialRing(QQ,'x')([1,0,3,0,1]) sage: K.<c> = NumberField(pol) sage: j = 1480640 + 565760*c^2 sage: E = EllipticCurve(j=j) sage: E.has_cm() True sage: E.has_rational_cm() True sage: E.cm_discriminant() -20 sage: C = E.isogeny_class() sage: len(C) 2 sage: C.matrix() [1 2] [2 1] sage: [E.ainvs() for E in C] [(0, 0, 0, 83490*c^2 - 147015, -64739840*c^2 - 84465260), (0, 0, 0, -161535*c^2 + 70785, -62264180*c^3 + 6229080*c)] sage: C.isogenies()[0][1] Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + (83490*c^2-147015)*x + (-64739840*c^2-84465260) over Number Field in c with defining polynomial x^4 + 3*x^2 + 1 to Elliptic Curve defined by y^2 = x^3 + (-161535*c^2+70785)*x + (-62264180*c^3+6229080*c) over Number Field in c with defining polynomial x^4 + 3*x^2 + 1
>>> from sage.all import * >>> pol = PolynomialRing(QQ,'x')([Integer(1),Integer(0),Integer(3),Integer(0),Integer(1)]) >>> K = NumberField(pol, names=('c',)); (c,) = K._first_ngens(1) >>> j = Integer(1480640) + Integer(565760)*c**Integer(2) >>> E = EllipticCurve(j=j) >>> E.has_cm() True >>> E.has_rational_cm() True >>> E.cm_discriminant() -20 >>> C = E.isogeny_class() >>> len(C) 2 >>> C.matrix() [1 2] [2 1] >>> [E.ainvs() for E in C] [(0, 0, 0, 83490*c^2 - 147015, -64739840*c^2 - 84465260), (0, 0, 0, -161535*c^2 + 70785, -62264180*c^3 + 6229080*c)] >>> C.isogenies()[Integer(0)][Integer(1)] Isogeny of degree 2 from Elliptic Curve defined by y^2 = x^3 + (83490*c^2-147015)*x + (-64739840*c^2-84465260) over Number Field in c with defining polynomial x^4 + 3*x^2 + 1 to Elliptic Curve defined by y^2 = x^3 + (-161535*c^2+70785)*x + (-62264180*c^3+6229080*c) over Number Field in c with defining polynomial x^4 + 3*x^2 + 1
An example with CM by \(\sqrt{-23}\) (class number \(3\)):
sage: pol = PolynomialRing(QQ,'x')([1,-3,5,-5,5,-3,1]) sage: L.<a> = NumberField(pol) sage: js = hilbert_class_polynomial(-23).roots(L,multiplicities=False); len(js) 3 sage: E = EllipticCurve(j=js[0]) sage: E.has_rational_cm() True sage: len(E.isogenies_prime_degree()) # long time 3 sage: C = E.isogeny_class(); len(C) # long time 6
>>> from sage.all import * >>> pol = PolynomialRing(QQ,'x')([Integer(1),-Integer(3),Integer(5),-Integer(5),Integer(5),-Integer(3),Integer(1)]) >>> L = NumberField(pol, names=('a',)); (a,) = L._first_ngens(1) >>> js = hilbert_class_polynomial(-Integer(23)).roots(L,multiplicities=False); len(js) 3 >>> E = EllipticCurve(j=js[Integer(0)]) >>> E.has_rational_cm() True >>> len(E.isogenies_prime_degree()) # long time 3 >>> C = E.isogeny_class(); len(C) # long time 6
The reason for the isogeny class having size six while the class number is only \(3\) is that the class also contains three curves with CM by the order of discriminant \(-92=4\cdot(-23)\), which also has class number \(3\). The curves in the class are sorted first by CM discriminant (then lexicographically using a-invariants):
sage: [F.cm_discriminant() for F in C] # long time [-23, -23, -23, -92, -92, -92]
>>> from sage.all import * >>> [F.cm_discriminant() for F in C] # long time [-23, -23, -23, -92, -92, -92]
\(2\) splits in the order with discriminant \(-23\), into two primes of order \(3\) in the class group, each of which induces a \(2\)-isogeny to a curve with the same endomorphism ring; the third \(2\)-isogeny is to a curve with the smaller endomorphism ring:
sage: [phi.codomain().cm_discriminant() for phi in E.isogenies_prime_degree()] # long time [-92, -23, -23] sage: C.matrix() # long time # random [1 2 2 4 4 2] [2 1 2 4 2 4] [2 2 1 2 4 4] [4 4 2 1 3 3] [4 2 4 3 1 3] [2 4 4 3 3 1]
>>> from sage.all import * >>> [phi.codomain().cm_discriminant() for phi in E.isogenies_prime_degree()] # long time [-92, -23, -23] >>> C.matrix() # long time # random [1 2 2 4 4 2] [2 1 2 4 2 4] [2 2 1 2 4 4] [4 4 2 1 3 3] [4 2 4 3 1 3] [2 4 4 3 3 1]
The graph of this isogeny class has a shape which does not occur over \(\QQ\): a triangular prism. Note that for curves without CM, the graph has an edge between two curves if and only if they are connected by an isogeny of prime degree, and this degree is uniquely determined by the two curves, but in the CM case this property does not hold, since for pairs of curves in the class with the same endomorphism ring \(O\), the set of degrees of isogenies between them is the set of integers represented by a primitive integral binary quadratic form of discriminant \(\text{disc}(O)\), and this form represents infinitely many primes. In the matrix we give a small prime represented by the appropriate form. In this example, the matrix is formed by four \(3\times3\) blocks. The isogenies of degree \(2\) indicated by the upper left \(3\times3\) block of the matrix could be replaced by isogenies of any degree represented by the quadratic form \(2x^2+xy+3y^2\) of discriminant \(-23\). Similarly in the lower right block, the entries of \(3\) could be represented by any integers represented by the quadratic form \(3x^2+2xy+8y^2\) of discriminant \(-92\). In the top right block and lower left blocks, by contrast, the prime entries \(2\) are unique determined:
sage: G = C.graph() # long time sage: G.adjacency_matrix() # long time # random [0 1 1 0 0 1] [1 0 1 0 1 0] [1 1 0 1 0 0] [0 0 1 0 1 1] [0 1 0 1 0 1] [1 0 0 1 1 0] sage: Graph(polytopes.simplex(2).prism().adjacency_matrix()).is_isomorphic(G) # long time True
>>> from sage.all import * >>> G = C.graph() # long time >>> G.adjacency_matrix() # long time # random [0 1 1 0 0 1] [1 0 1 0 1 0] [1 1 0 1 0 0] [0 0 1 0 1 1] [0 1 0 1 0 1] [1 0 0 1 1 0] >>> Graph(polytopes.simplex(Integer(2)).prism().adjacency_matrix()).is_isomorphic(G) # long time True
To display the graph without any edge labels:
sage: G.show() # not tested
>>> from sage.all import * >>> G.show() # not tested
To display the graph with edge labels: by default, for curves with rational CM, the labels are the coefficients of the associated quadratic forms:
sage: G.show(edge_labels=True) # not tested
>>> from sage.all import * >>> G.show(edge_labels=True) # not tested
For an alternative view, first relabel the edges using only 2 labels to distinguish between isogenies between curves with the same endomorphism ring and isogenies between curves with different endomorphism rings, then use a 3-dimensional plot which can be rotated:
sage: for i, j, l in G.edge_iterator(): # long time ....: G.set_edge_label(i, j, l.count(',')) sage: G.show3d(color_by_label=True) # long time
>>> from sage.all import * >>> for i, j, l in G.edge_iterator(): # long time ... G.set_edge_label(i, j, l.count(',')) >>> G.show3d(color_by_label=True) # long time
A class number \(6\) example. First we set up the fields:
pol
defines the same field aspol26
but is simpler:sage: pol26 = hilbert_class_polynomial(-4*26) sage: pol = x^6 - x^5 + 2*x^4 + x^3 - 2*x^2 - x - 1 sage: K.<a> = NumberField(pol) sage: L.<b> = K.extension(x^2 + 26)
>>> from sage.all import * >>> pol26 = hilbert_class_polynomial(-Integer(4)*Integer(26)) >>> pol = x**Integer(6) - x**Integer(5) + Integer(2)*x**Integer(4) + x**Integer(3) - Integer(2)*x**Integer(2) - x - Integer(1) >>> K = NumberField(pol, names=('a',)); (a,) = K._first_ngens(1) >>> L = K.extension(x**Integer(2) + Integer(26), names=('b',)); (b,) = L._first_ngens(1)
Only \(2\) of the \(j\)-invariants with discriminant -104 are in \(K\), though all are in \(L\):
sage: len(pol26.roots(K)) 2 sage: len(pol26.roots(L)) 6
>>> from sage.all import * >>> len(pol26.roots(K)) 2 >>> len(pol26.roots(L)) 6
We create an elliptic curve defined over \(K\) with one of the \(j\)-invariants in \(K\):
sage: j1 = pol26.roots(K)[0][0] sage: E = EllipticCurve(j=j1) sage: E.has_cm() True sage: E.has_rational_cm() False sage: E.has_rational_cm(L) True
>>> from sage.all import * >>> j1 = pol26.roots(K)[Integer(0)][Integer(0)] >>> E = EllipticCurve(j=j1) >>> E.has_cm() True >>> E.has_rational_cm() False >>> E.has_rational_cm(L) True
Over \(K\) the isogeny class has size \(4\), with \(2\) curves for each of the \(2\) \(K\)-rational \(j\)-invariants:
sage: C = E.isogeny_class(); len(C) # long time (~11s) 4 sage: C.matrix() # long time [ 1 13 2 26] [13 1 26 2] [ 2 26 1 13] [26 2 13 1] sage: len(Set([EE.j_invariant() for EE in C.curves])) # long time 2
>>> from sage.all import * >>> C = E.isogeny_class(); len(C) # long time (~11s) 4 >>> C.matrix() # long time [ 1 13 2 26] [13 1 26 2] [ 2 26 1 13] [26 2 13 1] >>> len(Set([EE.j_invariant() for EE in C.curves])) # long time 2
Over \(L\), the isogeny class grows to size \(6\) (the class number):
sage: EL = E.change_ring(L) sage: CL = EL.isogeny_class(minimal_models=False) # long time sage: len(CL) # long time 6 sage: s1 = Set([EE.j_invariant() for EE in CL.curves]) # long time sage: s2 = Set(pol26.roots(L, multiplicities=False)) # long time sage: s1 == s2 # long time True
>>> from sage.all import * >>> EL = E.change_ring(L) >>> CL = EL.isogeny_class(minimal_models=False) # long time >>> len(CL) # long time 6 >>> s1 = Set([EE.j_invariant() for EE in CL.curves]) # long time >>> s2 = Set(pol26.roots(L, multiplicities=False)) # long time >>> s1 == s2 # long time True
In each position in the matrix of degrees, we see primes (or \(1\)). In fact the set of degrees of cyclic isogenies from curve \(i\) to curve \(j\) is infinite, and is the set of all integers represented by one of the primitive binary quadratic forms of discriminant \(-104\), from which we have selected a small prime:
sage: CL.matrix() # long time # random (see :issue:`19229`) [1 2 3 3 5 5] [2 1 5 5 3 3] [3 5 1 3 2 5] [3 5 3 1 5 2] [5 3 2 5 1 3] [5 3 5 2 3 1]
>>> from sage.all import * >>> CL.matrix() # long time # random (see :issue:`19229`) [1 2 3 3 5 5] [2 1 5 5 3 3] [3 5 1 3 2 5] [3 5 3 1 5 2] [5 3 2 5 1 3] [5 3 5 2 3 1]
To see the array of binary quadratic forms:
sage: CL.qf_matrix() # long time # random (see :issue:`19229`) [[[1], [2, 0, 13], [3, -2, 9], [3, -2, 9], [5, -4, 6], [5, -4, 6]], [[2, 0, 13], [1], [5, -4, 6], [5, -4, 6], [3, -2, 9], [3, -2, 9]], [[3, -2, 9], [5, -4, 6], [1], [3, -2, 9], [2, 0, 13], [5, -4, 6]], [[3, -2, 9], [5, -4, 6], [3, -2, 9], [1], [5, -4, 6], [2, 0, 13]], [[5, -4, 6], [3, -2, 9], [2, 0, 13], [5, -4, 6], [1], [3, -2, 9]], [[5, -4, 6], [3, -2, 9], [5, -4, 6], [2, 0, 13], [3, -2, 9], [1]]]
>>> from sage.all import * >>> CL.qf_matrix() # long time # random (see :issue:`19229`) [[[1], [2, 0, 13], [3, -2, 9], [3, -2, 9], [5, -4, 6], [5, -4, 6]], [[2, 0, 13], [1], [5, -4, 6], [5, -4, 6], [3, -2, 9], [3, -2, 9]], [[3, -2, 9], [5, -4, 6], [1], [3, -2, 9], [2, 0, 13], [5, -4, 6]], [[3, -2, 9], [5, -4, 6], [3, -2, 9], [1], [5, -4, 6], [2, 0, 13]], [[5, -4, 6], [3, -2, 9], [2, 0, 13], [5, -4, 6], [1], [3, -2, 9]], [[5, -4, 6], [3, -2, 9], [5, -4, 6], [2, 0, 13], [3, -2, 9], [1]]]
As in the non-CM case, the isogeny class may be visualized by obtaining its graph and plotting it. Since there are more edges than in the non-CM case, it may be preferable to omit the edge_labels:
sage: G = C.graph() sage: G.show(edge_labels=False) # long time
>>> from sage.all import * >>> G = C.graph() >>> G.show(edge_labels=False) # long time
It is possible to display a 3-dimensional plot, with colours to represent the different edge labels, in a form which can be rotated!:
sage: G.show3d(color_by_label=True) # long time
>>> from sage.all import * >>> G.show3d(color_by_label=True) # long time
Over larger number fields several options make computations tractable. Here we use algorithm ‘heuristic’ which avoids a rigorous computation of the reducible primes, only testing those less than 1000, and setting
minimal_models
toFalse
avoid having to compute the class group of \(K\). To obtain minimal models set proof.number_field(False); the class group computation takes an additional 10s:sage: K.<z> = CyclotomicField(53) sage: E = EllipticCurve(K,[0,6,0,2,0]) sage: C = E.isogeny_class(algorithm='heuristic', minimal_models=False); C # long time (10s) Isogeny class of Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 sage: C.curves # long time [Elliptic Curve defined by y^2 = x^3 + 6*x^2 + (-8)*x + (-48) over Cyclotomic Field of order 53 and degree 52, Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52]
>>> from sage.all import * >>> K = CyclotomicField(Integer(53), names=('z',)); (z,) = K._first_ngens(1) >>> E = EllipticCurve(K,[Integer(0),Integer(6),Integer(0),Integer(2),Integer(0)]) >>> C = E.isogeny_class(algorithm='heuristic', minimal_models=False); C # long time (10s) Isogeny class of Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52 >>> C.curves # long time [Elliptic Curve defined by y^2 = x^3 + 6*x^2 + (-8)*x + (-48) over Cyclotomic Field of order 53 and degree 52, Elliptic Curve defined by y^2 = x^3 + 6*x^2 + 2*x over Cyclotomic Field of order 53 and degree 52]
- isogeny_degree(other)[source]¶
Return the minimal degree of an isogeny between
self
andother
, or 0 if no isogeny exists.INPUT:
other
– another elliptic curve
OUTPUT: integer; the degree of an isogeny from
self
toother
, or 0EXAMPLES:
sage: x = QQ['x'].0 sage: F = NumberField(x^2 - 2, 's'); F Number Field in s with defining polynomial x^2 - 2 sage: E = EllipticCurve('14a1') sage: EE = EllipticCurve('14a2') sage: E1 = E.change_ring(F) sage: E2 = EE.change_ring(F) sage: E1.isogeny_degree(E2) # long time 2 sage: E2.isogeny_degree(E2) 1 sage: E5 = EllipticCurve('14a5').change_ring(F) sage: E1.isogeny_degree(E5) # long time 6 sage: E = EllipticCurve('11a1') sage: [E2.label() for E2 in cremona_curves([11..20]) if E.isogeny_degree(E2)] ['11a1', '11a2', '11a3'] sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve([1+i, -i, i, 1, 0]) sage: C = E.isogeny_class() # long time sage: [E.isogeny_degree(F) for F in C] # long time [2, 6, 18, 9, 3, 1]
>>> from sage.all import * >>> x = QQ['x'].gen(0) >>> F = NumberField(x**Integer(2) - Integer(2), 's'); F Number Field in s with defining polynomial x^2 - 2 >>> E = EllipticCurve('14a1') >>> EE = EllipticCurve('14a2') >>> E1 = E.change_ring(F) >>> E2 = EE.change_ring(F) >>> E1.isogeny_degree(E2) # long time 2 >>> E2.isogeny_degree(E2) 1 >>> E5 = EllipticCurve('14a5').change_ring(F) >>> E1.isogeny_degree(E5) # long time 6 >>> E = EllipticCurve('11a1') >>> [E2.label() for E2 in cremona_curves((ellipsis_range(Integer(11),Ellipsis,Integer(20)))) if E.isogeny_degree(E2)] ['11a1', '11a2', '11a3'] >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(1)+i, -i, i, Integer(1), Integer(0)]) >>> C = E.isogeny_class() # long time >>> [E.isogeny_degree(F) for F in C] # long time [2, 6, 18, 9, 3, 1]
- kodaira_symbol(P, proof=None)[source]¶
Return the Kodaira Symbol of this elliptic curve at the prime \(P\).
INPUT:
P
– eitherNone
or a prime ideal of the base field ofself
proof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.
OUTPUT: the Kodaira Symbol of the curve at
P
, represented as a stringEXAMPLES:
sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve([20, 225, 750, 625*a + 6875, 31250*a + 46875]) sage: bad_primes = E.discriminant().support(); bad_primes [Fractional ideal (-a), Fractional ideal (7/2*a - 81/2), Fractional ideal (-a - 52), Fractional ideal (2)] sage: [E.kodaira_symbol(P) for P in bad_primes] [I0, I1, I1, II] sage: K.<a> = QuadraticField(-11) sage: E = EllipticCurve('11a1').change_ring(K) sage: [E.kodaira_symbol(P) for P in K(11).support()] [I10]
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(20), Integer(225), Integer(750), Integer(625)*a + Integer(6875), Integer(31250)*a + Integer(46875)]) >>> bad_primes = E.discriminant().support(); bad_primes [Fractional ideal (-a), Fractional ideal (7/2*a - 81/2), Fractional ideal (-a - 52), Fractional ideal (2)] >>> [E.kodaira_symbol(P) for P in bad_primes] [I0, I1, I1, II] >>> K = QuadraticField(-Integer(11), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve('11a1').change_ring(K) >>> [E.kodaira_symbol(P) for P in K(Integer(11)).support()] [I10]
- lll_reduce(points, height_matrix=None, precision=None)[source]¶
Return an LLL-reduced basis from a given basis, with transform matrix.
INPUT:
points
– list of points on this elliptic curve, which should be independentheight_matrix
– the height-pairing matrix of the points, orNone
. IfNone
, it will be computed.precision
– number of bits of precision of intermediate computations (default:None
, for default RealField precision; ignored ifheight_matrix
is supplied)
OUTPUT: a tuple (newpoints, U) where U is a unimodular integer matrix, new_points is the transform of points by U, such that new_points has LLL-reduced height pairing matrix
Note
If the input points are not independent, the output depends on the undocumented behaviour of PARI’s pari:qflllgram function when applied to a Gram matrix which is not positive definite.
EXAMPLES:
Some examples over \(\QQ\):
sage: E = EllipticCurve([0, 1, 1, -2, 42]) sage: Pi = E.gens(); Pi [(-4 : 1 : 1), (-3 : 5 : 1), (-11/4 : 43/8 : 1), (-2 : 6 : 1)] sage: Qi, U = E.lll_reduce(Pi) sage: all(sum(U[i,j]*Pi[i] for i in range(4)) == Qi[j] for j in range(4)) True sage: sorted(Qi) [(-4 : 1 : 1), (-3 : 5 : 1), (-2 : 6 : 1), (0 : 6 : 1)] sage: U.det() 1 sage: E.regulator_of_points(Pi) 4.59088036960573 sage: E.regulator_of_points(Qi) 4.59088036960574
>>> from sage.all import * >>> E = EllipticCurve([Integer(0), Integer(1), Integer(1), -Integer(2), Integer(42)]) >>> Pi = E.gens(); Pi [(-4 : 1 : 1), (-3 : 5 : 1), (-11/4 : 43/8 : 1), (-2 : 6 : 1)] >>> Qi, U = E.lll_reduce(Pi) >>> all(sum(U[i,j]*Pi[i] for i in range(Integer(4))) == Qi[j] for j in range(Integer(4))) True >>> sorted(Qi) [(-4 : 1 : 1), (-3 : 5 : 1), (-2 : 6 : 1), (0 : 6 : 1)] >>> U.det() 1 >>> E.regulator_of_points(Pi) 4.59088036960573 >>> E.regulator_of_points(Qi) 4.59088036960574
sage: E = EllipticCurve([1,0,1,-120039822036992245303534619191166796374,504224992484910670010801799168082726759443756222911415116]) sage: xi = [2005024558054813068, ....: -4690836759490453344, ....: 4700156326649806635, ....: 6785546256295273860, ....: 6823803569166584943, ....: 7788809602110240789, ....: 27385442304350994620556, ....: 54284682060285253719/4, ....: -94200235260395075139/25, ....: -3463661055331841724647/576, ....: -6684065934033506970637/676, ....: -956077386192640344198/2209, ....: -27067471797013364392578/2809, ....: -25538866857137199063309/3721, ....: -1026325011760259051894331/108241, ....: 9351361230729481250627334/1366561, ....: 10100878635879432897339615/1423249, ....: 11499655868211022625340735/17522596, ....: 110352253665081002517811734/21353641, ....: 414280096426033094143668538257/285204544, ....: 36101712290699828042930087436/4098432361, ....: 45442463408503524215460183165/5424617104, ....: 983886013344700707678587482584/141566320009, ....: 1124614335716851053281176544216033/152487126016] sage: points = [E.lift_x(x) for x in xi] sage: newpoints, U = E.lll_reduce(points) # long time (35s on sage.math, 2011) sage: [P[0] for P in newpoints] # long time [6823803569166584943, 5949539878899294213, 2005024558054813068, 5864879778877955778, 23955263915878682727/4, 5922188321411938518, 5286988283823825378, 11465667352242779838, -11451575907286171572, 3502708072571012181, 1500143935183238709184/225, 27180522378120223419/4, -5811874164190604461581/625, 26807786527159569093, 7041412654828066743, 475656155255883588, 265757454726766017891/49, 7272142121019825303, 50628679173833693415/4, 6951643522366348968, 6842515151518070703, 111593750389650846885/16, 2607467890531740394315/9, -1829928525835506297]
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(0),Integer(1),-Integer(120039822036992245303534619191166796374),Integer(504224992484910670010801799168082726759443756222911415116)]) >>> xi = [Integer(2005024558054813068), ... -Integer(4690836759490453344), ... Integer(4700156326649806635), ... Integer(6785546256295273860), ... Integer(6823803569166584943), ... Integer(7788809602110240789), ... Integer(27385442304350994620556), ... Integer(54284682060285253719)/Integer(4), ... -Integer(94200235260395075139)/Integer(25), ... -Integer(3463661055331841724647)/Integer(576), ... -Integer(6684065934033506970637)/Integer(676), ... -Integer(956077386192640344198)/Integer(2209), ... -Integer(27067471797013364392578)/Integer(2809), ... -Integer(25538866857137199063309)/Integer(3721), ... -Integer(1026325011760259051894331)/Integer(108241), ... Integer(9351361230729481250627334)/Integer(1366561), ... Integer(10100878635879432897339615)/Integer(1423249), ... Integer(11499655868211022625340735)/Integer(17522596), ... Integer(110352253665081002517811734)/Integer(21353641), ... Integer(414280096426033094143668538257)/Integer(285204544), ... Integer(36101712290699828042930087436)/Integer(4098432361), ... Integer(45442463408503524215460183165)/Integer(5424617104), ... Integer(983886013344700707678587482584)/Integer(141566320009), ... Integer(1124614335716851053281176544216033)/Integer(152487126016)] >>> points = [E.lift_x(x) for x in xi] >>> newpoints, U = E.lll_reduce(points) # long time (35s on sage.math, 2011) >>> [P[Integer(0)] for P in newpoints] # long time [6823803569166584943, 5949539878899294213, 2005024558054813068, 5864879778877955778, 23955263915878682727/4, 5922188321411938518, 5286988283823825378, 11465667352242779838, -11451575907286171572, 3502708072571012181, 1500143935183238709184/225, 27180522378120223419/4, -5811874164190604461581/625, 26807786527159569093, 7041412654828066743, 475656155255883588, 265757454726766017891/49, 7272142121019825303, 50628679173833693415/4, 6951643522366348968, 6842515151518070703, 111593750389650846885/16, 2607467890531740394315/9, -1829928525835506297]
An example to show the explicit use of the height pairing matrix:
sage: E = EllipticCurve([0, 1, 1, -2, 42]) sage: Pi = E.gens() sage: H = E.height_pairing_matrix(Pi,3) sage: E.lll_reduce(Pi,height_matrix=H) ( [1 0 0 1] [0 1 0 1] [0 0 0 1] [(-4 : 1 : 1), (-3 : 5 : 1), (-2 : 6 : 1), (1 : -7 : 1)], [0 0 1 1] )
>>> from sage.all import * >>> E = EllipticCurve([Integer(0), Integer(1), Integer(1), -Integer(2), Integer(42)]) >>> Pi = E.gens() >>> H = E.height_pairing_matrix(Pi,Integer(3)) >>> E.lll_reduce(Pi,height_matrix=H) ( [1 0 0 1] [0 1 0 1] [0 0 0 1] [(-4 : 1 : 1), (-3 : 5 : 1), (-2 : 6 : 1), (1 : -7 : 1)], [0 0 1 1] )
Some examples over number fields (see Issue #9411):
sage: K.<a> = QuadraticField(-23, 'a') sage: E = EllipticCurve(K, [0,0,1,-1,0]) sage: P = E(-2, -(a+1)/2) sage: Q = E(0,-1) sage: E.lll_reduce([P,Q]) ( [0 1] [(0 : -1 : 1), (-2 : -1/2*a - 1/2 : 1)], [1 0] )
>>> from sage.all import * >>> K = QuadraticField(-Integer(23), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),Integer(0),Integer(1),-Integer(1),Integer(0)]) >>> P = E(-Integer(2), -(a+Integer(1))/Integer(2)) >>> Q = E(Integer(0),-Integer(1)) >>> E.lll_reduce([P,Q]) ( [0 1] [(0 : -1 : 1), (-2 : -1/2*a - 1/2 : 1)], [1 0] )
sage: K.<a> = QuadraticField(-5) sage: E = EllipticCurve(K, [0,a]) sage: points = [E.point([-211/841*a - 6044/841,-209584/24389*a + 53634/24389]), ....: E.point([-17/18*a - 1/9, -109/108*a - 277/108])] sage: E.lll_reduce(points) ( [(-a + 4 : -3*a + 7 : 1), (-17/18*a - 1/9 : 109/108*a + 277/108 : 1)], [ 1 0] [ 1 -1] )
>>> from sage.all import * >>> K = QuadraticField(-Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),a]) >>> points = [E.point([-Integer(211)/Integer(841)*a - Integer(6044)/Integer(841),-Integer(209584)/Integer(24389)*a + Integer(53634)/Integer(24389)]), ... E.point([-Integer(17)/Integer(18)*a - Integer(1)/Integer(9), -Integer(109)/Integer(108)*a - Integer(277)/Integer(108)])] >>> E.lll_reduce(points) ( [(-a + 4 : -3*a + 7 : 1), (-17/18*a - 1/9 : 109/108*a + 277/108 : 1)], [ 1 0] [ 1 -1] )
- local_data(P=None, proof=None, algorithm='pari', globally=False)[source]¶
Local data for this elliptic curve at the prime \(P\).
INPUT:
P
– eitherNone
, a prime ideal of the base field ofself
, or an element of the base field that generates a prime idealproof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.algorithm
– string (default:'pari'
); ignored unless the base field is \(\QQ\). If'pari'
, use the PARI C-library pari:ellglobalred implementation of Tate’s algorithm over \(\QQ\). If'generic'
, use the general number field implementation.globally
– whether the local algorithm uses global generators for the prime ideals. Default is False, which will not require any information about the class group. IfTrue
, a generator for \(P\) will be used if \(P\) is principal. Otherwise, or ifglobally
is False, the minimal model returned will preserve integrality at other primes, but not minimality.
OUTPUT:
If \(P\) is specified, returns the
EllipticCurveLocalData
object associated to the prime \(P\) for this curve. Otherwise, returns a list of such objects, one for each prime \(P\) in the support of the discriminant of this model.Note
The model is not required to be integral on input.
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([1 + i, 0, 1, 0, 0]) sage: E.local_data() [Local data at Fractional ideal (2*i + 1): Reduction type: bad non-split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 1 Conductor exponent: 1 Kodaira Symbol: I1 Tamagawa Number: 1, Local data at Fractional ideal (-2*i + 3): Reduction type: bad split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 2 Conductor exponent: 1 Kodaira Symbol: I2 Tamagawa Number: 2] sage: E.local_data(K.ideal(3)) Local data at Fractional ideal (3): Reduction type: good Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 0 Conductor exponent: 0 Kodaira Symbol: I0 Tamagawa Number: 1 sage: E.local_data(2*i + 1) Local data at Fractional ideal (2*i + 1): Reduction type: bad non-split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 1 Conductor exponent: 1 Kodaira Symbol: I1 Tamagawa Number: 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(1) + i, Integer(0), Integer(1), Integer(0), Integer(0)]) >>> E.local_data() [Local data at Fractional ideal (2*i + 1): Reduction type: bad non-split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 1 Conductor exponent: 1 Kodaira Symbol: I1 Tamagawa Number: 1, Local data at Fractional ideal (-2*i + 3): Reduction type: bad split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 2 Conductor exponent: 1 Kodaira Symbol: I2 Tamagawa Number: 2] >>> E.local_data(K.ideal(Integer(3))) Local data at Fractional ideal (3): Reduction type: good Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 0 Conductor exponent: 0 Kodaira Symbol: I0 Tamagawa Number: 1 >>> E.local_data(Integer(2)*i + Integer(1)) Local data at Fractional ideal (2*i + 1): Reduction type: bad non-split multiplicative Local minimal model: Elliptic Curve defined by y^2 + (i+1)*x*y + y = x^3 over Number Field in i with defining polynomial x^2 + 1 Minimal discriminant valuation: 1 Conductor exponent: 1 Kodaira Symbol: I1 Tamagawa Number: 1
An example raised in Issue #3897:
sage: E = EllipticCurve([1,1]) sage: E.local_data(3) Local data at Principal ideal (3) of Integer Ring: Reduction type: good Local minimal model: Elliptic Curve defined by y^2 = x^3 + x + 1 over Rational Field Minimal discriminant valuation: 0 Conductor exponent: 0 Kodaira Symbol: I0 Tamagawa Number: 1
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(1)]) >>> E.local_data(Integer(3)) Local data at Principal ideal (3) of Integer Ring: Reduction type: good Local minimal model: Elliptic Curve defined by y^2 = x^3 + x + 1 over Rational Field Minimal discriminant valuation: 0 Conductor exponent: 0 Kodaira Symbol: I0 Tamagawa Number: 1
- local_integral_model(*P)[source]¶
Return a model of
self
which is integral at the prime ideal \(P\).Note
The integrality at other primes is not affected, even if \(P\) is non-principal.
INPUT:
*P
– a prime ideal, or a list or tuple of primes
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: P1, P2 = K.primes_above(5) sage: E = EllipticCurve([i/5,i/5,i/5,i/5,i/5]) sage: E.local_integral_model((P1,P2)) Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> P1, P2 = K.primes_above(Integer(5)) >>> E = EllipticCurve([i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5),i/Integer(5)]) >>> E.local_integral_model((P1,P2)) Elliptic Curve defined by y^2 + (-i)*x*y + (-25*i)*y = x^3 + 5*i*x^2 + 125*i*x + 3125*i over Number Field in i with defining polynomial x^2 + 1
- local_minimal_model(P, proof=None, algorithm='pari')[source]¶
Return a model which is integral at all primes and minimal at \(P\).
INPUT:
P
– eitherNone
or a prime ideal of the base field ofself
proof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.algorithm
– string (default:'pari'
); ignored unless the base field is \(\QQ\). If'pari'
, use the PARI C-library pari:ellglobalred implementation of Tate’s algorithm over \(\QQ\). If'generic'
, use the general number field implementation.
OUTPUT:
A model of the curve which is minimal (and integral) at \(P\).
Note
The model is not required to be integral on input.
For principal \(P\), a generator is used as a uniformizer, and integrality or minimality at other primes is not affected. For non-principal \(P\), the minimal model returned will preserve integrality at other primes, but not minimality.
EXAMPLES:
sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve([20, 225, 750, 1250*a + 6250, 62500*a + 15625]) sage: P = K.ideal(a) sage: E.local_minimal_model(P).ainvs() (0, 1, 0, 2*a - 34, -4*a + 66)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(20), Integer(225), Integer(750), Integer(1250)*a + Integer(6250), Integer(62500)*a + Integer(15625)]) >>> P = K.ideal(a) >>> E.local_minimal_model(P).ainvs() (0, 1, 0, 2*a - 34, -4*a + 66)
- minimal_discriminant_ideal()[source]¶
Return the minimal discriminant ideal of this elliptic curve.
OUTPUT:
The integral ideal \(D\) whose valuation at every prime \(P\) is that of the local minimal model for \(E\) at \(P\). If \(E\) has a global minimal model, this will be the principal ideal generated by the discriminant of any such model, but otherwise it can be a proper divisor of the discriminant of any model.
EXAMPLES:
sage: K.<a> = NumberField(x^2 - x - 57) sage: K.class_number() 3 sage: E = EllipticCurve([a, -a, a, -5692-820*a, -259213-36720*a]) sage: K.ideal(E.discriminant()) Fractional ideal (90118662980*a + 636812084644) sage: K.ideal(E.discriminant()).factor() (Fractional ideal (2))^2 * (Fractional ideal (3, a + 2))^12
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - x - Integer(57), names=('a',)); (a,) = K._first_ngens(1) >>> K.class_number() 3 >>> E = EllipticCurve([a, -a, a, -Integer(5692)-Integer(820)*a, -Integer(259213)-Integer(36720)*a]) >>> K.ideal(E.discriminant()) Fractional ideal (90118662980*a + 636812084644) >>> K.ideal(E.discriminant()).factor() (Fractional ideal (2))^2 * (Fractional ideal (3, a + 2))^12
Here the minimal discriminant ideal is principal but there is no global minimal model since the quotient is the 12th power of a non-principal ideal:
sage: E.minimal_discriminant_ideal() Fractional ideal (4) sage: E.minimal_discriminant_ideal().factor() (Fractional ideal (2))^2
>>> from sage.all import * >>> E.minimal_discriminant_ideal() Fractional ideal (4) >>> E.minimal_discriminant_ideal().factor() (Fractional ideal (2))^2
If (and only if) the curve has everywhere good reduction the result is the unit ideal:
sage: K.<a> = NumberField(x^2 - 26) sage: E = EllipticCurve([a, a-1, a+1, 4*a+10, 2*a+6]) sage: E.conductor() Fractional ideal (1) sage: E.discriminant() -104030*a - 530451 sage: E.minimal_discriminant_ideal() Fractional ideal (1)
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(26), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([a, a-Integer(1), a+Integer(1), Integer(4)*a+Integer(10), Integer(2)*a+Integer(6)]) >>> E.conductor() Fractional ideal (1) >>> E.discriminant() -104030*a - 530451 >>> E.minimal_discriminant_ideal() Fractional ideal (1)
Over \(\QQ\), the result returned is an ideal of \(\ZZ\) rather than a fractional ideal of \(\QQ\):
sage: E = EllipticCurve([1,2,3,4,5]) sage: E.minimal_discriminant_ideal() Principal ideal (10351) of Integer Ring
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(2),Integer(3),Integer(4),Integer(5)]) >>> E.minimal_discriminant_ideal() Principal ideal (10351) of Integer Ring
- non_minimal_primes()[source]¶
Return a list of primes at which this elliptic curve is not minimal.
OUTPUT:
A list of prime ideals (or prime numbers when the base field is \(\QQ\), empty if this is a global minimal model.
EXAMPLES:
sage: K.<a> = NumberField(x^2 - 10) sage: E = EllipticCurve([0, 0, 0, -22500, 750000*a]) sage: E.non_minimal_primes() [Fractional ideal (2, a), Fractional ideal (5, a)] sage: K.ideal(E.discriminant()).factor() (Fractional ideal (2, a))^24 * (Fractional ideal (3, a + 1))^5 * (Fractional ideal (3, a + 2))^5 * (Fractional ideal (5, a))^24 * (Fractional ideal (7)) sage: E.minimal_discriminant_ideal().factor() (Fractional ideal (2, a))^12 * (Fractional ideal (3, a + 1))^5 * (Fractional ideal (3, a + 2))^5 * (Fractional ideal (7))
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(10), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(22500), Integer(750000)*a]) >>> E.non_minimal_primes() [Fractional ideal (2, a), Fractional ideal (5, a)] >>> K.ideal(E.discriminant()).factor() (Fractional ideal (2, a))^24 * (Fractional ideal (3, a + 1))^5 * (Fractional ideal (3, a + 2))^5 * (Fractional ideal (5, a))^24 * (Fractional ideal (7)) >>> E.minimal_discriminant_ideal().factor() (Fractional ideal (2, a))^12 * (Fractional ideal (3, a + 1))^5 * (Fractional ideal (3, a + 2))^5 * (Fractional ideal (7))
Over \(\QQ\), the primes returned are integers, not ideals:
sage: E = EllipticCurve([0, 0, 0, -3024, 46224]) sage: E.non_minimal_primes() [2, 3] sage: Emin = E.global_minimal_model() sage: Emin.non_minimal_primes() []
>>> from sage.all import * >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), -Integer(3024), Integer(46224)]) >>> E.non_minimal_primes() [2, 3] >>> Emin = E.global_minimal_model() >>> Emin.non_minimal_primes() []
If the model is not globally integral, a
ValueError
is raised:sage: E = EllipticCurve([0, 0, 0, 1/2, 1/3]) sage: E.non_minimal_primes() Traceback (most recent call last): ... ValueError: non_minimal_primes only defined for integral models
>>> from sage.all import * >>> E = EllipticCurve([Integer(0), Integer(0), Integer(0), Integer(1)/Integer(2), Integer(1)/Integer(3)]) >>> E.non_minimal_primes() Traceback (most recent call last): ... ValueError: non_minimal_primes only defined for integral models
- period_lattice(embedding)[source]¶
Return the period lattice of the elliptic curve for the given embedding of its base field with respect to the differential \(dx/(2y + a_1x + a_3)\).
INPUT:
embedding
– an embedding of the base number field into \(\RR\) or \(\CC\)
Note
The precision of the embedding is ignored: we only use the given embedding to determine which embedding into
QQbar
to use. Once the lattice has been initialized, periods can be computed to arbitrary precision.EXAMPLES:
First define a field with two real embeddings:
sage: K.<a> = NumberField(x^3 - 2) sage: E = EllipticCurve([0,0,0,a,2]) sage: embs = K.embeddings(CC); len(embs) 3
>>> from sage.all import * >>> K = NumberField(x**Integer(3) - Integer(2), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),a,Integer(2)]) >>> embs = K.embeddings(CC); len(embs) 3
For each embedding we have a different period lattice:
sage: E.period_lattice(embs[0]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> -0.6299605249474365? - 1.091123635971722?*I sage: E.period_lattice(embs[1]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> -0.6299605249474365? + 1.091123635971722?*I sage: E.period_lattice(embs[2]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> 1.259921049894873?
>>> from sage.all import * >>> E.period_lattice(embs[Integer(0)]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> -0.6299605249474365? - 1.091123635971722?*I >>> E.period_lattice(embs[Integer(1)]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> -0.6299605249474365? + 1.091123635971722?*I >>> E.period_lattice(embs[Integer(2)]) Period lattice associated to Elliptic Curve defined by y^2 = x^3 + a*x + 2 over Number Field in a with defining polynomial x^3 - 2 with respect to the embedding Ring morphism: From: Number Field in a with defining polynomial x^3 - 2 To: Algebraic Field Defn: a |--> 1.259921049894873?
Although the original embeddings have only the default precision, we can obtain the basis with higher precision later:
sage: L = E.period_lattice(embs[0]) sage: L.basis() (1.86405007647981 - 0.903761485143226*I, -0.149344633143919 - 2.06619546272945*I) sage: L.basis(prec=100) (1.8640500764798108425920506200 - 0.90376148514322594749786960975*I, -0.14934463314391922099120107422 - 2.0661954627294548995621225062*I)
>>> from sage.all import * >>> L = E.period_lattice(embs[Integer(0)]) >>> L.basis() (1.86405007647981 - 0.903761485143226*I, -0.149344633143919 - 2.06619546272945*I) >>> L.basis(prec=Integer(100)) (1.8640500764798108425920506200 - 0.90376148514322594749786960975*I, -0.14934463314391922099120107422 - 2.0661954627294548995621225062*I)
- rank(**kwds)[source]¶
Return the rank of this elliptic curve, if it can be determined.
Note
The optional parameters control the Simon two descent algorithm; see the documentation of
simon_two_descent()
for more details.INPUT:
verbose
– 0, 1, 2, or 3 (default: 0); the verbosity levellim1
– (default: 2) limit on trivial points on quarticslim3
– (default: 4) limit on points on ELS quarticslimtriv
– (default: 2) limit on trivial points on elliptic curvemaxprob
– (default: 20)limbigprime
– (default: 30) to distinguish between small and large prime numbers. Use probabilistic tests for large primes. If 0, do not use probabilistic tests.known_points
– (default:None
) list of known points on the curve
OUTPUT:
If the upper and lower bounds given by Simon two-descent are the same, then the rank has been uniquely identified and we return this. Otherwise, we raise a
ValueError
with an error message specifying the upper and lower bounds.Note
For non-quadratic number fields, this code does return, but it takes a long time.
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 + 23, 'a') sage: E = EllipticCurve(K, '37') sage: E == loads(dumps(E)) True sage: E.rank() 2
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) + Integer(23), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, '37') >>> E == loads(dumps(E)) True >>> E.rank() 2
Here is a curve with two-torsion in the Tate-Shafarevich group, so here the bounds given by the algorithm do not uniquely determine the rank:
sage: E = EllipticCurve("15a5") sage: K.<t> = NumberField(x^2 - 6) sage: EK = E.base_extend(K) sage: EK.rank(lim1=1, lim3=1, limtriv=1) Traceback (most recent call last): ... ValueError: There is insufficient data to determine the rank - 2-descent gave lower bound 0 and upper bound 2
>>> from sage.all import * >>> E = EllipticCurve("15a5") >>> K = NumberField(x**Integer(2) - Integer(6), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.rank(lim1=Integer(1), lim3=Integer(1), limtriv=Integer(1)) Traceback (most recent call last): ... ValueError: There is insufficient data to determine the rank - 2-descent gave lower bound 0 and upper bound 2
IMPLEMENTATION:
Uses Denis Simon’s PARI/GP scripts from http://www.math.unicaen.fr/~simon/.
- rank_bounds(**kwds)[source]¶
Return the lower and upper bounds using
simon_two_descent()
. The results ofsimon_two_descent()
are cached.Note
The optional parameters control the Simon two descent algorithm; see the documentation of
simon_two_descent()
for more details.INPUT:
verbose
– 0, 1, 2, or 3 (default: 0); the verbosity levellim1
– (default: 2) limit on trivial points on quarticslim3
– (default: 4) limit on points on ELS quarticslimtriv
– (default: 2) limit on trivial points on elliptic curvemaxprob
– (default: 20)limbigprime
– (default: 30) to distinguish between small and large prime numbers. Use probabilistic tests for large primes. If 0, do not use probabilistic tests.known_points
– (default:None
) list of known points on the curve
OUTPUT: lower and upper bounds for the rank of the Mordell-Weil group
Note
For non-quadratic number fields, this code does return, but it takes a long time.
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 + 23, 'a') sage: E = EllipticCurve(K, '37') sage: E == loads(dumps(E)) True sage: E.rank_bounds() (2, 2)
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) + Integer(23), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, '37') >>> E == loads(dumps(E)) True >>> E.rank_bounds() (2, 2)
Here is a curve with two-torsion, again the bounds coincide:
sage: Qrt5.<rt5> = NumberField(x^2 - 5) sage: E = EllipticCurve([0, 5-rt5, 0, rt5, 0]) sage: E.rank_bounds() (1, 1)
>>> from sage.all import * >>> Qrt5 = NumberField(x**Integer(2) - Integer(5), names=('rt5',)); (rt5,) = Qrt5._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(5)-rt5, Integer(0), rt5, Integer(0)]) >>> E.rank_bounds() (1, 1)
Finally an example with non-trivial 2-torsion in Sha. So the 2-descent will not be able to determine the rank, but can only give bounds:
sage: E = EllipticCurve("15a5") sage: K.<t> = NumberField(x^2 - 6) sage: EK = E.base_extend(K) sage: EK.rank_bounds(lim1=1, lim3=1, limtriv=1) (0, 2)
>>> from sage.all import * >>> E = EllipticCurve("15a5") >>> K = NumberField(x**Integer(2) - Integer(6), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.rank_bounds(lim1=Integer(1), lim3=Integer(1), limtriv=Integer(1)) (0, 2)
IMPLEMENTATION:
Uses Denis Simon’s PARI/GP scripts from http://www.math.unicaen.fr/~simon/.
- rational_points(**kwds)[source]¶
Find rational points on the elliptic curve, all arguments are passed on to
sage.schemes.generic.algebraic_scheme.rational_points()
.EXAMPLES:
sage: E = EllipticCurve('37a') sage: E.rational_points(bound=8) # long time [(-1 : -1 : 1), (-1 : 0 : 1), (0 : -1 : 1), (0 : 0 : 1), (0 : 1 : 0), (1/4 : -5/8 : 1), (1/4 : -3/8 : 1), (1 : -1 : 1), (1 : 0 : 1), (2 : -3 : 1), (2 : 2 : 1)]
>>> from sage.all import * >>> E = EllipticCurve('37a') >>> E.rational_points(bound=Integer(8)) # long time [(-1 : -1 : 1), (-1 : 0 : 1), (0 : -1 : 1), (0 : 0 : 1), (0 : 1 : 0), (1/4 : -5/8 : 1), (1/4 : -3/8 : 1), (1 : -1 : 1), (1 : 0 : 1), (2 : -3 : 1), (2 : 2 : 1)]
Check that Issue #26677 is fixed:
sage: E = EllipticCurve("11a1") sage: E.rational_points(bound=5) [(0 : 1 : 0), (5 : 5 : 1)] sage: E.rational_points(bound=6) # long time [(0 : 1 : 0), (5 : -6 : 1), (5 : 5 : 1)]
>>> from sage.all import * >>> E = EllipticCurve("11a1") >>> E.rational_points(bound=Integer(5)) [(0 : 1 : 0), (5 : 5 : 1)] >>> E.rational_points(bound=Integer(6)) # long time [(0 : 1 : 0), (5 : -6 : 1), (5 : 5 : 1)]
An example over a number field:
sage: E = EllipticCurve([1,0]) sage: pts = E.rational_points(bound=2, F=QuadraticField(-1)) sage: pts [(-a : 0 : 1), (0 : 0 : 1), (0 : 1 : 0), (a : 0 : 1)] sage: pts[0] + pts[1] (a : 0 : 1)
>>> from sage.all import * >>> E = EllipticCurve([Integer(1),Integer(0)]) >>> pts = E.rational_points(bound=Integer(2), F=QuadraticField(-Integer(1))) >>> pts [(-a : 0 : 1), (0 : 0 : 1), (0 : 1 : 0), (a : 0 : 1)] >>> pts[Integer(0)] + pts[Integer(1)] (a : 0 : 1)
- real_components(embedding)[source]¶
Return the number of real components with respect to a real embedding of the base field.
EXAMPLES:
sage: K.<a> = QuadraticField(5) sage: embs = K.real_embeddings() sage: E = EllipticCurve([0,1,1,a,a]) sage: [e(E.discriminant()) > 0 for e in embs] [True, False] sage: [E.real_components(e) for e in embs] [2, 1]
>>> from sage.all import * >>> K = QuadraticField(Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> embs = K.real_embeddings() >>> E = EllipticCurve([Integer(0),Integer(1),Integer(1),a,a]) >>> [e(E.discriminant()) > Integer(0) for e in embs] [True, False] >>> [E.real_components(e) for e in embs] [2, 1]
- reducible_primes(algorithm='Billerey', max_l=None, num_l=None, verbose=False)[source]¶
Return a finite set of primes \(\ell\) for which \(E\) has a K-rational \(\ell\)-isogeny.
For curves without CM the list returned is exactly the finite set of primes \(\ell\) for which the mod-\(\ell\) Galois representation is reducible. For curves with CM this set is infinite; we return a (not necessarily minimal) finite list of primes \(\ell\) such that every curve isogenous to this curve can be obtained by a finite sequence of isogenies of degree one of the primes in the list.
INPUT:
algorithm
– string; only relevant for non-CM curves Either ‘Billerey”, to use the methods of [Bil2011], ‘Larson’ to use Larson’s implementation using Galois representations, or ‘heuristic’ (see below).max_l
– integer orNone
; only relevant for non-CM curves and algorithms ‘Billerey’ and ‘heuristic. Controls the maximum prime used in either algorithm. IfNone
, use the default for that algorithm.num_l
– integer orNone
; only relevant for non-CM curves and algorithm ‘Billerey’. Controls the maximum number of primes used in the algorithm. IfNone
, use the default for that algorithm.
Note
The ‘heuristic’ algorithm only checks primes up to the bound
max_l
. This is faster but not guaranteed to be complete. Both the Billerey and Larson algorithms are rigorous.EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K = NumberField(x**2 - 29, 'a'); a = K.gen() sage: E = EllipticCurve([1, 0, ((5 + a)/2)**2, 0, 0]) sage: rho = E.galois_representation() sage: rho.reducible_primes() # long time [3, 5] sage: E.reducible_primes() # long time [3, 5] sage: K = NumberField(x**2 + 1, 'a') sage: E = EllipticCurve_from_j(K(1728)) # CM over K sage: rho = E.galois_representation() sage: rho.reducible_primes() # CM curves always return [0] [0] sage: E.reducible_primes() [2, 5] sage: E = EllipticCurve_from_j(K(0)) # CM but NOT over K sage: rho = E.galois_representation() sage: rho.reducible_primes() # long time [2, 3] sage: E.reducible_primes() [2, 3] sage: E = EllipticCurve_from_j(K(2268945/128)).global_minimal_model() # c.f. [Sut2012] sage: rho = E.galois_representation() sage: rho.isogeny_bound() # ..but there is no 7-isogeny, long time [7] sage: rho.reducible_primes() # long time [] sage: E.reducible_primes() # long time []
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(29), 'a'); a = K.gen() >>> E = EllipticCurve([Integer(1), Integer(0), ((Integer(5) + a)/Integer(2))**Integer(2), Integer(0), Integer(0)]) >>> rho = E.galois_representation() >>> rho.reducible_primes() # long time [3, 5] >>> E.reducible_primes() # long time [3, 5] >>> K = NumberField(x**Integer(2) + Integer(1), 'a') >>> E = EllipticCurve_from_j(K(Integer(1728))) # CM over K >>> rho = E.galois_representation() >>> rho.reducible_primes() # CM curves always return [0] [0] >>> E.reducible_primes() [2, 5] >>> E = EllipticCurve_from_j(K(Integer(0))) # CM but NOT over K >>> rho = E.galois_representation() >>> rho.reducible_primes() # long time [2, 3] >>> E.reducible_primes() [2, 3] >>> E = EllipticCurve_from_j(K(Integer(2268945)/Integer(128))).global_minimal_model() # c.f. [Sut2012] >>> rho = E.galois_representation() >>> rho.isogeny_bound() # ..but there is no 7-isogeny, long time [7] >>> rho.reducible_primes() # long time [] >>> E.reducible_primes() # long time []
- reduction(place)[source]¶
Return the reduction of the elliptic curve at a place of good reduction.
INPUT:
place
– a prime ideal in the base field of the curve
OUTPUT: an elliptic curve over a finite field, the residue field of the place
EXAMPLES:
sage: K.<i> = QuadraticField(-1) sage: EK = EllipticCurve([0, 0, 0, i, i+3]) sage: v = K.fractional_ideal(2*i+3) sage: EK.reduction(v) Elliptic Curve defined by y^2 = x^3 + 5*x + 8 over Residue field of Fractional ideal (2*i + 3) sage: EK.reduction(K.ideal(1+i)) Traceback (most recent call last): ... ValueError: The curve must have good reduction at the place. sage: EK.reduction(K.ideal(2)) Traceback (most recent call last): ... ValueError: The ideal must be prime. sage: K = QQ.extension(x^2 + x + 1, "a") sage: E = EllipticCurve([1024*K.0, 1024*K.0]) sage: E.reduction(2*K) Elliptic Curve defined by y^2 + (abar+1)*y = x^3 over Residue field in abar of Fractional ideal (2)
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EK = EllipticCurve([Integer(0), Integer(0), Integer(0), i, i+Integer(3)]) >>> v = K.fractional_ideal(Integer(2)*i+Integer(3)) >>> EK.reduction(v) Elliptic Curve defined by y^2 = x^3 + 5*x + 8 over Residue field of Fractional ideal (2*i + 3) >>> EK.reduction(K.ideal(Integer(1)+i)) Traceback (most recent call last): ... ValueError: The curve must have good reduction at the place. >>> EK.reduction(K.ideal(Integer(2))) Traceback (most recent call last): ... ValueError: The ideal must be prime. >>> K = QQ.extension(x**Integer(2) + x + Integer(1), "a") >>> E = EllipticCurve([Integer(1024)*K.gen(0), Integer(1024)*K.gen(0)]) >>> E.reduction(Integer(2)*K) Elliptic Curve defined by y^2 + (abar+1)*y = x^3 over Residue field in abar of Fractional ideal (2)
- regulator_of_points(points=[], precision=None, normalised=True)[source]¶
Return the regulator of the given points on this curve.
INPUT:
points
– (default: empty list) a list of points on this curveprecision
– integer orNone
(default); the precision in bits of the result (default: real precision ifNone
)normalised
– boolean (default:True
); ifTrue
, use normalised heights which are independent of base change. Otherwise use the non-normalised Néron-Tate height, as required for the regulator in the BSD conjecture
EXAMPLES:
sage: E = EllipticCurve('37a1') sage: P = E(0,0) sage: Q = E(1,0) sage: E.regulator_of_points([P,Q]) 0.000000000000000 sage: 2*P == Q True
>>> from sage.all import * >>> E = EllipticCurve('37a1') >>> P = E(Integer(0),Integer(0)) >>> Q = E(Integer(1),Integer(0)) >>> E.regulator_of_points([P,Q]) 0.000000000000000 >>> Integer(2)*P == Q True
sage: E = EllipticCurve('5077a1') sage: points = [E.lift_x(x) for x in [-2,-7/4,1]] sage: E.regulator_of_points(points) 0.417143558758384 sage: E.regulator_of_points(points, precision=100) 0.41714355875838396981711954462
>>> from sage.all import * >>> E = EllipticCurve('5077a1') >>> points = [E.lift_x(x) for x in [-Integer(2),-Integer(7)/Integer(4),Integer(1)]] >>> E.regulator_of_points(points) 0.417143558758384 >>> E.regulator_of_points(points, precision=Integer(100)) 0.41714355875838396981711954462
sage: E = EllipticCurve('389a') sage: E.regulator_of_points() 1.00000000000000 sage: points = [P,Q] = [E(-1,1), E(0,-1)] sage: E.regulator_of_points(points) 0.152460177943144 sage: E.regulator_of_points(points, precision=100) 0.15246017794314375162432475705 sage: E.regulator_of_points(points, precision=200) 0.15246017794314375162432475704945582324372707748663081784028 sage: E.regulator_of_points(points, precision=300) 0.152460177943143751624324757049455823243727077486630817840280980046053225683562463604114816
>>> from sage.all import * >>> E = EllipticCurve('389a') >>> E.regulator_of_points() 1.00000000000000 >>> points = [P,Q] = [E(-Integer(1),Integer(1)), E(Integer(0),-Integer(1))] >>> E.regulator_of_points(points) 0.152460177943144 >>> E.regulator_of_points(points, precision=Integer(100)) 0.15246017794314375162432475705 >>> E.regulator_of_points(points, precision=Integer(200)) 0.15246017794314375162432475704945582324372707748663081784028 >>> E.regulator_of_points(points, precision=Integer(300)) 0.152460177943143751624324757049455823243727077486630817840280980046053225683562463604114816
Examples over number fields:
sage: K.<a> = QuadraticField(97) sage: E = EllipticCurve(K, [1,1]) sage: P = E(0,1) sage: P.height() 0.476223106404866 sage: E.regulator_of_points([P]) 0.476223106404866
>>> from sage.all import * >>> K = QuadraticField(Integer(97), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(1),Integer(1)]) >>> P = E(Integer(0),Integer(1)) >>> P.height() 0.476223106404866 >>> E.regulator_of_points([P]) 0.476223106404866
When the parameter
normalised
is set toFalse
, each height is multiplied by the degree \(d\) of the base field, and the regulator of \(r\) points is multiplied by \(d^r\):sage: P.height(normalised=False) 0.952446212809731 sage: E.regulator_of_points([P], normalised=False) 0.952446212809731
>>> from sage.all import * >>> P.height(normalised=False) 0.952446212809731 >>> E.regulator_of_points([P], normalised=False) 0.952446212809731
sage: E = EllipticCurve('11a1') sage: x = polygen(QQ) sage: K.<t> = NumberField(x^2 + 47) sage: EK = E.base_extend(K) sage: T = EK(5, 5) sage: T.order() 5 sage: P = EK(-2, -1/2*t - 1/2) sage: P.order() +Infinity sage: EK.regulator_of_points([P,T]) # random very small output -1.23259516440783e-32 sage: EK.regulator_of_points([P,T]).abs() < 1e-30 True
>>> from sage.all import * >>> E = EllipticCurve('11a1') >>> x = polygen(QQ) >>> K = NumberField(x**Integer(2) + Integer(47), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> T = EK(Integer(5), Integer(5)) >>> T.order() 5 >>> P = EK(-Integer(2), -Integer(1)/Integer(2)*t - Integer(1)/Integer(2)) >>> P.order() +Infinity >>> EK.regulator_of_points([P,T]) # random very small output -1.23259516440783e-32 >>> EK.regulator_of_points([P,T]).abs() < RealNumber('1e-30') True
sage: E = EllipticCurve('389a1') sage: P,Q = E.gens() sage: E.regulator_of_points([P,Q]) 0.152460177943144 sage: K.<t> = NumberField(x^2 + 47) sage: EK = E.base_extend(K) sage: EK.regulator_of_points([EK(P),EK(Q)]) 0.152460177943144
>>> from sage.all import * >>> E = EllipticCurve('389a1') >>> P,Q = E.gens() >>> E.regulator_of_points([P,Q]) 0.152460177943144 >>> K = NumberField(x**Integer(2) + Integer(47), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.regulator_of_points([EK(P),EK(Q)]) 0.152460177943144
sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve([0,0,0,i,i]) sage: P = E(-9+4*i, -18-25*i) sage: Q = E(i, -i) sage: E.height_pairing_matrix([P,Q]) [ 2.16941934493768 -0.870059380421505] [-0.870059380421505 0.424585837470709] sage: E.regulator_of_points([P,Q]) 0.164101403936070
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0),Integer(0),Integer(0),i,i]) >>> P = E(-Integer(9)+Integer(4)*i, -Integer(18)-Integer(25)*i) >>> Q = E(i, -i) >>> E.height_pairing_matrix([P,Q]) [ 2.16941934493768 -0.870059380421505] [-0.870059380421505 0.424585837470709] >>> E.regulator_of_points([P,Q]) 0.164101403936070
- saturation(points, verbose=False, max_prime=0, one_prime=0, odd_primes_only=False, lower_ht_bound=None, reg=None, debug=False)[source]¶
Given a list of rational points on \(E\) over \(K\), compute the saturation in \(E(K)\) of the subgroup they generate.
INPUT:
points (list)
– list of points on E. Points of finite order are ignored; the remaining points should be independent, or an error is raised.verbose
– boolean (default:False
); ifTrue
, give verbose outputmax_prime
– integer (default: 0); saturation is performed for all primes up tomax_prime
. Ifmax_prime
is 0, perform saturation at all primes, i.e., compute the true saturation.odd_primes_only
– boolean (default:False
); only do saturation at odd primesone_prime
– integer (default: 0); if nonzero, only do saturation at this prime
The following two inputs are optional, and may be provided to speed up the computation.
lower_ht_bound
– real (default:None
); lower bound of the regulator \(E(K)\), if knownreg
– real (default:None
); regulator of the span of points, if knowndebug
– integer (default: 0); used for debugging and testing
OUTPUT:
saturation
– list; points that form a basis for the saturationindex
– integer; the index of the group generated by the input points in their saturationregulator
– real with default precision, orNone
; regulator of saturated points
EXAMPLES:
sage: K.<i> = QuadraticField(-1) sage: E = EllipticCurve('389a1') sage: EK = E.change_ring(K) sage: P = EK(-1,1); Q = EK(0,-1) sage: EK.saturation([2*P], max_prime=2) ([(-1 : 1 : 1)], 2, 0.686667083305587) sage: EK.saturation([12*P], max_prime=2) ([(26/361 : -5720/6859 : 1)], 4, 6.18000374975028) sage: EK.saturation([12*P], lower_ht_bound=0.1) ([(-1 : 1 : 1)], 12, 0.686667083305587) sage: EK.saturation([2*P, Q], max_prime=2) ([(-1 : 1 : 1), (0 : -1 : 1)], 2, 0.152460177943144) sage: EK.saturation([P + Q, P - Q], lower_ht_bound=.1, debug=2) ([(-1 : 1 : 1), (1 : 0 : 1)], 2, 0.152460177943144) sage: EK.saturation([P + Q, 17*Q], lower_ht_bound=0.1) # long time ([(4 : 8 : 1), (0 : -1 : 1)], 17, 0.152460177943143) sage: R = EK(i-2,-i-3) sage: EK.saturation([P + R, P + Q, Q + R], lower_ht_bound=0.1) ([(841/1369*i - 171/1369 : 41334/50653*i - 74525/50653 : 1), (4 : 8 : 1), (-1/25*i + 18/25 : -69/125*i - 58/125 : 1)], 2, 0.103174443217351) sage: EK.saturation([26*Q], lower_ht_bound=0.1) ([(0 : -1 : 1)], 26, 0.327000773651605)
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve('389a1') >>> EK = E.change_ring(K) >>> P = EK(-Integer(1),Integer(1)); Q = EK(Integer(0),-Integer(1)) >>> EK.saturation([Integer(2)*P], max_prime=Integer(2)) ([(-1 : 1 : 1)], 2, 0.686667083305587) >>> EK.saturation([Integer(12)*P], max_prime=Integer(2)) ([(26/361 : -5720/6859 : 1)], 4, 6.18000374975028) >>> EK.saturation([Integer(12)*P], lower_ht_bound=RealNumber('0.1')) ([(-1 : 1 : 1)], 12, 0.686667083305587) >>> EK.saturation([Integer(2)*P, Q], max_prime=Integer(2)) ([(-1 : 1 : 1), (0 : -1 : 1)], 2, 0.152460177943144) >>> EK.saturation([P + Q, P - Q], lower_ht_bound=RealNumber('.1'), debug=Integer(2)) ([(-1 : 1 : 1), (1 : 0 : 1)], 2, 0.152460177943144) >>> EK.saturation([P + Q, Integer(17)*Q], lower_ht_bound=RealNumber('0.1')) # long time ([(4 : 8 : 1), (0 : -1 : 1)], 17, 0.152460177943143) >>> R = EK(i-Integer(2),-i-Integer(3)) >>> EK.saturation([P + R, P + Q, Q + R], lower_ht_bound=RealNumber('0.1')) ([(841/1369*i - 171/1369 : 41334/50653*i - 74525/50653 : 1), (4 : 8 : 1), (-1/25*i + 18/25 : -69/125*i - 58/125 : 1)], 2, 0.103174443217351) >>> EK.saturation([Integer(26)*Q], lower_ht_bound=RealNumber('0.1')) ([(0 : -1 : 1)], 26, 0.327000773651605)
Another number field:
sage: E = EllipticCurve('389a1') sage: K.<a> = NumberField(x^3 - x + 1) sage: EK = E.change_ring(K) sage: P = EK(-1,1); Q = EK(0,-1) sage: EK.saturation([P + Q, P - Q], lower_ht_bound=0.1) ([(-1 : 1 : 1), (1 : 0 : 1)], 2, 0.152460177943144) sage: EK.saturation([3*P, P + 5*Q], lower_ht_bound=0.1) ([(-185/2209 : -119510/103823 : 1), (80041/34225 : -26714961/6331625 : 1)], 15, 0.152460177943144)
>>> from sage.all import * >>> E = EllipticCurve('389a1') >>> K = NumberField(x**Integer(3) - x + Integer(1), names=('a',)); (a,) = K._first_ngens(1) >>> EK = E.change_ring(K) >>> P = EK(-Integer(1),Integer(1)); Q = EK(Integer(0),-Integer(1)) >>> EK.saturation([P + Q, P - Q], lower_ht_bound=RealNumber('0.1')) ([(-1 : 1 : 1), (1 : 0 : 1)], 2, 0.152460177943144) >>> EK.saturation([Integer(3)*P, P + Integer(5)*Q], lower_ht_bound=RealNumber('0.1')) ([(-185/2209 : -119510/103823 : 1), (80041/34225 : -26714961/6331625 : 1)], 15, 0.152460177943144)
A different curve:
sage: K.<a> = QuadraticField(3) sage: E = EllipticCurve('37a1') sage: EK = E.change_ring(K) sage: P = EK(0,0); Q = EK(2-a, 2*a-4) sage: EK.saturation([3*P - Q, 3*P + Q], lower_ht_bound=.01) ([(0 : 0 : 1), (1/2*a : -1/4*a - 1/4 : 1)], 6, 0.0317814530725985)
>>> from sage.all import * >>> K = QuadraticField(Integer(3), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve('37a1') >>> EK = E.change_ring(K) >>> P = EK(Integer(0),Integer(0)); Q = EK(Integer(2)-a, Integer(2)*a-Integer(4)) >>> EK.saturation([Integer(3)*P - Q, Integer(3)*P + Q], lower_ht_bound=RealNumber('.01')) ([(0 : 0 : 1), (1/2*a : -1/4*a - 1/4 : 1)], 6, 0.0317814530725985)
The points must be linearly independent:
sage: EK.saturation([2*P, 3*Q, P-Q]) Traceback (most recent call last): ... ValueError: points not linearly independent in saturation()
>>> from sage.all import * >>> EK.saturation([Integer(2)*P, Integer(3)*Q, P-Q]) Traceback (most recent call last): ... ValueError: points not linearly independent in saturation()
Degenerate case:
sage: EK.saturation([]) ([], 1, 1.00000000000000)
>>> from sage.all import * >>> EK.saturation([]) ([], 1, 1.00000000000000)
ALGORITHM:
For rank 1 subgroups, simply do trial division up to the maximal prime divisor. For higher rank subgroups, perform trial division on all linear combinations for small primes, and look for projections \(E(K) \rightarrow \oplus E(k) \otimes \mathbf{F}_p\) which are either full rank or provide \(p\)-divisible linear combinations, where the \(k\) here are residue fields of \(K\).
- simon_two_descent(verbose=0, lim1=2, lim3=4, limtriv=2, maxprob=20, limbigprime=30, known_points=None)[source]¶
Return lower and upper bounds on the rank of the Mordell-Weil group \(E(K)\) and a list of points.
This method is used internally by the
rank()
,rank_bounds()
andgens()
methods.INPUT:
self
– an elliptic curve \(E\) over a number field \(K\)verbose
– 0, 1, 2, or 3 (default: 0); the verbosity levellim1
– (default: 2) limit on trivial points on quarticslim3
– (default: 4) limit on points on ELS quarticslimtriv
– (default: 2) limit on trivial points on \(E\)maxprob
– (default: 20)limbigprime
– (default: 30) to distinguish between small and large prime numbers. Use probabilistic tests for large primes. If 0, do not use probabilistic tests.known_points
– (default:None
) list of known points on the curve
OUTPUT: a triple
(lower, upper, list)
consisting oflower
– integer; lower bound on the rankupper
– integer; upper bound on the ranklist
– list of points in \(E(K)\)
The integer
upper
is in fact an upper bound on the dimension of the 2-Selmer group, hence on the dimension of \(E(K)/2E(K)\). It is equal to the dimension of the 2-Selmer group except possibly if \(E(K)[2]\) has dimension 1. In that case,upper
may exceed the dimension of the 2-Selmer group by an even number, due to the fact that the algorithm does not perform a second descent.Note
For non-quadratic number fields, this code does return, but it takes a long time.
ALGORITHM:
Uses Denis Simon’s PARI/GP scripts from https://simond.users.lmno.cnrs.fr/.
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 + 23, 'a') sage: E = EllipticCurve(K, '37') sage: E == loads(dumps(E)) True sage: E.simon_two_descent() (2, 2, [(0 : 0 : 1), (1/18*a + 7/18 : -5/54*a - 17/54 : 1)]) sage: E.simon_two_descent(lim1=5, lim3=5, limtriv=10, maxprob=7, limbigprime=10) (2, 2, [(-1 : 0 : 1), (-2 : -1/2*a - 1/2 : 1)])
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) + Integer(23), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, '37') >>> E == loads(dumps(E)) True >>> E.simon_two_descent() (2, 2, [(0 : 0 : 1), (1/18*a + 7/18 : -5/54*a - 17/54 : 1)]) >>> E.simon_two_descent(lim1=Integer(5), lim3=Integer(5), limtriv=Integer(10), maxprob=Integer(7), limbigprime=Integer(10)) (2, 2, [(-1 : 0 : 1), (-2 : -1/2*a - 1/2 : 1)])
sage: K.<a> = NumberField(x^2 + 7, 'a') sage: E = EllipticCurve(K, [0,0,0,1,a]); E Elliptic Curve defined by y^2 = x^3 + x + a over Number Field in a with defining polynomial x^2 + 7 sage: v = E.simon_two_descent(verbose=1); v elliptic curve: Y^2 = x^3 + Mod(1, y^2 + 7)*x + Mod(y, y^2 + 7) Trivial points on the curve = [[1, 1, 0], [Mod(1/2*y + 3/2, y^2 + 7), Mod(-y - 2, y^2 + 7), 1]] #S(E/K)[2] = 2 #E(K)/2E(K) = 2 #III(E/K)[2] = 1 rank(E/K) = 1 listpoints = [[Mod(1/2*y + 3/2, y^2 + 7), Mod(-y - 2, y^2 + 7), 1]] (1, 1, [(1/2*a + 3/2 : -a - 2 : 1)])
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(7), 'a', names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, [Integer(0),Integer(0),Integer(0),Integer(1),a]); E Elliptic Curve defined by y^2 = x^3 + x + a over Number Field in a with defining polynomial x^2 + 7 >>> v = E.simon_two_descent(verbose=Integer(1)); v elliptic curve: Y^2 = x^3 + Mod(1, y^2 + 7)*x + Mod(y, y^2 + 7) Trivial points on the curve = [[1, 1, 0], [Mod(1/2*y + 3/2, y^2 + 7), Mod(-y - 2, y^2 + 7), 1]] #S(E/K)[2] = 2 #E(K)/2E(K) = 2 #III(E/K)[2] = 1 rank(E/K) = 1 listpoints = [[Mod(1/2*y + 3/2, y^2 + 7), Mod(-y - 2, y^2 + 7), 1]] (1, 1, [(1/2*a + 3/2 : -a - 2 : 1)])
A curve with 2-torsion:
sage: K.<a> = NumberField(x^2 + 7) sage: E = EllipticCurve(K, '15a') sage: E.simon_two_descent() # long time (3s on sage.math, 2013), points can vary (1, 3, [...])
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(7), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve(K, '15a') >>> E.simon_two_descent() # long time (3s on sage.math, 2013), points can vary (1, 3, [...])
Check that the bug reported in Issue #15483 is fixed:
sage: K.<s> = QuadraticField(229) sage: c4 = 2173 - 235*(1 - s)/2 sage: c6 = -124369 + 15988*(1 - s)/2 sage: E = EllipticCurve([-c4/48, -c6/864]) sage: E.simon_two_descent() (0, 0, []) sage: R.<t> = QQ[] sage: L.<g> = NumberField(t^3 - 9*t^2 + 13*t - 4) sage: E1 = EllipticCurve(L, [1-g*(g-1), -g^2*(g-1), -g^2*(g-1), 0, 0]) sage: E1.rank() # long time (about 5 s) 0 sage: K = CyclotomicField(43).subfields(3)[0][0] sage: E = EllipticCurve(K, '37') sage: E.simon_two_descent() # long time (4s on sage.math, 2013) (3, 3, [(1/8*zeta43_0^2 - 3/8*zeta43_0 - 1/4 : -5/16*zeta43_0^2 + 7/16*zeta43_0 + 1/8 : 1), (0 : 0 : 1)])
>>> from sage.all import * >>> K = QuadraticField(Integer(229), names=('s',)); (s,) = K._first_ngens(1) >>> c4 = Integer(2173) - Integer(235)*(Integer(1) - s)/Integer(2) >>> c6 = -Integer(124369) + Integer(15988)*(Integer(1) - s)/Integer(2) >>> E = EllipticCurve([-c4/Integer(48), -c6/Integer(864)]) >>> E.simon_two_descent() (0, 0, []) >>> R = QQ['t']; (t,) = R._first_ngens(1) >>> L = NumberField(t**Integer(3) - Integer(9)*t**Integer(2) + Integer(13)*t - Integer(4), names=('g',)); (g,) = L._first_ngens(1) >>> E1 = EllipticCurve(L, [Integer(1)-g*(g-Integer(1)), -g**Integer(2)*(g-Integer(1)), -g**Integer(2)*(g-Integer(1)), Integer(0), Integer(0)]) >>> E1.rank() # long time (about 5 s) 0 >>> K = CyclotomicField(Integer(43)).subfields(Integer(3))[Integer(0)][Integer(0)] >>> E = EllipticCurve(K, '37') >>> E.simon_two_descent() # long time (4s on sage.math, 2013) (3, 3, [(1/8*zeta43_0^2 - 3/8*zeta43_0 - 1/4 : -5/16*zeta43_0^2 + 7/16*zeta43_0 + 1/8 : 1), (0 : 0 : 1)])
- tamagawa_exponent(P, proof=None)[source]¶
Return the Tamagawa index of this elliptic curve at the prime \(P\).
INPUT:
P
– eitherNone
or a prime ideal of the base field ofself
proof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.
OUTPUT: positive integer; the Tamagawa index of the curve at P
EXAMPLES:
sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve([20, 225, 750, 625*a + 6875, 31250*a + 46875]) sage: [E.tamagawa_exponent(P) for P in E.discriminant().support()] [1, 1, 1, 1] sage: K.<a> = QuadraticField(-11) sage: E = EllipticCurve('11a1').change_ring(K) sage: [E.tamagawa_exponent(P) for P in K(11).support()] [10]
>>> from sage.all import * >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(20), Integer(225), Integer(750), Integer(625)*a + Integer(6875), Integer(31250)*a + Integer(46875)]) >>> [E.tamagawa_exponent(P) for P in E.discriminant().support()] [1, 1, 1, 1] >>> K = QuadraticField(-Integer(11), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve('11a1').change_ring(K) >>> [E.tamagawa_exponent(P) for P in K(Integer(11)).support()] [10]
- tamagawa_number(P, proof=None)[source]¶
Return the Tamagawa number of this elliptic curve at the prime \(P\).
INPUT:
P
– eitherNone
or a prime ideal of the base field ofself
proof
– whether to only use provably correct methods (default controlled by global proof module). Note that the proof module is number_field, not elliptic_curves, since the functions that actually need the flag are in number fields.
OUTPUT: positive integer; the Tamagawa number of the curve at \(P\)
EXAMPLES:
sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 - 5) sage: E = EllipticCurve([20, 225, 750, 625*a + 6875, 31250*a + 46875]) sage: [E.tamagawa_number(P) for P in E.discriminant().support()] [1, 1, 1, 1] sage: K.<a> = QuadraticField(-11) sage: E = EllipticCurve('11a1').change_ring(K) sage: [E.tamagawa_number(P) for P in K(11).support()] [10]
>>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(20), Integer(225), Integer(750), Integer(625)*a + Integer(6875), Integer(31250)*a + Integer(46875)]) >>> [E.tamagawa_number(P) for P in E.discriminant().support()] [1, 1, 1, 1] >>> K = QuadraticField(-Integer(11), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve('11a1').change_ring(K) >>> [E.tamagawa_number(P) for P in K(Integer(11)).support()] [10]
- tamagawa_numbers()[source]¶
Return a list of all Tamagawa numbers for all prime divisors of the conductor (in order).
EXAMPLES:
sage: e = EllipticCurve('30a1') sage: e.tamagawa_numbers() [2, 3, 1] sage: vector(e.tamagawa_numbers()) (2, 3, 1) sage: K.<a> = NumberField(x^2 + 3) sage: eK = e.base_extend(K) sage: eK.tamagawa_numbers() [4, 6, 1]
>>> from sage.all import * >>> e = EllipticCurve('30a1') >>> e.tamagawa_numbers() [2, 3, 1] >>> vector(e.tamagawa_numbers()) (2, 3, 1) >>> K = NumberField(x**Integer(2) + Integer(3), names=('a',)); (a,) = K._first_ngens(1) >>> eK = e.base_extend(K) >>> eK.tamagawa_numbers() [4, 6, 1]
- tamagawa_product()[source]¶
Return the product of the Tamagawa numbers \(c_v\) where \(v\) runs over all prime ideals of \(K\).
Note
See also tamagawa_product_bsd(), which includes an additional factor when the model is not globally minimal, as required by the BSD formula.
OUTPUT: a positive integer
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([0, 2+i]) sage: E.tamagawa_product() 1 sage: E = EllipticCurve([(2*i+1)^2, i*(2*i+1)^7]) sage: E.tamagawa_product() 4
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(2)+i]) >>> E.tamagawa_product() 1 >>> E = EllipticCurve([(Integer(2)*i+Integer(1))**Integer(2), i*(Integer(2)*i+Integer(1))**Integer(7)]) >>> E.tamagawa_product() 4
An example over \(\QQ\):
sage: E = EllipticCurve('30a') sage: E.tamagawa_product() 6
>>> from sage.all import * >>> E = EllipticCurve('30a') >>> E.tamagawa_product() 6
An example with everywhere good reduction, where the product is empty:
sage: x = polygen(QQ) sage: K.<a> = NumberField(x^2 - 38) sage: E = EllipticCurve( [a, -a + 1, a + 1, -5*a + 15, -5*a + 21]) sage: E.tamagawa_numbers() [] sage: E.tamagawa_product() 1
>>> from sage.all import * >>> x = polygen(QQ) >>> K = NumberField(x**Integer(2) - Integer(38), names=('a',)); (a,) = K._first_ngens(1) >>> E = EllipticCurve( [a, -a + Integer(1), a + Integer(1), -Integer(5)*a + Integer(15), -Integer(5)*a + Integer(21)]) >>> E.tamagawa_numbers() [] >>> E.tamagawa_product() 1
- tamagawa_product_bsd()[source]¶
Given an elliptic curve \(E\) over a number field \(K\), this function returns the integer \(C(E/K)\) that appears in the Birch and Swinnerton-Dyer conjecture accounting for the local information at finite places. If the model is a global minimal model then \(C(E/K)\) is simply the product of the Tamagawa numbers \(c_v\) where \(v\) runs over all prime ideals of \(K\). Otherwise, if the model has to be changed at a place \(v\) a correction factor appears. The definition is such that \(C(E/K)\) times the periods at the infinite places is invariant under change of the Weierstrass model. See [Tate1966] and [DD2010] for details.
Note
This definition differs from the definition of
tamagawa_product
for curves defined over \(\QQ\). Over the rational number it is always defined to be the product of the Tamagawa numbers, so the two definitions only agree when the model is global minimal.OUTPUT: a rational number
EXAMPLES:
sage: K.<i> = NumberField(x^2 + 1) sage: E = EllipticCurve([0, 2+i]) sage: E.tamagawa_product_bsd() 1 sage: E = EllipticCurve([(2*i+1)^2, i*(2*i+1)^7]) sage: E.tamagawa_product_bsd() 4
>>> from sage.all import * >>> K = NumberField(x**Integer(2) + Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> E = EllipticCurve([Integer(0), Integer(2)+i]) >>> E.tamagawa_product_bsd() 1 >>> E = EllipticCurve([(Integer(2)*i+Integer(1))**Integer(2), i*(Integer(2)*i+Integer(1))**Integer(7)]) >>> E.tamagawa_product_bsd() 4
An example where the Neron model changes over K:
sage: K.<t> = NumberField(x^5 - 10*x^3 + 5*x^2 + 10*x + 1) sage: E = EllipticCurve(K, '75a1') sage: E.tamagawa_product_bsd() 5 sage: da = E.local_data() sage: [dav.tamagawa_number() for dav in da] [1, 1]
>>> from sage.all import * >>> K = NumberField(x**Integer(5) - Integer(10)*x**Integer(3) + Integer(5)*x**Integer(2) + Integer(10)*x + Integer(1), names=('t',)); (t,) = K._first_ngens(1) >>> E = EllipticCurve(K, '75a1') >>> E.tamagawa_product_bsd() 5 >>> da = E.local_data() >>> [dav.tamagawa_number() for dav in da] [1, 1]
An example over \(\QQ\) (Issue #9413):
sage: E = EllipticCurve('30a') sage: E.tamagawa_product_bsd() 6
>>> from sage.all import * >>> E = EllipticCurve('30a') >>> E.tamagawa_product_bsd() 6
- torsion_order()[source]¶
Return the order of the torsion subgroup of this elliptic curve.
OUTPUT: integer
EXAMPLES:
sage: E = EllipticCurve('11a1') sage: x = polygen(ZZ, 'x') sage: K.<t> = NumberField(x^4 + x^3 + 11*x^2 + 41*x + 101) sage: EK = E.base_extend(K) sage: EK.torsion_order() # long time (2s on sage.math, 2014) 25
>>> from sage.all import * >>> E = EllipticCurve('11a1') >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(4) + x**Integer(3) + Integer(11)*x**Integer(2) + Integer(41)*x + Integer(101), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_order() # long time (2s on sage.math, 2014) 25
sage: E = EllipticCurve('15a1') sage: K.<t> = NumberField(x^2 + 2*x + 10) sage: EK = E.base_extend(K) sage: EK.torsion_order() 16
>>> from sage.all import * >>> E = EllipticCurve('15a1') >>> K = NumberField(x**Integer(2) + Integer(2)*x + Integer(10), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_order() 16
sage: E = EllipticCurve('19a1') sage: K.<t> = NumberField(x^9 - 3*x^8 - 4*x^7 + 16*x^6 - 3*x^5 - 21*x^4 + 5*x^3 + 7*x^2 - 7*x + 1) sage: EK = E.base_extend(K) sage: EK.torsion_order() 9
>>> from sage.all import * >>> E = EllipticCurve('19a1') >>> K = NumberField(x**Integer(9) - Integer(3)*x**Integer(8) - Integer(4)*x**Integer(7) + Integer(16)*x**Integer(6) - Integer(3)*x**Integer(5) - Integer(21)*x**Integer(4) + Integer(5)*x**Integer(3) + Integer(7)*x**Integer(2) - Integer(7)*x + Integer(1), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_order() 9
sage: K.<i> = QuadraticField(-1) sage: EK = EllipticCurve([0, 0, 0, i, i + 3]) sage: EK.torsion_order() 1
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EK = EllipticCurve([Integer(0), Integer(0), Integer(0), i, i + Integer(3)]) >>> EK.torsion_order() 1
- torsion_points()[source]¶
Return a list of the torsion points of this elliptic curve.
OUTPUT: sorted list of the torsion points
EXAMPLES:
sage: E = EllipticCurve('11a1') sage: E.torsion_points() [(0 : 1 : 0), (5 : -6 : 1), (5 : 5 : 1), (16 : -61 : 1), (16 : 60 : 1)] sage: x = polygen(ZZ, 'x') sage: K.<t> = NumberField(x^4 + x^3 + 11*x^2 + 41*x + 101) sage: EK = E.base_extend(K) sage: EK.torsion_points() # long time (1s on sage.math, 2014) [(0 : 1 : 0), (t : 1/11*t^3 + 6/11*t^2 + 19/11*t + 48/11 : 1), (1/11*t^3 - 5/11*t^2 + 19/11*t - 40/11 : -6/11*t^3 - 3/11*t^2 - 26/11*t - 321/11 : 1), (1/11*t^3 - 5/11*t^2 + 19/11*t - 40/11 : 6/11*t^3 + 3/11*t^2 + 26/11*t + 310/11 : 1), (t : -1/11*t^3 - 6/11*t^2 - 19/11*t - 59/11 : 1), (16 : 60 : 1), (-3/55*t^3 - 7/55*t^2 - 2/55*t - 133/55 : 6/55*t^3 + 3/55*t^2 + 25/11*t + 156/55 : 1), (14/121*t^3 - 15/121*t^2 + 90/121*t + 232/121 : 16/121*t^3 - 69/121*t^2 + 293/121*t - 46/121 : 1), (-26/121*t^3 + 20/121*t^2 - 219/121*t - 995/121 : -15/121*t^3 - 156/121*t^2 + 232/121*t - 2887/121 : 1), (10/121*t^3 + 49/121*t^2 + 168/121*t + 73/121 : -32/121*t^3 - 60/121*t^2 + 261/121*t + 686/121 : 1), (5 : 5 : 1), (-9/121*t^3 - 21/121*t^2 - 127/121*t - 377/121 : -7/121*t^3 + 24/121*t^2 + 197/121*t + 16/121 : 1), (3/55*t^3 + 7/55*t^2 + 2/55*t + 78/55 : 7/55*t^3 - 24/55*t^2 + 9/11*t + 17/55 : 1), (-5/121*t^3 + 36/121*t^2 - 84/121*t + 24/121 : -34/121*t^3 + 27/121*t^2 - 305/121*t - 829/121 : 1), (5/121*t^3 - 14/121*t^2 - 158/121*t - 453/121 : 49/121*t^3 + 129/121*t^2 + 315/121*t + 86/121 : 1), (5 : -6 : 1), (5/121*t^3 - 14/121*t^2 - 158/121*t - 453/121 : -49/121*t^3 - 129/121*t^2 - 315/121*t - 207/121 : 1), (-5/121*t^3 + 36/121*t^2 - 84/121*t + 24/121 : 34/121*t^3 - 27/121*t^2 + 305/121*t + 708/121 : 1), (3/55*t^3 + 7/55*t^2 + 2/55*t + 78/55 : -7/55*t^3 + 24/55*t^2 - 9/11*t - 72/55 : 1), (-9/121*t^3 - 21/121*t^2 - 127/121*t - 377/121 : 7/121*t^3 - 24/121*t^2 - 197/121*t - 137/121 : 1), (16 : -61 : 1), (10/121*t^3 + 49/121*t^2 + 168/121*t + 73/121 : 32/121*t^3 + 60/121*t^2 - 261/121*t - 807/121 : 1), (-26/121*t^3 + 20/121*t^2 - 219/121*t - 995/121 : 15/121*t^3 + 156/121*t^2 - 232/121*t + 2766/121 : 1), (14/121*t^3 - 15/121*t^2 + 90/121*t + 232/121 : -16/121*t^3 + 69/121*t^2 - 293/121*t - 75/121 : 1), (-3/55*t^3 - 7/55*t^2 - 2/55*t - 133/55 : -6/55*t^3 - 3/55*t^2 - 25/11*t - 211/55 : 1)]
>>> from sage.all import * >>> E = EllipticCurve('11a1') >>> E.torsion_points() [(0 : 1 : 0), (5 : -6 : 1), (5 : 5 : 1), (16 : -61 : 1), (16 : 60 : 1)] >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(4) + x**Integer(3) + Integer(11)*x**Integer(2) + Integer(41)*x + Integer(101), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_points() # long time (1s on sage.math, 2014) [(0 : 1 : 0), (t : 1/11*t^3 + 6/11*t^2 + 19/11*t + 48/11 : 1), (1/11*t^3 - 5/11*t^2 + 19/11*t - 40/11 : -6/11*t^3 - 3/11*t^2 - 26/11*t - 321/11 : 1), (1/11*t^3 - 5/11*t^2 + 19/11*t - 40/11 : 6/11*t^3 + 3/11*t^2 + 26/11*t + 310/11 : 1), (t : -1/11*t^3 - 6/11*t^2 - 19/11*t - 59/11 : 1), (16 : 60 : 1), (-3/55*t^3 - 7/55*t^2 - 2/55*t - 133/55 : 6/55*t^3 + 3/55*t^2 + 25/11*t + 156/55 : 1), (14/121*t^3 - 15/121*t^2 + 90/121*t + 232/121 : 16/121*t^3 - 69/121*t^2 + 293/121*t - 46/121 : 1), (-26/121*t^3 + 20/121*t^2 - 219/121*t - 995/121 : -15/121*t^3 - 156/121*t^2 + 232/121*t - 2887/121 : 1), (10/121*t^3 + 49/121*t^2 + 168/121*t + 73/121 : -32/121*t^3 - 60/121*t^2 + 261/121*t + 686/121 : 1), (5 : 5 : 1), (-9/121*t^3 - 21/121*t^2 - 127/121*t - 377/121 : -7/121*t^3 + 24/121*t^2 + 197/121*t + 16/121 : 1), (3/55*t^3 + 7/55*t^2 + 2/55*t + 78/55 : 7/55*t^3 - 24/55*t^2 + 9/11*t + 17/55 : 1), (-5/121*t^3 + 36/121*t^2 - 84/121*t + 24/121 : -34/121*t^3 + 27/121*t^2 - 305/121*t - 829/121 : 1), (5/121*t^3 - 14/121*t^2 - 158/121*t - 453/121 : 49/121*t^3 + 129/121*t^2 + 315/121*t + 86/121 : 1), (5 : -6 : 1), (5/121*t^3 - 14/121*t^2 - 158/121*t - 453/121 : -49/121*t^3 - 129/121*t^2 - 315/121*t - 207/121 : 1), (-5/121*t^3 + 36/121*t^2 - 84/121*t + 24/121 : 34/121*t^3 - 27/121*t^2 + 305/121*t + 708/121 : 1), (3/55*t^3 + 7/55*t^2 + 2/55*t + 78/55 : -7/55*t^3 + 24/55*t^2 - 9/11*t - 72/55 : 1), (-9/121*t^3 - 21/121*t^2 - 127/121*t - 377/121 : 7/121*t^3 - 24/121*t^2 - 197/121*t - 137/121 : 1), (16 : -61 : 1), (10/121*t^3 + 49/121*t^2 + 168/121*t + 73/121 : 32/121*t^3 + 60/121*t^2 - 261/121*t - 807/121 : 1), (-26/121*t^3 + 20/121*t^2 - 219/121*t - 995/121 : 15/121*t^3 + 156/121*t^2 - 232/121*t + 2766/121 : 1), (14/121*t^3 - 15/121*t^2 + 90/121*t + 232/121 : -16/121*t^3 + 69/121*t^2 - 293/121*t - 75/121 : 1), (-3/55*t^3 - 7/55*t^2 - 2/55*t - 133/55 : -6/55*t^3 - 3/55*t^2 - 25/11*t - 211/55 : 1)]
sage: E = EllipticCurve('15a1') sage: K.<t> = NumberField(x^2 + 2*x + 10) sage: EK = E.base_extend(K) sage: EK.torsion_points() [(0 : 1 : 0), (-7 : -5*t - 2 : 1), (-7 : 5*t + 8 : 1), (-13/4 : 9/8 : 1), (-2 : -2 : 1), (-2 : 3 : 1), (-t - 2 : -t - 7 : 1), (-t - 2 : 2*t + 8 : 1), (-1 : 0 : 1), (t : t - 5 : 1), (t : -2*t + 4 : 1), (1/2 : -5/4*t - 2 : 1), (1/2 : 5/4*t + 1/2 : 1), (3 : -2 : 1), (8 : -27 : 1), (8 : 18 : 1)]
>>> from sage.all import * >>> E = EllipticCurve('15a1') >>> K = NumberField(x**Integer(2) + Integer(2)*x + Integer(10), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_points() [(0 : 1 : 0), (-7 : -5*t - 2 : 1), (-7 : 5*t + 8 : 1), (-13/4 : 9/8 : 1), (-2 : -2 : 1), (-2 : 3 : 1), (-t - 2 : -t - 7 : 1), (-t - 2 : 2*t + 8 : 1), (-1 : 0 : 1), (t : t - 5 : 1), (t : -2*t + 4 : 1), (1/2 : -5/4*t - 2 : 1), (1/2 : 5/4*t + 1/2 : 1), (3 : -2 : 1), (8 : -27 : 1), (8 : 18 : 1)]
sage: K.<i> = QuadraticField(-1) sage: EK = EllipticCurve(K, [0,0,0,0,-1]) sage: EK.torsion_points() [(0 : 1 : 0), (-2 : -3*i : 1), (-2 : 3*i : 1), (0 : -i : 1), (0 : i : 1), (1 : 0 : 1)]
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EK = EllipticCurve(K, [Integer(0),Integer(0),Integer(0),Integer(0),-Integer(1)]) >>> EK.torsion_points() [(0 : 1 : 0), (-2 : -3*i : 1), (-2 : 3*i : 1), (0 : -i : 1), (0 : i : 1), (1 : 0 : 1)]
- torsion_subgroup()[source]¶
Return the torsion subgroup of this elliptic curve.
OUTPUT: the
EllipticCurveTorsionSubgroup
associated to this elliptic curve.EXAMPLES:
sage: E = EllipticCurve('11a1') sage: x = polygen(ZZ, 'x') sage: K.<t> = NumberField(x^4 + x^3 + 11*x^2 + 41*x + 101) sage: EK = E.base_extend(K) sage: tor = EK.torsion_subgroup() # long time (2s on sage.math, 2014) sage: tor # long time Torsion Subgroup isomorphic to Z/5 + Z/5 associated to the Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 + (-10)*x + (-20) over Number Field in t with defining polynomial x^4 + x^3 + 11*x^2 + 41*x + 101 sage: tor.gens() # long time ((16 : 60 : 1), (t : 1/11*t^3 + 6/11*t^2 + 19/11*t + 48/11 : 1))
>>> from sage.all import * >>> E = EllipticCurve('11a1') >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(4) + x**Integer(3) + Integer(11)*x**Integer(2) + Integer(41)*x + Integer(101), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> tor = EK.torsion_subgroup() # long time (2s on sage.math, 2014) >>> tor # long time Torsion Subgroup isomorphic to Z/5 + Z/5 associated to the Elliptic Curve defined by y^2 + y = x^3 + (-1)*x^2 + (-10)*x + (-20) over Number Field in t with defining polynomial x^4 + x^3 + 11*x^2 + 41*x + 101 >>> tor.gens() # long time ((16 : 60 : 1), (t : 1/11*t^3 + 6/11*t^2 + 19/11*t + 48/11 : 1))
sage: E = EllipticCurve('15a1') sage: K.<t> = NumberField(x^2 + 2*x + 10) sage: EK = E.base_extend(K) sage: EK.torsion_subgroup() Torsion Subgroup isomorphic to Z/4 + Z/4 associated to the Elliptic Curve defined by y^2 + x*y + y = x^3 + x^2 + (-10)*x + (-10) over Number Field in t with defining polynomial x^2 + 2*x + 10
>>> from sage.all import * >>> E = EllipticCurve('15a1') >>> K = NumberField(x**Integer(2) + Integer(2)*x + Integer(10), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_subgroup() Torsion Subgroup isomorphic to Z/4 + Z/4 associated to the Elliptic Curve defined by y^2 + x*y + y = x^3 + x^2 + (-10)*x + (-10) over Number Field in t with defining polynomial x^2 + 2*x + 10
sage: E = EllipticCurve('19a1') sage: K.<t> = NumberField(x^9-3*x^8-4*x^7+16*x^6-3*x^5-21*x^4+5*x^3+7*x^2-7*x+1) sage: EK = E.base_extend(K) sage: EK.torsion_subgroup() Torsion Subgroup isomorphic to Z/9 associated to the Elliptic Curve defined by y^2 + y = x^3 + x^2 + (-9)*x + (-15) over Number Field in t with defining polynomial x^9 - 3*x^8 - 4*x^7 + 16*x^6 - 3*x^5 - 21*x^4 + 5*x^3 + 7*x^2 - 7*x + 1
>>> from sage.all import * >>> E = EllipticCurve('19a1') >>> K = NumberField(x**Integer(9)-Integer(3)*x**Integer(8)-Integer(4)*x**Integer(7)+Integer(16)*x**Integer(6)-Integer(3)*x**Integer(5)-Integer(21)*x**Integer(4)+Integer(5)*x**Integer(3)+Integer(7)*x**Integer(2)-Integer(7)*x+Integer(1), names=('t',)); (t,) = K._first_ngens(1) >>> EK = E.base_extend(K) >>> EK.torsion_subgroup() Torsion Subgroup isomorphic to Z/9 associated to the Elliptic Curve defined by y^2 + y = x^3 + x^2 + (-9)*x + (-15) over Number Field in t with defining polynomial x^9 - 3*x^8 - 4*x^7 + 16*x^6 - 3*x^5 - 21*x^4 + 5*x^3 + 7*x^2 - 7*x + 1
sage: K.<i> = QuadraticField(-1) sage: EK = EllipticCurve([0, 0, 0, i, i+3]) sage: EK.torsion_subgroup () Torsion Subgroup isomorphic to Trivial group associated to the Elliptic Curve defined by y^2 = x^3 + i*x + (i+3) over Number Field in i with defining polynomial x^2 + 1 with i = 1*I
>>> from sage.all import * >>> K = QuadraticField(-Integer(1), names=('i',)); (i,) = K._first_ngens(1) >>> EK = EllipticCurve([Integer(0), Integer(0), Integer(0), i, i+Integer(3)]) >>> EK.torsion_subgroup () Torsion Subgroup isomorphic to Trivial group associated to the Elliptic Curve defined by y^2 = x^3 + i*x + (i+3) over Number Field in i with defining polynomial x^2 + 1 with i = 1*I
See also
Use
division_field()
to determine the field of definition of the \(\ell\)-torsion subgroup.