HepLib
Loading...
Searching...
No Matches
Others.cpp
Go to the documentation of this file.
1
6#include "HEP.h"
7#include <cmath>
8
9namespace HepLib {
10
11 namespace {
12 int ep_ldegree(const ex & e) {
13 auto ee = exnormal(e);
14 auto nd = ee.numer_denom();
15 if(nd.op(0).is_polynomial(ep) && nd.op(1).is_polynomial(ep)) {
16 return nd.op(0).ldegree(ep) - nd.op(1).ldegree(ep);
17 }
18 cout << "numer or denom is NOT a polynormial of ep." << endl;
19 abort();
20 }
21
22
23 class Family {
24 public:
25 int pn;
26 lst intg;
27 Family(int pn_) : pn(pn_) { }
28 };
29
30 exmap _sp2x_(const ex & ips, const ex & eps) {
31 int ni = 0;
32 exmap sp2x;
33 for(auto ip : ips) {
34 for(auto ipx : ips) {
35 if(sp2x.find(ip*ipx)==sp2x.end()) {
36 auto xi = x(ni);
37 sp2x[ip*ipx] = xi;
38 sp2x[w*ip*ipx] = w*xi;
39 ni++;
40 }
41 }
42 for(auto ep : eps) {
43 auto xi = x(ni);
44 sp2x[ep*ip] = xi;
45 sp2x[w*ep*ip] = w*xi;
46 ni++;
47 }
48 }
49 return sp2x;
50 }
51
52 bool is_complete(const ex & props, const ex & ips, const ex & eps) {
53 auto sp2x = _sp2x_(ips, eps);
54 auto ps = subs(props, sp2x);
55 int n = ps.nops();
56 matrix mat(n,n);
57 for(int r=0; r<n; r++) {
58 auto xi = x(r);
59 for(int c=0; c<n; c++) {
60 mat(r,c) = ps.op(c).coeff(xi);
61 }
62 }
63 return mat.rank()==n;
64 }
65 bool is_complete(const ex & props, const exmap & sp2x) {
66 auto ps = subs(props, sp2x);
67 int n = ps.nops();
68 matrix mat(n,n);
69 for(int r=0; r<n; r++) {
70 auto xi = x(r);
71 for(int c=0; c<n; c++) {
72 mat(r,c) = ps.op(c).coeff(xi);
73 }
74 }
75 return mat.rank()==n;
76 }
77 }
78
79 bool Exp2AMF::is_loop(const ex & e) {
80 for(auto ei : Internal) if(ei.is_equal(e)) return true;
81 return false;
82 }
83
84 void Exp2AMF::Export(const ex & expr, const string & dir) {
85 static string wlo = R"EOF(
86current = If[$FrontEnd===Null,$InputFileName,NotebookFileName[]]//DirectoryName;
87<<AMFlow`AMFlow`
88SetReductionOptions["IBPReducer"->"FiniteFlow+LiteRed"];
89ID = ToExpression[$ScriptCommandLine[[2]]];
90PropsInts=Get[FileNameJoin[{current,ToString[ID]<>".m"}]];
91
92AMFlowInfo["Family"] = Symbol["I"<>ToString[ID]];
93AMFlowInfo["Loop"] = <<Loop>>;
94AMFlowInfo["Leg"] = <<Leg>>;
95AMFlowInfo["Conservation"] = { };
96AMFlowInfo["Replacement"] = <<Replacement>>;
97AMFlowInfo["Propagator"] = PropsInts[[1]]/.<<M2M2>>;
98AMFlowInfo["Numeric"] = <<Numeric>>;
99AMFlowInfo["NThread"] = 6;
100
101integrals = Map[(j[Symbol["I"<>ToString[ID]], Sequence@@#])&, PropsInts[[2]]];
102precision = <<Precision>>;
103epsorder = 2*Length@AMFlowInfo["Loop"]-PropsInts[[3]] + <<Order>>;
104sol = SolveIntegrals[integrals, precision, epsorder];
105Put[sol, FileNameJoin[{current, "sol"<>ToString[ID]<>".m"}]];
106
107Quit[];
108)EOF";
109
110 string sLoop = ex2str(Internal);
111 string_replace_all(sLoop, "==", "->");
112 string sLeg = ex2str(External);
113 string_replace_all(sLeg, "==", "->");
114 string sReplacement = ex2str(Replacement);
115 string_replace_all(sReplacement, "==", "->");
116 string sNumeric = ex2str(Numeric);
117 string_replace_all(sNumeric, "==", "->");
118 string sM2M2 = ex2str(M2M2);
119 string_replace_all(sM2M2, "==", "->");
120 string wl = wlo;
121 string_replace_all(wl, "<<Loop>>", sLoop);
122 string_replace_all(wl, "<<Leg>>", sLeg);
123 string_replace_all(wl, "<<Replacement>>", sReplacement);
124 string_replace_all(wl, "<<Numeric>>", sNumeric);
125 string_replace_all(wl, "<<M2M2>>", sM2M2);
126 string_replace_all(wl, "<<Order>>", ex2str(Order));
127 string_replace_all(wl, "<<Precision>>", ex2str(Precision));
128
129 system(("mkdir -p "+dir).c_str());
130 str2file(wl, dir+"/run.wl");
131
132 auto res = expr;
133
134 // handle (qi*pi)^-n
135 cout << "Replacing linear propagators ..." << endl;
136 sp2x = _sp2x_(Internal, External);
137 exset pq_fs;
138 find(res, F(w1,w2), pq_fs);
139 exmap pq_f2f;
140 for(auto f : pq_fs) {
141 lst ps = ex_to<lst>(f.op(0));
142 lst ns = ex_to<lst>(f.op(1));
143 lst ns_p; // problematic ns
144 for(int i=0; i<ps.nops(); i++) {
145 bool ok = false;
146 for(auto ip : Internal) {
147 if(ps.op(i).has(ip*ip)) {
148 ok = true;
149 break;
150 }
151 }
152 if(!ok) {
153 ex pi = ps.op(i);
154 if(!is_a<mul>(pi) || !pi.nops()==2) {
155 cout << "prop is NOT the form of a*b" << endl;
156 abort();
157 }
158
159 if(ns.op(i)==-1) {
160 ns_p.append(i);
161 } else if(ns.op(i)!=0) {
162 cout << f << endl;
163 cout << "n of qi.pi is NOT -1" << endl;
164 abort();
165 } else { // ns.op(i)==0
166 auto pa = pi.op(0), pb = pi.op(1);
167 ex pab = expand(pow(pa+pb,2)).subs(Replacement);
168 ps.let_op(i) = pab;
169 bool ic = is_complete(ps, sp2x);
170 if(!ic && is_loop(pa)) {
171 ps.let_op(i) = expand(pow(pa,2)).subs(Replacement);
172 ic = is_complete(ps, sp2x);
173 }
174 if(!ic && is_loop(pb)) {
175 ps.let_op(i) = expand(pow(pb,2)).subs(Replacement);
176 ic = is_complete(ps, sp2x);
177 }
178 if(!ic) {
179 cout << "1st: ic is NOT true!" << endl;
180 abort();
181 }
182 }
183 }
184 }
185 if(ns_p.nops()>1) {
186 cout << f << endl;
187 abort();
188 } else if(ns_p.nops()>0) {
189 auto idx = ex2int(ns_p.op(0));
190 ns.let_op(idx) = 0;
191 auto pidx = ps.op(idx);
192
193 ex pa = pidx.op(0);
194 ex pb = pidx.op(1);
195
196 ex pab = expand(pow(pa+pb,2)).subs(Replacement);
197 ps.let_op(idx) = pab;
198 bool ic = is_complete(ps, sp2x);
199 if(!ic && is_loop(pa)) {
200 ps.let_op(idx) = expand(pow(pa,2)).subs(Replacement);
201 ic = is_complete(ps, sp2x);
202 }
203 if(!ic && is_loop(pb)) {
204 ps.let_op(idx) = expand(pow(pb,2)).subs(Replacement);
205 ic = is_complete(ps, sp2x);
206 }
207 if(!ic) {
208 cout << "2nd: ic is NOT true!" << endl;
209 abort();
210 }
211
212 auto xps = subs(ps,sp2x);
213 auto xpidx = pidx.subs(sp2x);
214 for(auto ip : Internal) {
215 if(xps.has(ip)) {
216 cout << "xps still has ip" << endl;
217 cout << "xps: " << xps << endl;
218 abort();
219 }
220 if(xpidx.has(ip)) {
221 cout << "xpidx still has ip" << endl;
222 cout << "xpidx: " << xpidx << endl;
223 abort();
224 }
225 }
226
227 ex xeq = 0, eq = 0, fres = 0;
228 lst ys;
229 for(int ii=0; ii<ps.nops(); ii++) {
230 symbol yi;
231 xeq += yi*xps.op(ii);
232 eq += yi*ps.op(ii);
233 lst nn = ns;
234 nn.let_op(ii) = ns.op(ii)-1;
235 fres += yi*F(ps, nn);
236 ys.append(yi);
237 }
238 lst eqs;
239 for(int ii=0; ii<ps.nops(); ii++) {
240 auto xi = x(ii);
241 eqs.append(xeq.coeff(xi)==xpidx.coeff(xi));
242 }
243 auto sol = lsolve(eqs,ys);
244
245 if(sol.nops()<1 || sol.has(x(w))) {
246 cout << "wrong sol: " << sol << endl;
247 }
248
249 auto rem = pidx-eq.subs(sol);
250 fres = fres.subs(sol);
251 if(!is_zero(rem)) {
252 fres += rem * F(ps, ns);
253 }
254
255 pq_f2f[f] = fres;
256 ex chk = exnormal(F2ex(f-pq_f2f[f]));
257 if(chk!=0) {
258 cout << "Failed: chk=" << chk << endl;
259 }
260 } else if(!f.op(0).is_equal(ps)) {
261 pq_f2f[f] = F(ps, ns);
262 ex chk = exnormal(F2ex(f-pq_f2f[f]));
263 if(chk!=0) {
264 cout << "Failed: chk=" << chk << endl;
265 }
266 }
267 }
268
269 cout << "Collecting ..." << endl;
270
271 res = MapFunction([&](const ex & e, MapFunction &self)->ex{
272 if(e.match(F(w1, w2))) {
273 auto itr = pq_f2f.find(e);
274 if(itr==pq_f2f.end()) {
275 cout << "F Not Found" << endl;
276 abort();
277 }
278 return itr->second;
279 } else return e.map(self);
280 })(res);
281
282 res = collect_ex(res, F(w1,w2));
283
284 exset fs;
285 find(res, F(w1,w2), fs);
286 map<ex,Family,ex_is_less> p2f;
287 exmap f2f;
288 int pn = 0;
289 for(auto f : fs) {
290 auto itr = p2f.find(f.op(0));
291 if(itr == p2f.end()) {
292 itr = p2f.insert({f.op(0), Family(pn)}).first;
293 pn++;
294 }
295 itr->second.intg.append(f.op(1));
296 f2f[f] = F(itr->second.pn, f.op(1));
297 }
298
299 res = MapFunction([&](const ex & e, MapFunction &self)->ex{
300 if(e.match(F(w1, w2))) {
301 auto itr = f2f.find(e);
302 if(itr==f2f.end()) {
303 cout << "F Not Found" << endl;
304 abort();
305 }
306 return itr->second;
307 } else return e.map(self);
308 })(res);
309
310 cout << "Exporting res.txt ..." << endl;
311 ex2file(res.subs(Numeric), dir+"/res.txt");
312
313 string sres = R"EOF(
314current = If[$FrontEnd===Null,$InputFileName,NotebookFileName[]]//DirectoryName;
315<<HepLib`
316Module[{sols, res},
317sols = Flatten@Table[Import[FileNameJoin[{current,"sol"<>ToString[i]<>".m"}]], {i,0,<<NN>>}];
318res = C2M@Import[FileNameJoin[{current, "res.txt"}]];
319res = res/.{F[p_, n_] :> j[Symbol["I"<>ToString[p]], Sequence@@n]} /. sols;
320res//Print
321]
322)EOF";
323
324 string_replace_all(sres, "<<NN>>", to_string(pn-1));
325 str2file(sres, dir+"/res.m");
326
327 int tot = p2f.size(), cur = 0;
328 for(auto kv : p2f) {
329 cur++;
330 cout << "[ " << cur << " / " << tot << "] pn = " << kv.second.pn << endl;
331 cout << kv.first << endl;
332 cout << kv.second.intg << endl;
333 cout << endl;
334
335 lst prop = ex_to<lst>(kv.first);
336 auto & f = kv.second;
337 for(int i=0; i<prop.nops(); i++) {
338 bool ok = false;
339 for(auto ip : Internal) {
340 if(prop.op(i).has(ip*ip)) {
341 ok = true;
342 break;
343 }
344 }
345 if(!ok) {
346 cout << "Propagators: " << prop.op(i) << " @ pos: " << i << endl;
347 cout << "Integrals: " << f.intg << endl;
348 abort();
349 }
350 }
351
352 GiNaC_Parallel_Verb["Exp2AMF"] = 0;
353 auto order_vec = GiNaC_Parallel(f.intg.nops(), [&](int idx)->ex{
354 auto ns = f.intg.op(idx);
355 ex cc = res.coeff(F(f.pn, ns)).subs(Numeric);
356 cc = cc.subs(d==4-2*ep);
357 auto ldeg = ep_ldegree(cc);
358 return ldeg;
359 }, "Exp2AMF");
360
361 int order = 0;
362 for(auto ldeg : order_vec) {
363 if(order>ldeg) order = ex2int(ldeg);
364 }
365
366 ex2file(lst{ prop, f.intg, order }, dir+"/"+to_string(f.pn)+".m");
367 }
368
369 }
370
371}
HEP header file.
int pn
Definition Others.cpp:25
lst intg
Definition Others.cpp:26
lst Replacement
Definition HEP.h:643
lst Internal
Definition HEP.h:641
int Precision
Definition HEP.h:647
void Export(const ex &expr, const string &dir)
Definition Others.cpp:84
lst External
Definition HEP.h:642
lst Numeric
Definition HEP.h:644
class to wrap map_function of GiNaC
Definition BASIC.h:632
HepLib namespace.
Definition BASIC.cpp:17
void ex2file(const ex &expr, string filename)
export expression file
Definition BASIC.cpp:845
const Symbol ep
ex collect_ex(ex const &expr_in, std::function< bool(const ex &)> has_func, int opt)
the collect function like Mathematica
Definition BASIC.cpp:1202
const Symbol eps
ex F2ex(const ex &expr_in)
convert F(ps, ns) to normal ex, ns is like FIRE convention
Definition ApartIBP.cpp:87
ex w
Definition Init.cpp:93
map< string, int > GiNaC_Parallel_Verb
Definition Init.cpp:148
ex exnormal(const ex &expr, int opt)
normalize a expression
Definition BASIC.cpp:1916
void string_replace_all(string &str, const string &from, const string &to)
string ex2str(const ex &expr)
convert ex to output string, the defalut printer format will be used
Definition BASIC.cpp:715
void str2file(const string &ostr, string filename)
export string to a file
Definition BASIC.cpp:787
ex w1
Definition BASIC.h:499
exvector GiNaC_Parallel(int ntotal, std::function< ex(int)> f, const string &key)
GiNaC Parallel Evaluation using fork.
Definition BASIC.cpp:260
ex w2
Definition BASIC.h:499
int ex2int(ex num)
ex to integer
Definition BASIC.cpp:893
ex subs(const ex &e, init_list sl)
Definition BASIC.h:51