egprog() { short nperms,ngens,nb,stab,olfb,nlfb,i,l,m,z,ct; if (full) { if ((ip=fopen(inf2,"r"))==0) { printf("Cannot open %s.\n",inf2); return(-1); } fscanf(ip,"%hd%hd",&i,&ngens); fclose(ip); } if ((ip=fopen(inf,"r"))==0) { printf("Cannot open %s.\n",inf); return(-1); } if (stabcall) { printf("Stabilize how many points? "); scanf("%hd",&stab);} else stab=0; fscanf(ip,"%hd%hd%hd%hd",&npt,&nperms,&nb,&l); if (npt>mnpt) {fprintf(stderr,"npt,too big. Increase NPT.\n"); return(-1); } if (nperms>mp) { fprintf(stderr,"nperms too big. Increase MXP.\n"); return(-1);} if (nb>mb) {fprintf(stderr,"nb too big. Increase MNB.\n"); return(-1); } if (l<=0) {fprintf(stderr,"Wrong input format.\n"); return(-1); } for (i=0;i<=nperms;i++) pptr[i]=perm+(npt+1)*i-1; for (i=1;i<=nb;i++) svptr[i]=sv2+npt*(i-1)-1; for (i=1;i<=npt;i++) pptr[0][i]=i; readbaselo(nb,base,lorb); for (i=1;i<=nperms;i++) readvec(pptr[i],1); fclose(ip); /* Main algorithm begins */ op=fopen(outf,"w"); *pno=0; fixpt[nb+1]=0; for (l=nb;l>stab;l--) { for (i=1;i<=nperms;i++) if (pptr[i][npt+1]==l) {(*pno)++; pno[*pno]=i; } fixpt[l]= *pno; } for (l=stab;l>=1;l--) fixpt[l]=fixpt[stab+1]; for (l=nb;l>stab;l--) { olfb=fixpt[l+1]; nlfb=fixpt[l]; if ((nlfb-olfb)>1) for (i=olfb+1;i<=nlfb;i++) { z=pno[i]; /* We test whether perm no z is redundant as a generator. If so, we leave pno[i]=0, if not we put it back to z. */ if (full==0 || z>ngens) { pno[i]=0; for (m=l;m>stab && pno[i]==0;m--) {*pno=fixpt[m]; if (orbitsv(base[m],svptr[m],0)<lorb[m]) pno[i]=z;} } } } *pno=0; l=0; for (i=nb;i>=1;i--) { z=fixpt[i]; while (l<z) { l++; if (pno[l]>0) { (*pno)++; pno[*pno]=pno[l]; } } lorb[i]=orbitsv(base[i],svptr[i],0); } fprintf(op,"%4d%4d%4d%4d\n",npt,*pno,nb,3); printbaselo(nb,base,lorb); printpsv(nb,pno,svptr); for (i=1;i<=*pno;i++) fprintf(op,"%4d",pno[i]); fprintf(op,"\n"); return(0); }
int sylprog (int x) /* x!=0 means that a subgroup of the sylp-group has already been computed, and is stored in inf2. The search should now start at bno=abs(x). sylnorm always returns the current bno, if it exits in order to compute the normalizer. It returns 0 if it completes the computation of the sylp-group If x>0, its normalizer has also been computed, and lies in outf1, and so the next element of the sylp-group will be sought from this normalizer. Any p-element will do in this case. */ { char nontriv,seek,b,incadno,comm,pow,id; short i,j,k,l,m,n,lnt,fnt,mxp,mnb,mxexp,nperms,ct,stp, *z,*ap,*sva,ct1,ct2,ct3,ord,kord,pt,stbno,skct, *tpk,*itpk,*commp,*icommp,*spptr; int quot; adpt=obase; lorbdef=nbase; ntfpt=ntorno; ntbpt=lorbn; invbase=reg; facord=ipno; orno=tsv1; lporb=tsv2; deft=tsv3; orbp=orep, gorb=intorb; if (chpar) /* par1,par2 and par3 are used when seeking random elts. See line 152 (approx). When searching for an element normalizing a subgroup, we give up after par4 attempts, and use normalizer program to compute the complete normalizer first. */ { printf("Choose values of par1,par2,par3,par4. (Defaults are %d %d %d %d.)\n", par1,par2,par3,par4); scanf("%hd%hd%hd%hd",&par1,&par2,&par3,&par4); } if ((ip=fopen(inf1,"r"))==0) { fprintf(stderr,"Cannot open %s.\n",inf1); return(-1);} if (x==0) { printf("Input prime! "); scanf("%hd",&prime);} fscanf(ip,"%hd%hd%hd%hd",&npt,&nperms,&nb,&l); if (npt>mnpt) {fprintf(stderr,"npt too big. Increase NPT.\n"); return(-1);} if (nb>mb) {fprintf(stderr,"nb too big. Increase MB.\n"); return(-1);} if (nb*npt*2>svsp) { fprintf(stderr,"Out of sv space. Increase SVSP.\n");return(-1);} if (l<=2) { fprintf(stderr,"Wrong input format.\n"); return(-1); } readbaselo(nb,gbase,lorbg); for (i=1;i<=npt;i++) invbase[i]=0; for (i=1;i<=nb;i++) invbase[gbase[i]]=i; lnt=0; fnt=0; /* Determine order of sylp subgroup . Links ntfpt and ntbpt are used to bypass trivial indices in the stab chain. */ for (i=nb;i>=1;i--) { j=lorbg[i]; lorbh[i]=1; if (j>1) { if (lnt==0) { lnt=i; k=i; ntfpt[lnt]=nb+1; } else { ntbpt[k]=i; ntfpt[i]=k; k=i; } while (j%prime==0) {lorbh[i]*=prime; j/=prime; } if (lorbh[i]>1) fnt=i; } else invbase[gbase[i]]=0; } ntbpt[fnt]=0; ntfpt[0]=fnt; if (fnt==0) {fprintf(stderr,"Sylp-group is trivial!\n"); return(-1); } printf("Orbit lengths of Sylp-group:\n"); for (i=1;i<=nb;i++) printf("%4d",lorbh[i]); printf(".\n"); quot=psp/(npt+1); if (quot>mp) quot=mp; mxp=quot; quot=sp/npt; if (quot>mexp) quot=mexp; mxexp=quot; np2=2*nperms-1; if (np2>=mxp) {fprintf(stderr,"Out of space. Increase PSP (or MP).\n"); return(-1); } for (i=0;i<mxp;i++) pptr[i]=perm+(npt+1)*i-1; for (i=1;i<=nb;i++) svgptr[i]=sv+npt*(i-1)-1; for (i=1;i<=nb;i++) svhptr[i]=sv+npt*(i+nb-1)-1; readpsv(0,nb,nperms,svgptr); fclose(ip); tpk=space-1; itpk=tpk+npt; commp=itpk+npt; icommp=commp+npt; spptr=icommp+npt+1; sp-=4*npt; for (i=fnt;i<=nb;i=ntfpt[i]) { gorb[i]=spptr; sva=svgptr[i]; for (j=1;j<=npt;j++) if (sva[j]!=0) *(spptr++)=j; sp-=lorbg[i]; } /* We now determine how much space we have to store cosetrep perms, and calculate these with the function exprep */ lexp=0; ct=0; for (i=nb;i>fnt;i--) { ct+=(lorbg[i]-1); if (ct+i>=mxexp) {lexp=i; break; } } if (lexp==0) lexp=fnt; z=spptr-1; i=0; while (i!=lexp) { i= (i==0) ? fnt : ntfpt[i]; expptr[i]=z; z+=npt; start[i]=i-1; } printf("lexp=%d.\n",lexp); start[lexp+1]=lexp; k=lexp; for (i=lexp+1;i<=lnt;i++) { l=lorbg[i]; start[i+1]=start[i]+l-1; if (l>1) for (j=1;j<=npt;j++) if (svgptr[i][j]>0) { k++; expptr[k]=z; z+=npt; exprep(j,k,svgptr[i]); } } /* nontriv nonzero means in this case that any p-element in G will do; i.e. we are not looking for an element in the normalizer of the group found so far. This is the case either when x=0, right at the beginning, or when x>0, and we have already computed this normalizer. */ nontriv = (x>=0) ? 0 : 1; *pno=0; deft[0]=0; stp=np2+1; if (x!=0) { ip=fopen(inf2,"r"); fscanf(ip,"%hd%hd%hd%hd",&i,&nperms,&i,&i); readbaselo(nb,gbase,lorbdef); readpsv(stp,nb,nperms,svhptr); for (i=1;i<=nperms;i++) { (*pno)++; pno[*pno]= ++np2; fscanf(ip,"%hd",facord+np2); np2++;} fclose(ip); k=abs(x); l=0; for (i=1;i<=npt;i++) if (svhptr[k][i]!=0) orb[++l]=i; } else for (i=1;i<=nb;i++) { for (j=1;j<=npt;j++) {svhptr[i][j]=0; svhptr[i][gbase[i]]= -1; } lorbdef[i]=1; } /* Starting search. lorbdef records the length of the orbit ofthe Sylp group found so far. tp is a perm that we are currently considering as a possible new element of Sylp. nontriv is set true as soon as the first element of Sylp has been found. adno is then number in the stabilizer chain for which the coset reps are currently being advanced in the search. */ stbno= (x==0) ? lnt : abs(x); for (bno=stbno;bno>=fnt;bno=ntbpt[bno]) { printf("bno=%d.\n",bno); if (bno!=abs(x)) orb[1]=gbase[bno]; adno=bno; while (lorbdef[bno]<lorbh[bno]) { if (np2+2>=mxp) {fprintf(stderr,"Out of space. Increase PSP.\n"); return(-1); } *expcp=0; tp=pptr[np2+1]; itp=pptr[np2+2]; tp[npt+1]=bno; /* When any p-element suffices, we seek random elements until one of them has order a multiple of p. If after par1 elements, we have not found one, we try commutators of the last element with new elements. If after par2 such attempts we still have not found one, we try commutators with powers of the last element. Then if after par3 tries we are still unsuccessful, we try a new random element, and use commutators with that and so on... */ if (nontriv==0) { ct1=0; seek=1; comm=0; while (seek) { (*expcp)=0; ranelt(); ord=findord(tp,itp); if (ord%prime!=0) { ct1++; if (comm) { id=1; for (i=1;i<=npt;i++) { j=itpk[itp[tpk[tp[i]]]]; if (id && j!=i) id=0; commp[i]=j; icommp[i]=0; } if (id) { ct2++; if (ct2>par2) comm=0; continue; } ct2=0; ord=findord(commp,icommp); if (ord%prime!=0) { ct3++; if (ct3>par3) { if (pow) { comm=0; continue;} printf("Trying powers.\n"); pow=1; ct3=0; ct2=0; for (i=2;i<=kord;i++) if (kord%i==0) { if (kord==i) { comm=0; continue;} k=kord/i; for (i=1;i<=npt;i++) { pt=i; for (j=1;j<=k;j++) pt=tpk[pt]; itpk[pt]=i;} invert(itpk,tpk); } } } else { for (i=1;i<=npt;i++) {tp[i]=commp[i]; itp[i]=icommp[i];} seek=0; } } /* if (comm) */ else if (ct1>par1 && ord>1) { printf("Trying commutators.\n"); ct2=0; ct3=0; kord=ord; comm=1; pow=0; for (i=1;i<=npt;i++) { tpk[i]=tp[i]; itpk[i]=itp[i];} } } /* if (ord%prime... */ else seek=0; if (seek==0) { while (ord%prime==0) ord/=prime; if (ord>1) { for (i=1;i<=npt;i++) { pt=i; for (j=1;j<=ord;j++) pt=tp[pt]; itp[pt]=i; } invert(itp,tp); } if (svhptr[bno][tp[gbase[bno]]]!=0) { seek=1; comm=0; } else { nontriv=1; fndelt();} } } /* while (seek) */ continue; } /* if (nontriv==0) */ for (i=1;i<=npt;i++) tp[i]=i; /* Now we compute all orbits of Sylp. The new element must permute these orbits. */ allorbs(lporb,orno); bnoorno=orno[gbase[bno]]; bt=0; opno=bnoorno; opct=1; skct=0; for (i=1;i<= *lporb;i++) {orbp[i]=0; deft[i]=0; } seek=1; while (seek) { skct++; if (skct>par4) { printf("Exiting to try normrun.\n"); for (i=1;i<=npt;i++) if (svhptr[bno][i]== -2) svhptr[bno][i]=0; goto exit; } for (i=1;i<=npt;i++) itp[i]=0; b=1; ok=1; while(1) { k=expcp[*expcp]; /* Advance to next element in search. ok is set false if this element is eliminated from membership of Sylp, and we advance again. */ if (b && (*expcp==0 || k<=start[adno])) { (*expcp)++; if (adno<=lexp) {adpt[adno]=0; expcp[*expcp]=adno;} else expcp[*expcp]=start[adno]; b=0; } else { if (adno<=lexp) { sva=svgptr[adno]; ap=adpt+adno; (*ap)++; while (sva[*ap]<=0 && *ap<=npt) (*ap)++; if (*ap<=npt) { exprep(*ap,adno,sva); break; } } else if (k<start[adno+1]) {expcp[*expcp]++; break;} if (*expcp==1) { fprintf(stderr,"Premature end of search.\n"); return(-1); } (*expcp)--; bt=1; adno=ntbpt[adno]; b=1; if (adno==bno) { im=expimage(gbase[adno]); l=orno[im]; if (lporb[l]>1) for (i=1;i<=npt;i++) if (orno[i]==l) svhptr[adno][i]= -2; /* Setting sv= -2 when bno=adno avoids testing more elements than necessary which map the main orbit of gbase[bno] to another. */ } } } incadno=1; while (incadno) { cb=gbase[adno]; im=expimage(cb); if (adno==bno && svhptr[adno][im]!=0) {ok=0;break;} if (bt) { for (i=1;i<=*lporb;i++) { j=orbp[i]; if (deft[j]>=adno) {orbp[i]=0;deft[j]=0;}} opno=bnoorno; opct=1; l=orbp[opno]; while (l!=0 && l!=bnoorno) { opct++; opno=l; l=orbp[l];} } bt=0; deforbp(); if (ok==0) break; /* This means either that the element does not permute the orbits, or that it permutes the base orbit with a cycle of p' length. */ else tp[cb]=im; if (adno==lnt) { incadno=0; for (cb=1;cb<=npt;cb++) if (invbase[cb]==0) { im=expimage(cb); deforbp(); if (ok==0) { bt=1; break; } else tp[cb]=im; } } else adno=ntfpt[adno]; } if (ok) { ord=findord(tp,itp); while (ord%prime==0) ord/=prime; if (ord>1) { for (i=1;i<=npt;i++) { n=i; for (j=1;j<=ord;j++) n=tp[n]; itp[n]=i;} invert(itp,tp); } } /* if ok then tp is a p-element permuting the orbits. The final test is to check that it normalizes Sylp. */ if (ok) for (i=stp;i<np2 && ok;i+=2) { *cp=0; for (j=fnt;j<=lnt;j=ntfpt[j]) { k=image(tp[pptr[i][itp[gbase[j]]]]); if (svhptr[j][k]!=0) addsv(k,svhptr[j]); else {ok=0; bt=1; break; } } if (ok==0 && ord>1) for (j=bno;j<lnt;j=ntfpt[j]) tp[gbase[j]]=expimage(gbase[j]); } /* if ok then tp is the new element of Sylp */ if (ok) { seek=0; fndelt();} } } for (i=1;i<=npt;i++) if (svhptr[bno][i]== -2) svhptr[bno][i]=0; } exit: op=fopen(inf2,"w"); fprintf(op,"%4d%4d%4d%4d\n",npt,*pno,nb,4); printbaselo(nb,gbase,lorbdef); printpsv(nb,pno,svhptr); if (bno==0) fprintf(op,"%3d",prime); for (i=stp;i<np2;i+=2) fprintf(op,"%4d",facord[i]); fprintf(op,"\n"); fclose(op); if (x<=0 && bno>0) { op=fopen(outf1,"w"); *pno=0; for (i=0;i<stp;i+=2) if (pptr[i][npt+1]>=bno) { (*pno)++; pno[*pno]=i;} for (i=1;i<bno;i++) if (lorbg[i]>1) { lorbg[i]=1; for (j=1;j<=npt;j++) svgptr[i][j]=0; svgptr[i][gbase[i]]= -1; } fprintf(op,"%4d%4d%4d%4d\n",npt,*pno,nb,3); printbaselo(nb,gbase,lorbg); printpsv(nb,pno,svgptr); fclose(op); } return(bno); }