-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathqq_quat_int.c
748 lines (703 loc) · 33.7 KB
/
qq_quat_int.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
//Intersections in quaternion algebras over Q
#include <pari.h>
#include "qquadraticdecl.h"
//STATIC DECLARATIONS
static GEN qa_inum_roots_f1bds(GEN r1, GEN r2, GEN n2, GEN coef);
static GEN qa_inum_roots_ghsearch(GEN Q, GEN ordinv, GEN rta, GEN maxd3d4, GEN D1, GEN rtD1, GEN e, GEN f, GEN res, GEN r1, GEN r2, GEN n2, int orient);
static int sides_int_indices(long i1, long i2, long j1, long j2);
//INTERSECTION NUMBER BASED OFF OF ROOTS BOUND AREA
//Must have ei being the image if (p_Di+sqrt(Di))/2.
GEN qa_inum_roots(GEN Q, GEN order, GEN e1, GEN e2, GEN D1, GEN D2, int data, long prec){
pari_sp top=avma, mid;
GEN abvec=qa_getabvec(Q), pel1=pell(D1), pel2=pell(D2);
int isswapped=0;
if(cmpii(gel(pel1, 1), gel(pel2, 1))==-1){//Swapping so pell(D2)<pell(D1); this makes it MUCH faster when there is a large discrepancy
isswapped=1;
GEN temp=e1;e1=e2;e2=temp;
temp=D1;D1=D2;D2=temp;
temp=pel2;pel2=pel1;pel1=temp;
}
GEN e2pell=cgetg(5, t_VEC);
gel(e2pell, 1)=gdivgs(gel(pel2, 1), 2);//T/2
for(int i=2;i<=4;i++) gel(e2pell, i)=gmul(gel(pel2, 2), gel(e2, i));//U*e2[i]. Now e2pell is the image of epsilon_D2 under e2
GEN e2roots=qa_roots(Q, e2pell, prec);
GEN e2sl2r=qa_m2rembed(Q, e2pell);
//We now need to pick an n1 in R union oo, and find all root geodesics of equivalent embeddings to e1 that intersect ell_{e2roots} (rt geod for e2pell) and ell_{n1, n2=e2sl2r(n1)}. If [r1, r2]=[0, oo], we choose [n1, n2]=[1, bigger]. Otherwise, assume r1<r2 and we find the n1 so that n1<r1<r2<n2 and (r1-n1)+(n2-r2) is maximized (this gives us the most efficient search later on).
GEN r1, r2, n1, n2, mat, matinv;
if(gcmp(gel(e2roots, 1), gel(e2roots, 2))==-1){r1=gel(e2roots, 1);r2=gel(e2roots, 2);mat=ginv(e2sl2r);matinv=e2sl2r;}
else{r2=gel(e2roots, 1);r1=gel(e2roots, 2);mat=e2sl2r;matinv=ginv(e2sl2r);}//r1<r2 are the roots of e2pell, mat is either e2sl2r or its inverse. We take n1<r1, and then mat(n1)>r2.
if(typ(r2)==t_INFINITY){//r2=oo, so r1=0 necessarily.
n1=gen_1;
n2=mat_eval(mat, n1);
}
else{//Finite
n1=gsub(mat_eval(matinv, mkoo()), gdivsg(1, gcoeff(mat, 2, 1)));//gcoeff(mat, 2, 1)>0 necessarily
n2=mat_eval(mat, n1);
}
//Great, now n1<r1<r2<n2 or [r1,r2]=[0,oo] and [n1,n2]=[1,something bigger]
//We are now searching for W=[e,f,g,h] in Gamma, conjugate to e1, for which the roots of W intersect ell_(r1,r2) and ell_(n1,n2). First, e=e1[1] necessarily, and it turns out there are finitely many possibilities with just this restriction, as well as assuming [e,f,g,h] is in Z/d[1]+Z/d[2]i+Z/d[3]j+Z/d[4]k. We will find this set, check which are actually in Gamma, and check for conjugation with e1 afterwards.
GEN maxdens=qa_getordmaxd(order), ord=qa_getord(order), ordinv=qa_getordinv(order);
GEN maxd34=lcmii(gel(maxdens, 3), gel(maxdens, 4));
GEN minf1, maxf1, rtD1=gsqrt(D1, prec), rta=gsqrt(gel(abvec, 1), prec), f, f1, poss, coef1, coef2, coef3, d3d4sq=Qdivii(sqri(maxd34), gel(abvec, 2));
coef1=gmul(d3d4sq, gdivgs(D1, 4));
coef2=gmul(d3d4sq, gneg(gel(abvec, 1)));//res=coef1+coef2*f^2=maxd3d4^2(D1/4-af^2)/b
coef3=gmul(gel(maxdens, 2), gsqrt(gdiv(D1, shifti(gel(abvec, 1), 2)), prec));//coef3=maxdens[2]*sqrt(D1/4a); passed into f1bounds method.
glist *S=NULL;
GEN partials=cgetg(1, t_VEC), partialstemp;
long nint=0;
if(typ(r2)==t_INFINITY){
minf1=gceil(gneg(coef3));
maxf1=gneg(minf1);//Roots separate ell_{0, oo}
mid=avma;
for(GEN f1=minf1;cmpii(f1, maxf1)!=1;f1=addis(f1, 1)){
f=gdiv(f1, gel(maxdens, 2));
poss=qa_inum_roots_ghsearch(Q, ordinv, rta, maxd34, D1, rtD1, gel(e1, 1), f, gadd(coef1, gmul(coef2, gsqr(f))), n1, r2, n2, 1);
for(long i=1;i<lg(poss);i++){
if(gequal0(qa_conjnorm(Q, ord, ordinv, gel(poss, i), e1, gen_1, 0, prec))) continue;//Nope, not conjugate
glist_putstart(&S, gel(poss, i));
nint++;
}
if(gc_needed(mid, 3)){
long j=lg(partials);
partialstemp=cgetg(j+nint, t_VEC);
for(long i=1;i<j;i++) gel(partialstemp, i)=gcopy(gel(partials, i));
partials=glist_togvec(S, nint, -1);
for(long i=1;i<=nint;i++){gel(partialstemp, j)=gel(partials, i);j++;}
partials=partialstemp;
gerepileall(mid, 2, &f1, &partials);
S=NULL;//Reset S
nint=0;//Reset nint
}
}
}
else{
//Case 1: bigger root is >n2
GEN fbds=qa_inum_roots_f1bds(r1, r2, n2, coef3);
minf1=gel(fbds, 1);maxf1=gel(fbds, 2);
mid=avma;
for(GEN f1=minf1;cmpii(f1, maxf1)!=1;f1=addis(f1, 1)){
f=gdiv(f1, gel(maxdens, 2));
poss=qa_inum_roots_ghsearch(Q, ordinv, rta, maxd34, D1, rtD1, gel(e1, 1), f, gadd(coef1, gmul(coef2, gsqr(f))), r1, r2, n2, 1);
for(long i=1;i<lg(poss);i++){
if(gequal0(qa_conjnorm(Q, ord, ordinv, gel(poss, i), e1, gen_1, 0, prec))) continue;//Nope, not conjugate
glist_putstart(&S, gel(poss, i));
nint++;
}
if(gc_needed(mid, 3)){
long j=lg(partials);
partialstemp=cgetg(j+nint, t_VEC);
for(long i=1;i<j;i++) gel(partialstemp, i)=gcopy(gel(partials, i));
partials=glist_togvec(S, nint, -1);
for(long i=1;i<=nint;i++){gel(partialstemp, j)=gel(partials, i);j++;}
partials=partialstemp;
gerepileall(mid, 2, &f1, &partials);
S=NULL;//Reset S
nint=0;//Reset nint
}
}
//Case 2: smaller root is <n1. We copy the previous code, replacing f with -f (also exchanging minf1 and maxf1), negating n1 and r1, r2
fbds=qa_inum_roots_f1bds(gneg(r2), gneg(r1), gneg(n1), coef3);
maxf1=gneg(gel(fbds, 1));minf1=gneg(gel(fbds, 2));
mid=avma;
for(GEN f1=minf1;cmpii(f1, maxf1)!=1;f1=addis(f1, 1)){
f=gdiv(f1, gel(maxdens, 2));
poss=qa_inum_roots_ghsearch(Q, ordinv, rta, maxd34, D1, rtD1, gel(e1, 1), gneg(f), gadd(coef1, gmul(coef2, gsqr(f))), gneg(r2), gneg(r1), gneg(n1), -1);
for(long i=1;i<lg(poss);i++){
if(gequal0(qa_conjnorm(Q, ord, ordinv, gel(poss, i), e1, gen_1, 0, prec))) continue;//Nope, not conjugate
glist_putstart(&S, gel(poss, i));
nint++;
}
if(gc_needed(mid, 3)){
long j=lg(partials);
partialstemp=cgetg(j+nint, t_VEC);
for(long i=1;i<j;i++) gel(partialstemp, i)=gcopy(gel(partials, i));
partials=glist_togvec(S, nint, -1);
for(long i=1;i<=nint;i++){gel(partialstemp, j)=gel(partials, i);j++;}
partials=partialstemp;
gerepileall(mid, 2, &f1, &partials);
S=NULL;//Reset S
nint=0;//Reset nint
}
}
//Case 3: One root is infinite, the other 0. Then e^2-af^2=nm, and we use the f1 variable to mean f.
if(gsigne(r1)==-1 && gsigne(r2)==1){
f1=Qdivii(D1, shifti(gel(abvec, 1), 2));//D1/4a
GEN f1root;
if(Q_issquareall(f1, &f1root)){
GEN emb=mkvec4(gel(e1, 1), f1root, gen_0, gen_0);
if(!gequal0(qa_conjnorm(Q, ord, ordinv, emb, e1, gen_1, 0, prec))){glist_putstart(&S, emb);nint++;}
emb=qa_conj(emb);
if(!gequal0(qa_conjnorm(Q, ord, ordinv, emb, e1, gen_1, 0, prec))){glist_putstart(&S, emb);nint++;}
}
}
}
GEN e1set=glist_togvec(S, nint, -1);
long j=lg(partials);
GEN pairs=cgetg(nint+j, t_VEC);
if(!isswapped){
for(long i=1;i<j;i++){
gel(pairs, i)=cgetg(3, t_VEC);
gel(gel(pairs, i), 1)=gcopy(gel(partials, i));
gel(gel(pairs, i), 2)=gcopy(e2);
}
for(long i=1;i<=nint;i++){
gel(pairs, j)=cgetg(3, t_VEC);
gel(gel(pairs, j), 1)=gcopy(gel(e1set, i));
gel(gel(pairs, j), 2)=gcopy(e2);
j++;
}
}
else{
for(long i=1;i<j;i++){
gel(pairs, i)=cgetg(3, t_VEC);
gel(gel(pairs, i), 1)=gcopy(e2);
gel(gel(pairs, i), 2)=gcopy(gel(partials, i));
}
for(long i=1;i<=nint;i++){
gel(pairs, j)=cgetg(3, t_VEC);
gel(gel(pairs, j), 1)=gcopy(e2);
gel(gel(pairs, j), 2)=gcopy(gel(e1set, i));
j++;
}
}
if(data==0) return gerepileupto(top, pairs);
//Now we care about the data.
GEN D1D2=mulii(D1, D2);
GEN ret=cgetg(3, t_VEC);
gel(ret, 1)=gcopy(pairs);
long lx;
gel(ret, 2)=cgetg_copy(pairs, &lx);
for(long i=1;i<lx;i++){
gel(gel(ret, 2), i)=qa_intlevel(Q, order, gel(gel(pairs, i), 1), gel(gel(pairs, i), 2), D1D2, prec);
}
return gerepileupto(top, ret);
}
//qa_inum_roots with typechecking
GEN qa_inum_roots_tc(GEN Q, GEN order, GEN e1, GEN e2, int data, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
qa_ordeichlercheck(order);
qa_eltcheck(e1);qa_eltcheck(e2);
GEN v1=qa_associatedemb(Q, order, e1, gen_0), v2=qa_associatedemb(Q, order, e2, gen_0);
return gerepileupto(top, qa_inum_roots(Q, order, gel(v1, 1), gel(v2, 1), gel(v1, 2), gel(v2, 2), data, prec));
}
//coef=maxdens[2]*sqrt(D1/4a)
//This generates the bounds for f1 based off of n1<r1< R1 <r2<n2 < R2, where the roots we search for are R1 and R2. Note that n1 is not needed in the input.
static GEN qa_inum_roots_f1bds(GEN r1, GEN r2, GEN n2, GEN coef){
pari_sp top=avma;
GEN minf1, maxf1;
if(gsigne(n2)==1){
if(gsigne(r2)==1){
maxf1=gfloor(gmul(coef, gdiv(gadd(r2, n2), gsub(n2, r2))));//Combining R2>n2 and R1<r2
if(gsigne(r1)==1) minf1=gceil(coef);//Use R1>r1>0
else minf1=gceil(gneg(coef));//Use R2>n2>0
}
else{
maxf1=gfloor(coef);//R1<r2<0
minf1=gceil(gneg(coef));//R2>n2>0
}
}
else{
maxf1=gfloor(coef);//R1<r2<0
minf1=gceil(gmul(coef, gdiv(gadd(n2, r2), gsub(n2, r2))));//R1<r2 and R2>n2
}
GEN ret=cgetg(3, t_VEC);
gel(ret, 1)=gcopy(minf1);
gel(ret, 2)=gcopy(maxf1);
return gerepileupto(top, ret);
}
//res=maxd3d4^2(D1/4-af^2)/b
//Starting with e, f, D1, r1, r2, n1, n2, we derive bounds on g, h and find the finite set of possible pairs(g, h) making embeddings [e, f, g, h] of discriminant D1 that intersect both root geodesics ell_{r1, r2} and ell_{n1, n2}. maxd3d4 is the lcm of qa_getordmaxd[3], [4]. Note that n1 is not inputted, it is not needed. Through the various cases we use this, sometimes we use r1 as n1, permute the inputs, etc. orient=1 if one root is in (r1, r2) and the other is >n2, and is 0 if the other is <n1. res=maxd3d4^2(D1/4-af^2)/b. Note that we are only finding finite rooted solutions here, must search for g=h=0 elsewhere.
static GEN qa_inum_roots_ghsearch(GEN Q, GEN ordinv, GEN rta, GEN maxd3d4, GEN D1, GEN rtD1, GEN e, GEN f, GEN res, GEN r1, GEN r2, GEN n2, int orient){
pari_sp top=avma;
if(gequal0(res) || typ(res)!=t_INT) return cgetg(1, t_VEC);
GEN forient, a=qa_geta(Q);
if(orient==1) forient=f;//At the end, if orient=-1 we are checking with -f not f
else forient=gneg(f);
GEN gmhmax, gmhmin;//We use the root bounds with the standard embedding to get bounds on g-hsqrt(a)
if(typ(r2)==t_INFINITY){
GEN ghnumer=gadd(gmul(f, rta), gdivgs(rtD1, 2));
gmhmax=gdiv(ghnumer, r1);
gmhmin=gdiv(ghnumer, n2);
}
else{
GEN ghnumer=gsub(gmul(f, rta), gdivgs(rtD1, 2));
gmhmin=gen_0, gmhmax=mkoo();
if(gsigne(n2)==1) gmhmax=gdiv(gadd(ghnumer, rtD1), n2);
else gmhmin=gmax_shallow(gdiv(gadd(ghnumer, rtD1), n2), gen_0);
if(gsigne(r1)==1) gmhmax=gmin_shallow(gdiv(ghnumer, r1), gmhmax);
else gmhmin=gmax_shallow(gdiv(ghnumer, r1), gmhmin);
if(gsigne(r2)==-1) gmhmax=gmin_shallow(gdiv(ghnumer, r2), gmhmax);
else gmhmin=gmax_shallow(gdiv(ghnumer, r2), gmhmin);
}
gmhmin=gmul(maxd3d4, gmhmin);//Scaling so that we get bounds on integers (maxd3d4*[g, h] is an integer vector necessarily.
gmhmax=gmul(maxd3d4, gmhmax);
if(gcmp(gmhmin, gmhmax)==1){avma=top;return cgetg(1, t_VEC);}//min>max
//The geodesic condition is <==> gmhmin<g1-h1*sqrt(a)gphmax (for g-hsqrt(a)>0).
//From (g-hsqrt(a))(g+hsqrt(a))=res, we get bounds for g+hsqrt(a)
GEN gphmin, gphmax;//would be more accurate to name g1ph1min/max since the bounds we get everywhere are for g1 and h1 not h and h
if(gsigne(res)==1){//res>0
gphmin=gdiv(res, gmhmax);
gphmax=gdiv(res, gmhmin);
}
else{//res<0
gphmin=gdiv(res, gmhmin);
gphmax=gdiv(res, gmhmax);
}
GEN g1min=gceil(gdivgs(gadd(gmhmin, gphmin), 2));//Bounds for g1
GEN g1max=gfloor(gdivgs(gadd(gmhmax, gphmax), 2));
GEN h1sq, h1abs, g, h, gmh, emb;
glist *S=NULL;
long nfound=0;
for(GEN g1=g1min;cmpii(g1, g1max)!=1;g1=addis(g1, 1)){
h1sq=Qdivii(subii(sqri(g1), res), a);
if(typ(h1sq)!=t_INT) continue;//Not integer
if(!Z_issquareall(h1sq, &h1abs)) continue;//Not square
g=Qdivii(g1, maxd3d4);
h=Qdivii(h1abs, maxd3d4);
gmh=gsub(g1, gmul(h1abs, rta));//remember, bounds are for g1 and h1 not g and h
if(gcmp(gmhmin, gmh)==-1 && gcmp(gmh, gmhmax)==-1){//Yes!
emb=mkvec4(e, forient, g, h);
if(qa_isinorder(Q, ordinv, emb)){glist_putstart(&S, emb);glist_putstart(&S, qa_conj(emb));nfound=nfound+2;}//Must add conjugate due to assuming g-hsqrt(a)>0
}
if(gequal0(h)) continue;//Now we do (g, -h), but of course only if h!=0
h=gneg(h);
gmh=gadd(g1, gmul(h1abs, rta));
if(gcmp(gmhmin, gmh)==-1 && gcmp(gmh, gmhmax)==-1){//Yes!
emb=mkvec4(e, forient, g, h);
if(qa_isinorder(Q, ordinv, emb)){glist_putstart(&S, emb);glist_putstart(&S, qa_conj(emb));nfound=nfound+2;}//Must add conjugate due to assuming g-hsqrt(a)>0
}
}
return gerepileupto(top, glist_togvec(S, nfound, -1));
}
//INTERSECTIONS BASED ON X-LINKAGE
//Intersection number based on x-linking. The output is [[pairs],[signed levels]], with pairs corresponding to respective levels."
GEN qa_inum_x(GEN Q, GEN order, GEN e1, GEN e2, GEN D1, GEN D2, int data, long prec){
pari_sp top=avma;
GEN xposs=qa_xposs(qa_getpram(Q), qa_getpramprod(Q), D1, D2, gen_0, gen_0);//The possible x's
long lgxposs=lg(xposs), nx=2*lgxposs-1;//Possible x's; double since we account for -x
if(lgxposs==1){avma=top;GEN retno=cgetg(3, t_VEC);gel(retno, 1)=cgetg(1, t_VEC);gel(retno, 2)=cgetg(1, t_VEC);return retno;}//No solutions
GEN pairs=cgetg(nx, t_VEC);
long i=1, j=1, npairs=1;
if(gequal0(gel(xposs, 1))){
gel(pairs, 1)=qa_xlink(Q, order, e1, e2, D1, D2, gen_0, prec);
i=2;j=2;
npairs=npairs+lg(gel(pairs, 1))-1;
}
for(;i<lgxposs;i++){
gel(pairs, j)=qa_xlink(Q, order, e1, e2, D1, D2, gel(xposs, i), prec);//x
npairs=npairs+lg(gel(pairs, j))-1;j++;
gel(pairs, j)=qa_xlink(Q, order, e1, e2, D1, D2, negi(gel(xposs, i)), prec);//-x
npairs=npairs+lg(gel(pairs, j))-1;j++;
}
if(data==0){
GEN allpairs=cgetg(npairs, t_VEC);
long pos=1;
for(i=1;i<j;i++){
for(long k=1;k<lg(gel(pairs, i));k++){gel(allpairs, pos)=gcopy(gel(gel(pairs, i), k));pos++;}
}
return gerepileupto(top, allpairs);
}
//Now we care about the data.
GEN D1D2=mulii(D1, D2);
GEN ret=cgetg(3, t_VEC);
gel(ret, 1)=cgetg(npairs, t_VEC);
gel(ret, 2)=cgetg(npairs, t_VEC);
long pos=1;
for(i=1;i<j;i++){
for(long k=1;k<lg(gel(pairs, i));k++){gel(gel(ret, 1), pos)=gcopy(gel(gel(pairs, i), k));pos++;}
}
for(i=1;i<npairs;i++){
gel(gel(ret, 2), i)=qa_intlevel(Q, order, gel(gel(gel(ret, 1), i), 1), gel(gel(gel(ret, 1), i), 2), D1D2, prec);
}
return gerepileupto(top, ret);
}
//qa_inum_x with typecheck
GEN qa_inum_x_tc(GEN Q, GEN order, GEN e1, GEN e2, int data, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
qa_ordeichlercheck(order);
qa_eltcheck(e1);qa_eltcheck(e2);
GEN v1=qa_associatedemb(Q, order, e1, gen_0), v2=qa_associatedemb(Q, order, e2, gen_0);
return gerepileupto(top, qa_inum_x(Q, order, gel(v1, 1), gel(v2, 1), gel(v1, 2), gel(v2, 2), data, prec));
}
//Finds all x-linkage of e1 with e2.
GEN qa_xlink(GEN Q, GEN order, GEN e1, GEN e2, GEN D1, GEN D2, GEN x, long prec){
pari_sp top=avma;
GEN zb=qa_getordtrace0basis(order), ord=qa_getord(order), ordinv=qa_getordinv(order);
GEN v1=gmulgs(e1, 2);gel(v1, 1)=gen_0;//v1 represents sqrt(D1) under phi1
GEN quadf=cgetg(7, t_VEC);//The embedding of D2 is given by emb=A*zb[1]+B*zb[2]+C*zb[3]. solving emb^2=D2 gives us the quadratic form quadf.
gel(quadf, 1)=negi(qa_norm(Q, gel(zb, 1)));//A^2 coeff
gel(quadf, 2)=negi(qa_norm(Q, gel(zb, 2)));//B^2 coeff
gel(quadf, 3)=negi(qa_norm(Q, gel(zb, 3)));//C^2 coeff
gel(quadf, 4)=qa_trace(qa_mul(Q, gel(zb, 1), gel(zb, 2)));//AB coeff
gel(quadf, 5)=qa_trace(qa_mul(Q, gel(zb, 1), gel(zb, 3)));//AC coeff
gel(quadf, 6)=qa_trace(qa_mul(Q, gel(zb, 2), gel(zb, 3)));//BC coeff
GEN lin=cgetg(4, t_VEC);//The x-linking gives the equation (v1*emb)[1]=x, and solving this gives the linear form lin.
gel(lin, 1)=gel(qa_mul(Q, v1, gel(zb, 1)), 1);//A coeff
gel(lin, 2)=gel(qa_mul(Q, v1, gel(zb, 2)), 1);//B coeff
gel(lin, 3)=gel(qa_mul(Q, v1, gel(zb, 3)), 1);//C coeff
GEN ABCs=bqf_linearsolve(quadf, D2, lin, x, prec);//Solving for A, B, C
if(gequal0(ABCs)){avma=top;return cgetg(1, t_VEC);}//No solutions
if(!equali1(gel(gel(ABCs, 1), 1))) pari_err_TYPE("Solutions are of an unexpected shape, please report!", ABCs);//The return should be type 1, unless D_1D_2-x^2=0, and we do NOT consider that case.
//Now ABCs=[[1, M, [s1, s2, s3]], [a1, b1, c1], [a2, b2, c2], ...], where a general solution is [A;B;C]=M^k*[ai;bi;ci]+[s1;s2;s3]
GEN scol=gtocol(gel(gel(ABCs, 1), 3)), M=gel(gel(ABCs, 1), 2);//s in column format, M
GEN sshift=gsub(scol, gmul(M, scol));//scol-M*scol; to go from k->k+1, multiply the solution by M and add sshift
GEN base, embofD2, baseemb=zerovec(4), emb=zerovec(4);
if(smodis(D2, 2)==1){gel(baseemb, 1)=ghalf;gel(emb, 1)=ghalf;}//Emb represents the embedding, and baseemb the base embedding
glist *S=NULL;//List of the embeddings
long count=0;
for(long i=2;i<lg(ABCs);i++){
base=gadd(gtocol(gel(ABCs, i)), scol);//The first solution
embofD2=gadd(gadd(gmul(gel(zb, 1), gel(base, 1)), gmul(gel(zb, 2), gel(base, 2))), gmul(gel(zb, 3), gel(base, 3)));//The embedding of sqrt(D2)
for(int j=2;j<=4;j++) gel(baseemb, j)=gdivgs(gel(embofD2, j), 2);//The embedding of (p_{D2}+sqrt(D2))/2
if(equali1(qa_conjnorm(Q, ord, ordinv, e2, baseemb, gen_1, 0, prec))){glist_putstart(&S, gcopy(baseemb));count++;}//Good!
for(;;){
base=gadd(gmul(M, base), sshift);
embofD2=gadd(gadd(gmul(gel(zb, 1), gel(base, 1)), gmul(gel(zb, 2), gel(base, 2))), gmul(gel(zb, 3), gel(base, 3)));//The embedding of sqrt(D2)
for(int j=2;j<=4;j++) gel(emb, j)=gdivgs(gel(embofD2, j), 2);//The embedding of (p_{D2}+sqrt(D2))/2
if(equali1(qa_norm(Q, qa_simulconj(Q, ord, ordinv, e1, baseemb, e1, emb, prec)))) break;//Back to equivalent embeddings.
if(equali1(qa_conjnorm(Q, ord, ordinv, e2, emb, gen_1, 0, prec))){glist_putstart(&S, gcopy(emb));count++;}//Good!
}
}
GEN allposs=gerepileupto(top, glist_togvec(S, count, -1));//All the possibilities for emb2. We do want to make sure we did not double count, so we go through and check them.
if(gequal0(allposs)){avma=top;return cgetg(1, t_VEC);}//No solutions
pari_sp mid;
long pos=1, lgallpos=lg(allposs);
GEN final=vectrunc_init(lgallpos), tempelt, n;
vectrunc_append(final, cgetg(3, t_VEC));
gel(gel(final, 1), 1)=gcopy(e1);
gel(gel(final, 1), 2)=gcopy(gel(allposs, 1));
for(long i=2;i<lgallpos;i++){
mid=avma;
tempelt=qa_simulconj(Q, ord, ordinv, e1, gel(gel(final, 1), 2), e1, gel(allposs, i), prec);
n=qa_norm(Q, tempelt);
if(equali1(n)){avma=mid;continue;}//Not a new solution.
avma=mid;
vectrunc_append(final, cgetg(3, t_VEC));
pos++;
gel(gel(final, pos), 1)=gcopy(e1);
gel(gel(final, pos), 2)=gcopy(gel(allposs, i));
}
return gerepileupto(top, final);
}
//qa_xlink with typechecking
GEN qa_xlink_tc(GEN Q, GEN order, GEN e1, GEN e2, GEN x, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
qa_ordeichlercheck(order);
qa_eltcheck(e1);qa_eltcheck(e2);
GEN v1=qa_associatedemb(Q, order, e1, gen_0), v2=qa_associatedemb(Q, order, e2, gen_0);
return gerepileupto(top, qa_xlink(Q, order, gel(v1, 1), gel(v2, 1), gel(v1, 2), gel(v2, 2), x, prec));
}
//Finds the set of x's bewteen xmin and xmax (inclusive) which exhibit x-linking in the indefinite quaternion algebra ramified at pset.
GEN qa_xposs(GEN pset, GEN Psetprod, GEN D1, GEN D2, GEN xmin, GEN xmax){
pari_sp top=avma;
GEN D=mulii(D1, D2), r;
GEN rtD=sqrtremi(D, &r);//D=rtD^2+r
if(gequal0(xmax) && gequal0(xmin)){//Auto setting max
if(gequal0(r)) xmax=subis(rtD, 1);//Must remove rtD from the range.
else xmax=rtD;
}
if(smodis(D, 2)!=smodis(xmin, 2)) xmin=addis(xmin, 1);//making xmin==D (2)
int isbad=0;
if(gequal0(r) && cmpii(xmin, rtD)!=1 && cmpii(rtD, xmax)!=1) isbad=1;//D is a square, must avoid rootD
GEN N=shifti(subii(sqri(xmin), D), -2), set2, x;//N=(x^2-D1D2)/4
glist *S=NULL;//Stores the set of x's
long count=0;
if(!isbad){
for(x=xmin;cmpii(x, xmax)!=1;x=addis(x, 2)){//For loop over all x's
if(!gequal0(modii(N, Psetprod))){
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
continue;//Need PsetProd to divide N
}
set2=qa_ram_fromab(D1, N);
if(gequal(set2, pset)){glist_putstart(&S, x);count++;}
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
}
}
else{
for(x=xmin;cmpii(x, rtD)==-1;x=addis(x, 2)){//For loop over all x's up to rtD
if(!gequal0(modii(N, Psetprod))){
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
continue;//Need PsetProd to divide N
}
set2=qa_ram_fromab(D1, N);
if(gequal(set2, pset)){glist_putstart(&S, x);count++;}
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
}
N=addii(N, addis(x, 1));//At this point, x=rtD, so we increment by 2
for(GEN x=addis(rtD, 2);cmpii(x, xmax)!=1;x=addis(x, 2)){//For loop over all x's from rtD to xmax
if(!gequal0(modii(N, Psetprod))){
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
continue;//Need PsetProd to divide N
}
set2=qa_ram_fromab(D1, N);
if(gequal(set2, pset)){glist_putstart(&S, x);count++;}
N=addii(N, addis(x, 1));//N->N+x+1, i.e. from (x^2-D1D2)/4 to ((x+2)^2-D1D2)/4
}
}
return gerepileupto(top, glist_togvec(S, count, -1));
}
//qa_xposs with typecheck and allowing either Q or pset to be inputted
GEN qa_xposs_tc(GEN Qorpset, GEN D1, GEN D2, GEN xmin, GEN xmax){
pari_sp top=avma;
if(typ(Qorpset)!=t_VEC) pari_err_TYPE("Qorpset is either a quaternion algebra or even sized set of primes", Qorpset);
if(typ(xmin)!=t_INT || typ(xmax)!=t_INT) pari_err_TYPE("xmin and xmax must be integers", xmin);
if(!isdisc(D1) || !isdisc(D2)) pari_err_TYPE("D1 and D2 must be discriminants", D1);
if(lg(Qorpset)==1) return qa_xposs(Qorpset, gen_1, D1, D2, xmin, xmax);//Ramified nowhere
if(lg(Qorpset)==2 || typ(gel(Qorpset, 2))==t_INT){//Qorpset is the prime set
GEN prod=gen_1;
for(long i=1;i<lg(Qorpset);i++) prod=mulii(prod, gel(Qorpset, i));
return gerepileupto(top, qa_xposs(Qorpset, prod, D1, D2, xmin, xmax));
}
qa_check(Qorpset);
return qa_xposs(qa_getpram(Qorpset), qa_getpramprod(Qorpset), D1, D2, xmin, xmax);
}
/*
//Returns [[algebras],[xes]], where [algebras] is the set of primes ramifying in quaternion algebras which have some x-linking of forms of discriminants D1, D2 (with x^2<D1D2), and [xes] is the respective set of possible positive xes.
GEN qa_qalgposs(GEN D1, GEN D2){
pari_sp top=avma;
}
addhelp(possqalg,"Inputs D1,D2: positive discriminants D1,D2.\n Outputs [[algebras],[xes]], where [algebras] is the set of primes ramifying in quaternion algebras which have some x-linking of forms of discriminants D1,D2, where x^2<D1D2, and [xes] is the respective set of possible xes.");
possqalg(D1,D2)={
my(plist,D,xmin,N,x,ps,rams,i,ind,imin,imax,c=1);\\c is initialized for the first pass through
plist=List();;\\plist[i]=[[primes ramifying],[xes]]
D=D1*D2;
xmin=D%2;
N=(xmin-2)^2-D;\\Initializing N=x^2-D1D2
forstep(x=xmin,ceil(sqrt(D))-1,2,
N=N+4*x-4;\\N=x^2-D1D2
ps=factor(abs(N))[,1]~;
rams=[];
for(i=1,length(ps),if(hilbert(D1,N,ps[i])==-1,rams=setunion(rams,[ps[i]])));\\Finding the ramified primes.
imin=1;
imax=length(plist)+1;
while(imin<imax,\\insert into plist and xes
ind=floor((imin+imax)/2);
c=cmp(rams,plist[ind][1]);
if(c<0,
imax=ind
,
if(c>0,
imin=ind+1;
,\\c=0, already there.
plist[ind][2]=setunion(plist[ind][2],[x]);break
);
);
);
if(c!=0,\\Must add it in.
plist=listinsertsorted(plist,[rams,[x]])[1]
);
);
return([vector(length(plist),i,plist[i][1]),vector(length(plist),i,plist[i][2])]);
}*/
//INTERSECTION NUMBER BASED ON FUNDAMENTAL DOMAIN
//Compute the intersection number given the geodesics
GEN qa_inum_fd_givengeod(GEN Q, GEN order, GEN U, GEN geod1, GEN geod2, GEN pell1, GEN pell2, GEN D1D2, int data, GEN tol, long prec){
pari_sp top=avma, mid;
long l1=lg(gel(geod1, 1)), l2=lg(gel(geod2, 1)), maxint=(l1-1)*(l2-1);
GEN ind1=vecsmalltrunc_init(maxint+1), ind2=vecsmalltrunc_init(maxint+1);//Stores indices of the intersections
long g1start, g1end;
int doesint, left1, left2;
GEN angi11, angi12, angi21, angi22;
for(long i1=1;i1<l1;i1++){
if(gel(geod1, 3)[i1]<=gel(geod1, 4)[i1]){
g1start=gel(geod1, 3)[i1];//The smaller index of side 1
g1end=gel(geod1, 4)[i1];//The larger index of side 1
}
else{
g1start=gel(geod1, 4)[i1];
g1end=gel(geod1, 3)[i1];
}
for(long i2=1;i2<l2;i2++){
mid=avma;
doesint=sides_int_indices(g1start, g1end, gel(geod2, 3)[i2], gel(geod2, 4)[i2]);
if(doesint==1){vecsmalltrunc_append(ind1, i1);vecsmalltrunc_append(ind2, i2);continue;}
else if(doesint==-1) continue;//No intersection
//Now we have equal sides, so must be more careful.
if(gequal(gel(gel(gel(geod1, 2), i1), 7), gen_1)){//Arc is going forward
angi11=garg(gel(gel(gel(geod1, 2), i1), 3), prec);//Angle to start of i1 arc
angi12=anglediff(garg(gel(gel(gel(geod1, 2), i1), 4), prec), angi11, tol, prec);//Angle to end of i1 arc from the start
}
else{
angi11=garg(gel(gel(gel(geod1, 2), i1), 4), prec);//Angle to start of i1 arc
angi12=anglediff(garg(gel(gel(gel(geod1, 2), i1), 3), prec), angi11, tol, prec);//Angle to end of i1 arc from the start
}
angi21=anglediff(garg(gel(gel(gel(geod2, 2), i2), 3), prec), angi11, tol, prec);//Angle to "start" of i2 arc from the start
if(gequal0(angi21)){avma=mid;continue;}//We intersect on the boundary at the start of side 1, so don't count it so as to not overcount.
angi22=anglediff(garg(gel(gel(gel(geod2, 2), i2), 4), prec), angi11, tol, prec);//Angle to "end" of i2 arc from the start
if(gequal0(angi22)){avma=mid;continue;}//We intersect on the boundary at the start of side 1, so don't count it so as to not overcount.
left1=tolcmp(angi21, angi12, tol, prec);
if(left1==0){avma=mid;vecsmalltrunc_append(ind1, i1);vecsmalltrunc_append(ind2, i2);continue;}//We intersect on the boundary at the end of side 1
left2=tolcmp(angi22, angi12, tol, prec);
if(left2==0){avma=mid;vecsmalltrunc_append(ind1, i1);vecsmalltrunc_append(ind2, i2);continue;}//We intersect on the boundary at the end of side 1
avma=mid;
if(left1==left2){continue;}//On same side, no intersection
vecsmalltrunc_append(ind1, i1);//Reaching here means intersection
vecsmalltrunc_append(ind2, i2);
}
}
GEN g1emb=cgetg(l1, t_VEC), g2emb=cgetg(l2, t_VEC);//The embeddings corresponding to the geodesics
GEN D1m2=gdivgs(gmodgs(gel(pell1, 1), 2), 2), D2m2=gdivgs(gmodgs(gel(pell2, 1), 2), 2);
for(long i=1;i<l1;i++){
gel(g1emb, i)=cgetg(5, t_VEC);
gel(gel(g1emb, i), 1)=D1m2;
gel(gel(g1emb, i), 2)=gdiv(gel(gel(gel(geod1, 1), i), 2), gel(pell1, 2));//Divide by U
gel(gel(g1emb, i), 3)=gdiv(gel(gel(gel(geod1, 1), i), 3), gel(pell1, 2));//Divide by U
gel(gel(g1emb, i), 4)=gdiv(gel(gel(gel(geod1, 1), i), 4), gel(pell1, 2));//Divide by U
}
for(long i=1;i<l2;i++){
gel(g2emb, i)=cgetg(5, t_VEC);
gel(gel(g2emb, i), 1)=D2m2;
gel(gel(g2emb, i), 2)=gdiv(gel(gel(gel(geod2, 1), i), 2), gel(pell2, 2));//Divide by U
gel(gel(g2emb, i), 3)=gdiv(gel(gel(gel(geod2, 1), i), 3), gel(pell2, 2));//Divide by U
gel(gel(g2emb, i), 4)=gdiv(gel(gel(gel(geod2, 1), i), 4), gel(pell2, 2));//Divide by U
}
if(data==0){
GEN pairs=cgetg(lg(ind1), t_VEC);
for(long i=1;i<lg(ind1);i++){
gel(pairs, i)=cgetg(3, t_VEC);
gel(gel(pairs, i), 1)=gcopy(gel(g1emb, ind1[i]));
gel(gel(pairs, i), 2)=gcopy(gel(g2emb, ind2[i]));
}
return gerepileupto(top, pairs);
}
//Now we want data!
GEN ret=cgetg(3, t_VEC);
gel(ret, 1)=cgetg(lg(ind1), t_VEC);//Pairs
for(long i=1;i<lg(ind1);i++){
gel(gel(ret, 1), i)=cgetg(3, t_VEC);
gel(gel(gel(ret, 1), i), 1)=gcopy(gel(g1emb, ind1[i]));
gel(gel(gel(ret, 1), i), 2)=gcopy(gel(g2emb, ind2[i]));
}
gel(ret, 2)=cgetg(lg(ind1), t_VEC);//Intersection data
for(long i=1;i<lg(ind1);i++){
gel(gel(ret, 2), i)=qa_intlevel(Q, order, gel(gel(gel(ret, 1), i), 1), gel(gel(gel(ret, 1), i), 2), D1D2, prec);
}
return gerepileupto(top, ret);
}
//In a fundamental domain, take geodesics going between sides i1<=i2 and j1, j2. This returns 1 if they are guaranteed to intersect, -1 if they do not, and 0 if this is not enough info (which is the case iff {i1, i2, j1, j2} has at most 3 distinct elements).
static int sides_int_indices(long i1, long i2, long j1, long j2){
if(j1==i1) return 0;
int left1;
if(j1<i1) left1=1;//j1 is "left" of i1->i2
else{
if(j1==i2) return 0;
if(j1<i2) left1=0;//j1 is "right of i1->i2
else left1=1;//j1 is "left" of i1->i2
}
if(j2==i1) return 0;
int left2;
if(j2<i1) left2=1;//j2 is "left" of i1->i2
else{
if(j2==i2) return 0;
if(j2<i2) left2=0;//j2 is "right of i1->i2
else left2=1;//j2 is "left" of i1->i2
}
if(left1!=left2) return 1;//Intersect! Other sides
return -1;//No intersect, on same side
}
//Compute the intersection number given the fundamental domain
GEN qa_inum_fd_tc(GEN Q, GEN order, GEN U, GEN e1, GEN e2, int data, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
if(typ(U)!=t_VEC || lg(U)!=9) pari_err_TYPE("Please input a normalized boundary", U);
qa_eltcheck(e1);qa_eltcheck(e2);
GEN v1=qa_associatedemb(Q, order, e1, gen_0), v2=qa_associatedemb(Q, order, e2, gen_0);
GEN D1D2=mulii(gel(v1, 2), gel(v2, 2));//D1*D2
GEN pell1=pell(gel(v1, 2)), pell2=pell(gel(v2, 2));
v1=gmul(gel(v1, 1), gel(pell1, 2));//Represents U1*sqrt(D1)/2+C for C rational
gel(v1, 1)=gdivgs(gel(pell1, 1), 2);//T1/2
v2=gmul(gel(v2, 1), gel(pell2, 2));//Represents U2*sqrt(D2)/2+C for C rational
gel(v2, 1)=gdivgs(gel(pell2, 1), 2);//T2/2
GEN tol=deftol(prec);
GEN geod1=qa_rootgeodesic_fd(Q, U, v1, tol, prec);
GEN geod2=qa_rootgeodesic_fd(Q, U, v2, tol, prec);
return gerepileupto(top, qa_inum_fd_givengeod(Q, order, U, geod1, geod2, pell1, pell2, D1D2, data, tol, prec));
}
//INTERSECTION SERIES
//Returns the intersection series associated to e1 and e2. U must be the fundamental domain, we go up to q^N, and type=0 means unsigned, 1=signed, >=2 means q-weighted for q=type.
GEN qa_inumseries(GEN Q, GEN order, GEN U, GEN e1, GEN e2, GEN D1, GEN D2, GEN N, int type, long prec){
pari_sp top=avma;
GEN reps=qa_orbitrepsrange(Q, order, N, prec);
long Np1=itos(N)+1;
GEN rvec=cgetg(Np1, t_VEC);
for(long i=1;i<Np1;i++){
GEN hforms=qa_hecke(Q, order, gel(reps, i), e2, D2, prec);
if(gequal0(hforms)){gel(rvec, i)=gen_0;continue;}//Not coprime to the area
GEN inum=gen_0;
for(long j=1;j<lg(hforms);j++){
GEN data=qa_inum_fd_tc(Q, order, U, e1, gel(gel(hforms, j), 2), 1, prec);
inum=addii(inum, mulii(gel(gel(hforms, j), 1), qa_inum_fromdata(gel(data, 2), type)));
}
gel(rvec, i)=inum;
}
return gerepilecopy(top, rvec);
}
//qa_inumseries with typecheck
GEN qa_inumseries_tc(GEN Q, GEN order, GEN U, GEN e1, GEN e2, GEN N, int type, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
if(typ(U)!=t_VEC || lg(U)!=9) pari_err_TYPE("Please input a fundamental domain", U);
qa_eltcheck(e1);
qa_eltcheck(e2);
if(typ(N)!=t_INT || signe(N)!=1) pari_err_TYPE("Please input a positive integer", N);
GEN v1=qa_associatedemb(Q, order, e1, gen_0), v2=qa_associatedemb(Q, order, e2, gen_0);
return gerepileupto(top, qa_inumseries(Q, order, U, gel(v1, 1), gel(v2, 1), gel(v1, 2), gel(v2, 2), N, type, prec));
}
//INTERSECTION DATA
//Returns [signed level, x] associated to the intersection pair e1, e2.
GEN qa_intlevel(GEN Q, GEN order, GEN e1, GEN e2, GEN D1D2, long prec){
pari_sp top=avma;
long lx;
GEN v1=cgetg_copy(e1, &lx);
gel(v1, 1)=gen_0;
for(int i=2;i<=4;i++) gel(v1, i)=gmulgs(gel(e1, i), 2);//image of sqrt(D1)
GEN v2=cgetg_copy(e2, &lx);
gel(v2, 1)=gen_0;
for(int i=2;i<=4;i++) gel(v2, i)=gmulgs(gel(e2, i), 2);//image of sqrt(D2)
GEN T=qa_mul(Q, v1, v2);
GEN D=subii(sqri(gel(T, 1)), D1D2);//gel(T, 1)=x, and D=x^2-D1D2
GEN aemb=qa_associatedemb(Q, order, gdivgs(T, 2), D);//The embedding associated to T, which is naively an emebdding of discriminant D=x^2-D1D2
GEN levelsq=diviiexact(D, gel(aemb, 2));//the level squared
GEN level=sqrti(levelsq);
GEN infor=qa_orinfinite(Q, gel(aemb, 1), gel(aemb, 2), prec);
GEN ret=cgetg(3, t_VEC);
if(equali1(infor)) gel(ret, 1)=icopy(level);
else gel(ret, 1)=negi(level);
gel(ret, 2)=icopy(gel(T, 1));
return gerepileupto(top, ret);
}
//qa_intlevel with typecheck
GEN qa_intlevel_tc(GEN Q, GEN order, GEN e1, GEN e2, GEN D1, GEN D2, long prec){
pari_sp top=avma;
qa_indefcheck(Q);
qa_ordeichlercheck(order);
qa_eltcheck(e1);qa_eltcheck(e2);
if(typ(D1)!=t_INT) pari_err_TYPE("D1 is either 0 or the discriminant of e1", D1);
if(typ(D2)!=t_INT) pari_err_TYPE("D2 is either 0 or the discriminant of e2", D2);
if(gequal0(D1)) D1=gel(qa_associatedemb(Q, order, e1, gen_0), 2);
if(gequal0(D2)) D2=gel(qa_associatedemb(Q, order, e2, gen_0), 2);//setting discriminnats
GEN D1D2=mulii(D1, D2);
return gerepileupto(top, qa_intlevel(Q, order, e1, e2, D1D2, prec));
}
//Returns the intersection number given the intersection data. type=0 means unsigned, 1=signed, >=2 means q-weighted for q=type.
GEN qa_inum_fromdata(GEN data, int type){
pari_sp top=avma;
if(type==0) return stoi(lg(data)-1);
long inum=0;
if(type==1){
for(long i=1;i<lg(data);i++){
if(signe(gel(gel(data, i), 1))==1) inum++;
else inum--;
}
return stoi(inum);
}
GEN q=stoi(type);
for(long i=1;i<lg(data);i++){
inum=inum+signe(gel(gel(data, i), 1))*(1+Z_pval(gel(gel(data, i), 1), q));
}
avma=top;
return stoi(inum);
}