13 static pair<matrix,matrix> RowReduce(matrix mat) {
15 int &v_max = fermat.vmax;
19 for(const_preorder_iterator i = tree.preorder_begin(); i != tree.preorder_end(); ++i) {
21 if(is_a<symbol>(e) || e.match(
a(
w))) rep_vs.append(e);
30 for(
auto vi : rep_vs) {
31 auto name =
"v" + to_string(fvi);
32 v2f[vi] = Symbol(name);
39 cout << rep_vs << endl;
40 throw Error(
"Fermat: Too many variables.");
43 for(
int i=v_max; i<fvi; i++) ss <<
"&(J=v" << i <<
");" << endl;
44 fermat.Execute(ss.str());
50 int nrow = mat.rows();
51 int ncol = mat.cols();
53 ss <<
"Array m[" << nrow <<
"," << ncol+1 <<
"];" << endl;
54 fermat.Execute(ss.str());
59 for(
int c=0; c<ncol; c++) {
60 for(
int r=0; r<nrow; r++) {
61 ss << mat(r,c).subs(
iEpsilon==0).subs(v2f) <<
",";
65 for(
int r=1; r<nrow; r++) ss <<
",0";
67 fermat.Execute(ss.str());
72 if(nrow>1100) sparse =
true;
73 if(sparse) fermat.Execute(
"Sparse([m]);");
74 fermat.Execute(
"Redrowech([m],[t]);");
75 ss <<
"&(U=1);" << endl;
76 matrix mr(nrow, ncol);
78 if(sparse) ss <<
"![m]" << endl;
79 else ss <<
"![m" << endl;
80 auto ostr = fermat.Execute(ss.str());
85 if(ostr[ostr.length()-1]!=
'0')
throw Error(
"RowReduce, last char is NOT 0.");
86 ostr = ostr.substr(0, ostr.length()-1);
89 ostr.erase(0, ostr.find(
":=")+2);
90 size_t sn = ostr.length();
92 for(
size_t i=0; i<sn; i++) {
96 if(sparse && i>0 && lc==
'}') ostr[i-1] =
',';
97 }
else if(c==
']') c =
'}';
98 else if(sparse && (c==
' '||c==
'\t'||c==
'\n'||c==
'\r'))
continue;
103 auto res = fp.Read(ostr);
105 for(
auto const & item : res) {
107 for(
auto const & it : item) {
108 if(r==-1) r =
ex2int(it)-1;
110 int c =
ex2int(it.op(0))-1;
116 for(
int r=0; r<nrow; r++) {
117 auto cur = res.op(r);
118 for(
int c=0; c<ncol; c++) mr(r,c) = cur.op(c);
122 matrix mt(nrow, nrow);
124 if(sparse) ss <<
"![t]" << endl;
125 else ss <<
"![t" << endl;
126 auto ostr = fermat.Execute(ss.str());
131 if(ostr[ostr.length()-1]!=
'0')
throw Error(
"Solver::Export, last char is NOT 0.");
132 ostr = ostr.substr(0, ostr.length()-1);
135 ostr.erase(0, ostr.find(
":=")+2);
136 size_t sn = ostr.length();
138 for(
size_t i=0; i<sn; i++) {
142 if(sparse && i>0 && lc==
'}') ostr[i-1] =
',';
143 }
else if(c==
']') c =
'}';
144 else if(sparse && (c==
' '||c==
'\t'||c==
'\n'||c==
'\r'))
continue;
148 auto res = fp.Read(ostr);
150 for(
auto const & item : res) {
152 for(
auto const & it : item) {
153 if(r==-1) r =
ex2int(it)-1;
155 int c =
ex2int(it.op(0))-1;
161 for(
int r=0; r<nrow; r++) {
162 auto cur = res.op(r);
163 for(
int c=0; c<nrow; c++) mt(r,c) = cur.op(c);
168 ss <<
"&(U=0);" << endl;
169 ss <<
"@([m],[t]);" << endl;
170 ss <<
"&_G;" << endl;
171 fermat.Execute(ss.str());
174 return make_pair(mr,mt);
181 int &v_max = fermat.
vmax;
184 for(
auto const & rv : smat)
for(
auto const & cv : rv.second) {
186 for(const_preorder_iterator i = tree.preorder_begin(); i != tree.preorder_end(); ++i) {
188 if(is_a<symbol>(e) || e.match(
a(
w))) rep_vs.append(e);
198 for(
auto vi : rep_vs) {
199 auto name =
"v" + to_string(fvi);
207 cout << rep_vs << endl;
208 throw Error(
"Fermat: Too many variables.");
211 for(
int i=v_max; i<fvi; i++) ss <<
"&(J=v" << i <<
");" << endl;
219 int nrow=-1, ncol=-1;
220 int rcur = 0, rtot = smat.size();
222 for(
auto const & rv : smat) {
224 if(rv.second.size()<1)
continue;
227 oss <<
"[" << r+1 <<
",";
228 int ccur = 0, ctot = rv.second.size();
229 for(
auto const & cv : rv.second) {
233 oss <<
"[" << c+1 <<
"," << cv.second.subs(
iEpsilon==0).subs(v2f) <<
"]";
234 if(ccur!=ctot) oss <<
",";
237 if(rcur!=rtot) oss <<
"`" << endl;
238 else oss <<
"];" << endl;
243 ss <<
"Array m[" << nrow <<
"," << ncol+1 <<
"] Sparse;" << endl;
248 ss <<
"[m]:=" << oss.str();
255 if(pn>0) fermat.
Execute(
"Redrowech([m],,,"+to_string(pn)+
");");
256 else fermat.
Execute(
"Redrowech([m]);");
257 ss <<
"&(U=1);" << endl;
260 ss <<
"![m]" << endl;
261 auto ostr = fermat.
Execute(ss.str());
266 if(ostr[ostr.length()-1]!=
'0')
throw Error(
"RowReduce, last char is NOT 0.");
267 ostr = ostr.substr(0, ostr.length()-1);
270 ostr.erase(0, ostr.find(
":=")+2);
271 size_t sn = ostr.length();
273 for(
size_t i=0; i<sn; i++) {
277 if(i>0 && lc==
'}') ostr[i-1] =
',';
278 }
else if(c==
']') c =
'}';
279 else if(c==
' '||c==
'\t'||c==
'\n'||c==
'\r')
continue;
284 auto res = fp.
Read(ostr);
285 for(
auto const & item : res) {
287 for(
auto const & it : item) {
288 if(r==-1) r =
ex2int(it)-1;
290 int c =
ex2int(it.op(0))-1;
291 smat[r][c] = it.op(1);
297 ss <<
"&(U=0);" << endl;
298 ss <<
"@([m]);" << endl;
299 ss <<
"&_G;" << endl;
313 for(
int i=0; i<pdim; i++) {
314 for(
int j1=0; j1<n; j1++)
for(
int j2=0; j2<n; j2++) {
315 ibps.append(ibps.op(i).subs(lst{a(j1)==a(j1)+1}).subs(lst{a(j2)==a(j2)-1}));
322 for(
int i=0; i<pdim; i++) {
323 for(
int j=0; j<n; j++) ibps.append(ibps.op(j).subs(
a(i)==
a(i)+1));
330 for(
int i=0; i<pdim; i++) {
331 for(
int j=0; j<n; j++) ibps.append(ibps.op(j).subs(
a(i)==
a(i)-1));
340 find(ibps, F(
w), fs);
344 for(
int i=0; i<fi.op(0).nops(); i++) a2a[
a(i)] =
a(i)-fi.op(0).op(i).subs(
a(i)==0);
345 for(
int j=0; j<n; j++) ibps.append(ibps.op(j).subs(a2a));
361 vector<int> nfix(pdim);
362 for(
int i=0; i<pdim; i++) nfix[i] = 0;
364 for(
auto const & kv : n2n) {
367 if(k<0) k = pdim + k;
368 a2n[
a(k)] = kv.second;
371 int nibps = ibps.nops();
372 for(
int i=0; i<nibps; i++) ibps.let_op(i) = ibps.op(i).subs(a2n);
376 map<ex,lst,ex_is_less> cons_vec;
377 while(ibps.nops()>0) {
382 find(ibps,F(
w),fset);
383 map<int,exvector> fmap_vec;
384 for(
auto fi : fset) {
385 int psum = 0, nsum = 0;
387 for(
int i=0; i<pdim; i++) {
388 auto ni = ns.op(i).subs(
a(i)==0,
nopat);
389 if(sector.op(i)==1) {
390 if(nfix[i]==1 && ni<=0) psum -= 100000;
391 else psum +=
ex2int(ni)-1;
393 if(nfix[i]==1 && ni>0) nsum +=
ex2int(ni);
398 fmap_vec[-key].push_back(fi);
401 map<ex,int,ex_is_less> f2n;
403 exvector fvec(fset.size());
405 for(
auto const & ev : fmap_vec) {
406 if(next_gi<1) next_gi = ev.second.size();
407 for(
auto const & v : ev.second) {
417 cout <<
"+----------------------------------------" << endl;
418 cout <<
"| IBPs: " << nr <<
", Fs: " << next_gi <<
"/" << nc << endl;
419 cout <<
"+----------------------------------------" << endl;
423 for(
int r=0; r<nr; r++) {
424 auto ibp = ibps.op(r);
426 for(
auto & cv : cvs) {
427 int c = f2n[cv.op(1)];
429 if(!cc.is_zero()) smat[r][c] = cc;
432 RowReduce(smat,next_gi);
435 for(
auto const & rv : smat) {
436 int next_sol_ibp = -1;
440 for(
auto const & cv : rv.second) {
441 if(next_sol_ibp==-1) {
442 if(cv.first<next_gi) {
444 if(!is_zero(cv.second-1))
throw Error(
"Solver::SolveSparse, NOT 1.");
445 ns = fvec[cv.first].op(0).subs(
a(
w)==0);
446 for(
int i=0; i<pdim; i++) {
447 if(nfix[i]==0) amap[
a(i)] =
a(i)-ns.op(i);
449 fc = fvec[cv.first].subs(amap,
nopat);
450 for(
int i=0; i<pdim; i++) {
453 if(sector.op(i)==1 && nv[i]<1)
goto next_row;
454 else if(sector.op(i)!=1 && nv[i]>0)
goto next_row;
455 }
else nv[i] = (sector.op(i)==1 ? 1 : 0);
458 }
else next_sol_ibp = 2;
461 if(next_sol_ibp==0) {
462 if(cv.first<next_gi)
goto next_row;
467 ex f = fvec[cv.first];
469 if(next_sol_ibp==1) {
470 f = f.subs(amap,
nopat);
471 nd = nd.subs(amap,
nopat).numer_denom();
474 ns = f.op(0).subs(
a(
w)==0);
475 for(
int i=0; i<pdim; i++) {
476 if(sector.op(i)!=1) {
479 if(nsi>0)
goto next_row;
485 ex cci = num.subs(
a(i)==1+nsi);
486 if(!cci.is_zero() || nsi>=nv[i])
break;
489 if(nsi<nv[i]) nv[i] = nsi;
493 if(!is_a<mul>(den)) den = lst{ den };
494 for(
const auto & item : den) {
496 if(is_a<power>(it)) it = it.op(0);
497 if(!it.has(
a(
w)) || it.has(
d))
continue;
501 cout <<
"size is NOT 1: " << it << endl;
502 throw Error(
"Solver::SolveSparse, error occured.");
504 ex aw = *(
as.begin());
505 if(it.degree(aw)!=1) {
506 cout <<
"degree is NOT 1: " << it << endl;
507 throw Error(
"Solver::SolveSparse, error occured.");
509 ex a0 = -it.coeff(aw,0)/it.coeff(aw,1);
510 int an =
ex2int(aw.op(0));
511 if(sector.op(an)!=1) {
512 if(a0<=nv[an]) nv[an] = a0-1;
514 if(a0>=nv[an]) nv[an] = a0+1;
522 if(next_sol_ibp==1) {
523 for(
int i=0; i<pdim; i++) {
524 if(nfix[i]==1 && fc.op(0).op(i)!=nv[i]) {
525 cout << i <<
": " << fc << endl << nv << endl;
526 throw Error(
"Solver::SolveSparse, error occured.");
527 }
else if(nfix[i]==0 && fc.op(0).op(i)!=
a(i)) {
528 cout << i <<
": " << fc << endl << nv << endl;
529 throw Error(
"Solver::SolveSparse, error occured.");
534 int psum = 0, nsum = 0;
535 for(
int i=0; i<pdim; i++) {
536 if(nfix[i]==1)
continue;
537 if(sector.op(i)==1 && !nv[i].is_equal(1)) psum++;
538 if(sector.op(i)!=1 && !nv[i].is_zero()) nsum++;
540 if(psum<=1 && nsum<=1) {
543 cons_vec[con].append(sol);
546 }
else ibps.append(sol);
553cout << cons << endl << endl;
555 for(
auto & item : cons) {
557 if(kv.second==0)
continue;
558 if(item.op(kv.first)!=kv.second)
goto done;
560 cout << endl << item << endl << cons_vec[item] << endl << endl;
571 for(
auto & k : cons) {
573 cons_vec[k].unique();
575 cout << cons_vec[k] << endl << endl;
577 cout << endl << cons << endl;
585 for(
auto ii :
Internal) InExternal.append(ii);
586 for(
auto ii :
External) InExternal.append(ii);
590 for(
auto ii : InExternal)
ISP.append(it*ii);
596 if(
ISP.nops() > pdim) {
597 cout <<
"ISP = " <<
ISP << endl;
598 cout <<
"Propagator = " <<
Propagator << endl;
599 throw Error(
"Solver::IBP: #(ISP) > #(Propagator).");
604 for(
auto item :
ISP) {
606 Symbol si(
"P"+to_string(_pic));
608 sp2s.append(
w*item==
w*si);
609 sp2s.append(item==si);
610 s2sp.append(si==item);
614 for(
int i=0; i<
ISP.nops(); i++) {
618 if(eq.has(iWF(
w)))
throw Error(
"Solver::IBP, iWF used in eq.");
619 leqns.append(eq == iWF(i));
621 auto s2p = lsolve(leqns, ss);
622 if(s2p.nops() !=
ISP.nops())
throw Error(
"Solver::IBP, lsolve failed.");
628 for(
auto p2 : InExternal)
629 DSP.append(lst{p1,p2});
635 for(
int i=0; i<Internal.nops(); i++)
636 DSP.append(lst{Internal.op(i), Internal.op(i==Internal.nops()-1 ? 0 : i+1)});
637 for(
auto p2 : InExternal) DSP.append(lst{Internal.op(0), p2});
638 DSP.append(lst{Internal, Internal});
643 for(
int i=0; i<pdim; i++) nsa.append(
a(i));
650 for(
int i=0; i<pdim; i++) {
652 ns.let_op(i) = nsa.op(i) + 1;
653 auto dp = Propagator.op(i).subs(ilp==ss).diff(ss).subs(ss==ilp);
654 ibp -= (
a(i)+Shift[i]) * F(ns) * dp;
659 ibp = ibp.subs(sp2s);
660 ibp = ibp.subs(Replacement);
664 for(
int i=0; i<pdim; i++) {
665 auto ci = ibp.coeff(iWF(i), 1);
666 ci = MapFunction([i](
const ex &e, MapFunction &self)->ex {
667 if(!e.has(F(
w)))
return e;
668 else if(e.match(F(
w))) {
669 lst tmp = ex_to<lst>(e.op(0));
670 tmp.let_op(i) = tmp.op(i)-1;
672 }
else return e.map(self);
676 res += ibp.subs(lst{iWF(
w)==0});
677 auto cilp = iep.coeff(ilp);
678 if(!is_zero(cilp)) res +=
d*cilp*F(nsa);
683 void Solver::Solve(
const ex & sector,
const map<int,int> & n2n) {
684 int pdim = Propagator.nops();
696 for(
int i=0; i<pdim; i++) {
697 for(
int j=0; j<n; j++) ibps.append(ibps.op(j).subs(
a(i)==
a(i)+1));
704 for(
int i=0; i<pdim; i++) {
705 for(
int j=0; j<n; j++) ibps.append(ibps.op(j).subs(
a(i)==
a(i)-1));
712 find(ibps, F(
w1,
w2), fs);
718 vector<int> nfix(pdim);
719 for(
int i=0; i<pdim; i++) nfix[i] = 0;
720 for(
auto const & kv : n2n) nfix[kv.first] = 1;
722 for(
auto const & kv : n2n) {
724 if(k<0) k = pdim + k;
725 a2n[
a(k)] = kv.second;
727 int nibps = ibps.nops();
728 for(
int i=0; i<nibps; i++) ibps.let_op(i) = ibps.op(i).subs(a2n);
731 map<ex,lst,ex_is_less> cons_vec;
732 while(ibps.nops()>0) {
738 find(ibps,F(
w),fset);
739 int nmax = -10000000, pmax = -10000000;
740 for(
auto fi : fset) {
741 int psum = 0, nsum = 0;
743 for(
int i=0; i<pdim; i++) {
744 auto ni = ns.op(i).subs(
a(i)==0,
nopat);
745 if(sector.op(i)==1) {
746 if(nfix[i]==1 && ni<=0) psum -= 100000;
747 else psum +=
ex2int(ni)-1;
749 if(nfix[i]==1 && ni>0) nsum +=
ex2int(ni);
754 if(psum+nsum<pmax+nmax)
continue;
755 if(psum+nsum>pmax+nmax) {
765 int nr = ibps.nops();
766 int nc = fvec.size();
770 cout <<
"+----------------------------------------" << endl;
771 cout <<
"| IBPs: " << nr <<
", Fs: " << nc << endl;
772 cout <<
"+----------------------------------------" << endl;
776 for(
int r=0; r<nr; r++) {
777 auto ibp = ibps.op(r);
778 for(
int c=0; c<nc; c++) {
780 ibp = ibp.subs(fvec[c]==0,
nopat);
781 mat(r,c) = (mat(r,c)-ibp).coeff(fvec[c]);
787 auto rt = RowReduce(mat);
798 for(
int c=0; c<nc; c++) {
799 if(mr(r,c).is_zero())
continue;
800 if(mr(r,c)==1 && cc==-1) cc = c;
805 auto ns = fvec[cc].op(0).
subs(
a(
w)==0);
807 for(
int i=0; i<pdim; i++) {
808 if(nfix[i]==0) amap[
a(i)] =
a(i)-ns.op(i);
810 fvec[cc] = fvec[cc].subs(amap,
nopat);
811 auto sol = bvec(r,0).subs(amap,
nopat);
815 for(
int i=0; i<pdim; i++) {
818 if(sector.op(i)==1 && nv[i]<1)
goto next_row;
819 else if(sector.op(i)!=1 && nv[i]>0)
goto next_row;
820 }
else nv[i] = (sector.op(i)==1 ? 1 : 0);
823 ex num = sol.numer();
826 for(
auto const & f : fset) {
828 ns = f.op(0).subs(
a(
w)==0);
829 for(
int i=0; i<pdim; i++) {
830 if(sector.op(i)!=1) {
833 if(nsi>0)
goto next_row;
839 ex cci = cc.subs(
a(i)==1+nsi);
840 if(!cci.is_zero() || nsi>=nv[i])
break;
843 if(nsi<nv[i]) nv[i] = nsi;
848 ex den = sol.denom();
849 if(!is_a<mul>(den)) den = lst{den};
850 for(
const auto & item : den) {
852 if(is_a<power>(it)) it = it.op(0);
853 if(!it.has(
a(
w)) || it.has(
d))
continue;
857 cout <<
"size is NOT 1: " << it << endl;
858 throw Error(
"Solver::Solve, error occured.");
860 ex aw = *(
as.begin());
861 if(it.degree(aw)!=1) {
862 cout <<
"degree is NOT 1: " << it << endl;
863 throw Error(
"Solver::Solve, error occured.");
865 ex a0 = -it.coeff(aw,0)/it.coeff(aw,1);
866 int an =
ex2int(aw.op(0));
867 if(sector.op(an)!=1) {
868 if(a0<=nv[an]) nv[an] = a0-1;
870 if(a0>=nv[an]) nv[an] = a0+1;
874 for(
int i=0; i<pdim; i++) {
875 if(nfix[i]==1 && fvec[cc].op(0).op(i)!=nv[i]) {
876 cout << fvec << endl << nv << endl;
877 throw Error(
"Solver::Solve, error occured.");
878 }
else if(nfix[i]==0 && fvec[cc].op(0).op(i)!=
a(i)) {
879 cout << fvec << endl << nv << endl;
880 throw Error(
"Solver::Solve, error occured.");
885 int psum = 0, nsum = 0;
886 for(
int i=0; i<pdim; i++) {
887 if(nfix[i]==1)
continue;
888 if(sector.op(i)==1 && !nv[i].is_equal(1)) psum++;
889 if(sector.op(i)!=1 && !nv[i].is_zero()) nsum++;
891 if(nsum<=1 && psum<=1) con = lst{
vec2lst(nv), sol };
892 }
else if(!bvec(r,0).is_zero()) ibp = bvec(r,0).numer();
894 return lst{ibp, con};
898 for(
int r=0; r<nr; r++) {
899 auto const & item = ibp_con_sol_vec[r];
900 if(!item.op(0).is_zero()) ibps.append(item.op(0));
901 if(is_a<lst>(item.op(1))) {
902 cons.append(item.op(1).op(0));
903 cons_vec[item.op(1).op(0)].append(item.op(1).op(1));
909cout << cons << endl << endl;
917 for(
auto & k : cons) {
919 cons_vec[k].unique();
921 cout << cons_vec[k] << endl << endl;
923 cout << endl << cons << endl;
929 void Solver::Export() {
944 void Solver::Import() {
class used to wrap error message
interface to communicate with Fermat program
class to parse for string or file, helpful with Symbol class
ex Read(const string &instr, bool s2S=true)
void SolveSparse(const ex §or, const map< int, int > &a2n={})
class extended to GiNaC symbol class, represent a positive symbol
ex exfactor(const ex &expr_in, int opt)
factorize a expression
ex sp(const ex &a, const ex &b)
translated the vector dot a.b to a*b, useful in SecDec
ex factor_flint(const ex &e, bool nd=true)
ex collect_ex(ex const &expr_in, std::function< bool(const ex &)> has_func, int opt)
the collect function like Mathematica
map< int, map< int, ex > > SparseMatrix
ex normal_flint(const ex &expr, int opt=o_flint)
lst collect_lst(ex const &expr_in, std::function< bool(const ex &)> has_func, int opt)
the collect function like Mathematica, reture the lst { {c1,v1}, {c2,v2}, ... }
void sort_lst(lst &ilst, bool less=true)
sort the list in less order, or the reverse
lst vec2lst(const exvector &ev)
convert exvector to lst
exvector GiNaC_Parallel(int ntotal, std::function< ex(int)> f, const string &key)
GiNaC Parallel Evaluation using fork.
void string_trim(string &str)
int ex2int(ex num)
ex to integer
ex subs(const ex &e, init_list sl)