You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ln_Smin_ME_ODSLF_XXZ.cc 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. /**********************************************************
  2. This software is part of J.-S. Caux's ABACUS library.
  3. Copyright (c) J.-S. Caux.
  4. -----------------------------------------------------------
  5. File: ln_Smin_ME_ODSLF_XXZ.cc
  6. Purpose: S^- matrix element
  7. ***********************************************************/
  8. #include "ABACUS.h"
  9. using namespace ABACUS;
  10. namespace ABACUS {
  11. inline complex<DP> ln_Fn_F (ODSLF_XXZ_Bethe_State& B, int k, int beta, int b)
  12. {
  13. complex<DP> ans = 0.0;
  14. complex<DP> prod_temp = 1.0;
  15. int counter = 0;
  16. int arg = 0;
  17. int absarg = 0;
  18. int par_comb_1, par_comb_2;
  19. for (int j = 0; j < B.chain.Nstrings; ++j) {
  20. par_comb_1 = B.chain.par[j] == B.chain.par[k] ? 1 : 0;
  21. par_comb_2 = B.chain.par[k] == B.chain.par[j] ? 0 : B.chain.par[k];
  22. for (int alpha = 0; alpha < B.base.Nrap[j]; ++alpha) {
  23. for (int a = 1; a <= B.chain.Str_L[j]; ++a) {
  24. if (!((j == k) && (alpha == beta) && (a == b))) {
  25. arg = B.chain.Str_L[j] - B.chain.Str_L[k] - 2 * (a - b);
  26. absarg = abs(arg);
  27. prod_temp *= ((B.sinhlambda[j][alpha] * B.coshlambda[k][beta]
  28. - B.coshlambda[j][alpha] * B.sinhlambda[k][beta])
  29. * (B.chain.co_n_anis_over_2[absarg] * par_comb_1
  30. - sgn_int(arg) * B.chain.si_n_anis_over_2[absarg] * par_comb_2)
  31. + II * (B.coshlambda[j][alpha] * B.coshlambda[k][beta]
  32. - B.sinhlambda[j][alpha] * B.sinhlambda[k][beta])
  33. * (sgn_int(arg) * B.chain.si_n_anis_over_2[absarg] * par_comb_1
  34. + B.chain.co_n_anis_over_2[absarg] * par_comb_2));
  35. }
  36. if (counter++ > 100) { // we do at most 100 products before taking a log
  37. ans += log(prod_temp);
  38. prod_temp = 1.0;
  39. counter = 0;
  40. }
  41. }}}
  42. return(ans + log(prod_temp));
  43. }
  44. inline complex<DP> ln_Fn_G (ODSLF_XXZ_Bethe_State& A, ODSLF_XXZ_Bethe_State& B, int k, int beta, int b)
  45. {
  46. complex<DP> ans = 0.0;
  47. complex<DP> prod_temp = 1.0;
  48. int counter = 0;
  49. int arg = 0;
  50. int absarg = 0;
  51. int par_comb_1, par_comb_2;
  52. for (int j = 0; j < A.chain.Nstrings; ++j) {
  53. par_comb_1 = A.chain.par[j] == B.chain.par[k] ? 1 : 0;
  54. par_comb_2 = B.chain.par[k] == A.chain.par[j] ? 0 : B.chain.par[k];
  55. for (int alpha = 0; alpha < A.base.Nrap[j]; ++alpha) {
  56. for (int a = 1; a <= A.chain.Str_L[j]; ++a) {
  57. arg = A.chain.Str_L[j] - B.chain.Str_L[k] - 2 * (a - b);
  58. absarg = abs(arg);
  59. prod_temp *= ((A.sinhlambda[j][alpha] * B.coshlambda[k][beta]
  60. - A.coshlambda[j][alpha] * B.sinhlambda[k][beta])
  61. * (A.chain.co_n_anis_over_2[absarg] * par_comb_1
  62. - sgn_int(arg) * A.chain.si_n_anis_over_2[absarg] * par_comb_2)
  63. + II * (A.coshlambda[j][alpha] * B.coshlambda[k][beta]
  64. - A.sinhlambda[j][alpha] * B.sinhlambda[k][beta])
  65. * (sgn_int(arg) * A.chain.si_n_anis_over_2[absarg] * par_comb_1
  66. + A.chain.co_n_anis_over_2[absarg] * par_comb_2));
  67. if (counter++ > 100) { // we do at most 100 products before taking a log
  68. ans += log(prod_temp);
  69. prod_temp = 1.0;
  70. counter = 0;
  71. }
  72. }}}
  73. return(ans + log(prod_temp));
  74. }
  75. inline complex<DP> Fn_K (ODSLF_XXZ_Bethe_State& A, int j, int alpha, int a,
  76. ODSLF_XXZ_Bethe_State& B, int k, int beta, int b)
  77. {
  78. int arg1 = A.chain.Str_L[j] - B.chain.Str_L[k] - 2 * (a - b);
  79. int absarg1 = abs(arg1);
  80. int arg2 = arg1 + 2;
  81. int absarg2 = abs(arg2);
  82. return(4.0/(
  83. ((A.sinhlambda[j][alpha] * B.coshlambda[k][beta] - A.coshlambda[j][alpha] * B.sinhlambda[k][beta])
  84. * (A.chain.co_n_anis_over_2[absarg1] * (1.0 + A.chain.par[j] * B.chain.par[k])
  85. - sgn_int(arg1) * A.chain.si_n_anis_over_2[absarg1] * (B.chain.par[k] - A.chain.par[j]))
  86. + II * (A.coshlambda[j][alpha] * B.coshlambda[k][beta] - A.sinhlambda[j][alpha] * B.sinhlambda[k][beta])
  87. * (sgn_int(arg1) * A.chain.si_n_anis_over_2[absarg1] * (1.0 + A.chain.par[j] * B.chain.par[k])
  88. + A.chain.co_n_anis_over_2[absarg1] * (B.chain.par[k] - A.chain.par[j])) )
  89. *
  90. ((A.sinhlambda[j][alpha] * B.coshlambda[k][beta] - A.coshlambda[j][alpha] * B.sinhlambda[k][beta])
  91. * (A.chain.co_n_anis_over_2[absarg2] * (1.0 + A.chain.par[j] * B.chain.par[k])
  92. - sgn_int(arg2) * A.chain.si_n_anis_over_2[absarg2] * (B.chain.par[k] - A.chain.par[j]))
  93. + II * (A.coshlambda[j][alpha] * B.coshlambda[k][beta] - A.sinhlambda[j][alpha] * B.sinhlambda[k][beta])
  94. * (sgn_int(arg2) * A.chain.si_n_anis_over_2[absarg2] * (1.0 + A.chain.par[j] * B.chain.par[k])
  95. + A.chain.co_n_anis_over_2[absarg2] * (B.chain.par[k] - A.chain.par[j])) )
  96. ));
  97. }
  98. inline complex<DP> Fn_L (ODSLF_XXZ_Bethe_State& A, int j, int alpha, int a,
  99. ODSLF_XXZ_Bethe_State& B, int k, int beta, int b)
  100. {
  101. return (sinh(2.0 * (A.lambda[j][alpha] - B.lambda[k][beta]
  102. + 0.5 * II * B.chain.anis * (A.chain.Str_L[j] - B.chain.Str_L[k] - 2.0 * (a - b - 0.5))
  103. + 0.25 * II * PI * complex<DP>(-A.chain.par[j] + B.chain.par[k])))
  104. * pow(Fn_K (A, j, alpha, a, B, k, beta, b), 2.0));
  105. }
  106. complex<DP> ln_Smin_ME (ODSLF_XXZ_Bethe_State& A, ODSLF_XXZ_Bethe_State& B)
  107. {
  108. // This function returns the natural log of the S^- operator matrix element.
  109. // The A and B states can contain strings.
  110. // Check that the two states are compatible
  111. if (A.chain != B.chain)
  112. ABACUSerror("Incompatible ODSLF_XXZ_Chains in Smin matrix element.");
  113. // Check that A and B are Mdown-compatible:
  114. if (A.base.Mdown != B.base.Mdown + 1)
  115. ABACUSerror("Incompatible Mdown between the two states in Smin matrix element!");
  116. // Compute the sinh and cosh of rapidities
  117. A.Compute_sinhlambda();
  118. A.Compute_coshlambda();
  119. B.Compute_sinhlambda();
  120. B.Compute_coshlambda();
  121. // Some convenient arrays
  122. ODSLF_Lambda re_ln_Fn_F_B_0(B.chain, B.base);
  123. ODSLF_Lambda im_ln_Fn_F_B_0(B.chain, B.base);
  124. ODSLF_Lambda re_ln_Fn_G_0(B.chain, B.base);
  125. ODSLF_Lambda im_ln_Fn_G_0(B.chain, B.base);
  126. ODSLF_Lambda re_ln_Fn_G_2(B.chain, B.base);
  127. ODSLF_Lambda im_ln_Fn_G_2(B.chain, B.base);
  128. complex<DP> ln_prod1 = 0.0;
  129. complex<DP> ln_prod2 = 0.0;
  130. complex<DP> ln_prod3 = 0.0;
  131. complex<DP> ln_prod4 = 0.0;
  132. for (int i = 0; i < A.chain.Nstrings; ++i)
  133. for (int alpha = 0; alpha < A.base.Nrap[i]; ++alpha)
  134. for (int a = 1; a <= A.chain.Str_L[i]; ++a)
  135. ln_prod1 += log(norm(sinh(A.lambda[i][alpha] + 0.5 * II * A.chain.anis * (A.chain.Str_L[i] + 1.0 - 2.0 * a - 1.0)
  136. + 0.25 * II * PI * (1.0 - A.chain.par[i]))));
  137. for (int i = 0; i < B.chain.Nstrings; ++i)
  138. for (int alpha = 0; alpha < B.base.Nrap[i]; ++alpha)
  139. for (int a = 1; a <= B.chain.Str_L[i]; ++a)
  140. if (norm(sinh(B.lambda[i][alpha] + 0.5 * II * B.chain.anis * (B.chain.Str_L[i] + 1.0 - 2.0 * a - 1.0)
  141. + 0.25 * II * PI * (1.0 - B.chain.par[i]))) > 100.0 * MACHINE_EPS_SQ)
  142. ln_prod2 += log(norm(sinh(B.lambda[i][alpha] + 0.5 * II * B.chain.anis * (B.chain.Str_L[i] + 1.0 - 2.0 * a - 1.0)
  143. + 0.25 * II * PI * (1.0 - B.chain.par[i]))));
  144. // Define the F ones earlier...
  145. for (int j = 0; j < B.chain.Nstrings; ++j) {
  146. for (int alpha = 0; alpha < B.base.Nrap[j]; ++alpha) {
  147. re_ln_Fn_F_B_0[j][alpha] = real(ln_Fn_F(B, j, alpha, 0));
  148. im_ln_Fn_F_B_0[j][alpha] = imag(ln_Fn_F(B, j, alpha, 0));
  149. re_ln_Fn_G_0[j][alpha] = real(ln_Fn_G(A, B, j, alpha, 0));
  150. im_ln_Fn_G_0[j][alpha] = imag(ln_Fn_G(A, B, j, alpha, 0));
  151. re_ln_Fn_G_2[j][alpha] = real(ln_Fn_G(A, B, j, alpha, 2));
  152. im_ln_Fn_G_2[j][alpha] = imag(ln_Fn_G(A, B, j, alpha, 2));
  153. }
  154. }
  155. DP logabssinzeta = log(abs(sin(A.chain.anis)));
  156. // Define regularized products in prefactors
  157. for (int j = 0; j < A.chain.Nstrings; ++j)
  158. for (int alpha = 0; alpha < A.base.Nrap[j]; ++alpha)
  159. for (int a = 1; a <= A.chain.Str_L[j]; ++a)
  160. ln_prod3 += ln_Fn_F(A, j, alpha, a - 1); // assume only one-strings here
  161. ln_prod3 -= A.base.Mdown * log(abs(sin(A.chain.anis)));
  162. for (int k = 0; k < B.chain.Nstrings; ++k)
  163. for (int beta = 0; beta < B.base.Nrap[k]; ++beta)
  164. for (int b = 1; b <= B.chain.Str_L[k]; ++b) {
  165. if (b == 1) ln_prod4 += re_ln_Fn_F_B_0[k][beta];
  166. else if (b > 1) ln_prod4 += ln_Fn_F(B, k, beta, b - 1);
  167. }
  168. ln_prod4 -= B.base.Mdown * log(abs(sin(B.chain.anis)));
  169. // Now proceed to build the Hm matrix
  170. SQMat_CX Hm(0.0, A.base.Mdown);
  171. int index_a = 0;
  172. int index_b = 0;
  173. complex<DP> sum1 = 0.0;
  174. complex<DP> sum2 = 0.0;
  175. complex<DP> prod_num = 0.0;
  176. complex<DP> Fn_K_0_G_0 = 0.0;
  177. complex<DP> Prod_powerN = 0.0;
  178. complex<DP> Fn_K_1_G_2 = 0.0;
  179. complex<DP> one_over_A_sinhlambda_sq_plus_sinzetaover2sq;
  180. for (int j = 0; j < A.chain.Nstrings; ++j) {
  181. for (int alpha = 0; alpha < A.base.Nrap[j]; ++alpha) {
  182. for (int a = 1; a <= A.chain.Str_L[j]; ++a) {
  183. index_b = 0;
  184. one_over_A_sinhlambda_sq_plus_sinzetaover2sq =
  185. 1.0/((sinh(A.lambda[j][alpha] + 0.5 * II * A.chain.anis * (A.chain.Str_L[j] + 1.0 - 2.0 * a)
  186. + 0.25 * II * PI * (1.0 - A.chain.par[j])))
  187. * (sinh(A.lambda[j][alpha] + 0.5 * II * A.chain.anis * (A.chain.Str_L[j] + 1.0 - 2.0 * a)
  188. + 0.25 * II * PI * (1.0 - A.chain.par[j])))
  189. + pow(sin(0.5*A.chain.anis), 2.0));
  190. for (int k = 0; k < B.chain.Nstrings; ++k) {
  191. for (int beta = 0; beta < B.base.Nrap[k]; ++beta) {
  192. for (int b = 1; b <= B.chain.Str_L[k]; ++b) {
  193. if (B.chain.Str_L[k] == 1) {
  194. // use simplified code for one-string here: original form of Hm matrix
  195. Fn_K_0_G_0 = Fn_K (A, j, alpha, a, B, k, beta, 0) *
  196. exp(re_ln_Fn_G_0[k][beta] + II * im_ln_Fn_G_0[k][beta] - re_ln_Fn_F_B_0[k][beta] + logabssinzeta);
  197. Fn_K_1_G_2 = Fn_K (A, j, alpha, a, B, k, beta, 1) *
  198. exp(re_ln_Fn_G_2[k][beta] + II * im_ln_Fn_G_2[k][beta] - re_ln_Fn_F_B_0[k][beta] + logabssinzeta);
  199. Prod_powerN = pow( B.chain.par[k] == 1 ?
  200. (B.sinhlambda[k][beta] * B.chain.co_n_anis_over_2[1]
  201. + II * B.coshlambda[k][beta] * B.chain.si_n_anis_over_2[1])
  202. /(B.sinhlambda[k][beta] * B.chain.co_n_anis_over_2[1]
  203. - II * B.coshlambda[k][beta] * B.chain.si_n_anis_over_2[1])
  204. :
  205. (B.coshlambda[k][beta] * B.chain.co_n_anis_over_2[1]
  206. + II * B.sinhlambda[k][beta] * B.chain.si_n_anis_over_2[1])
  207. /(B.coshlambda[k][beta] * B.chain.co_n_anis_over_2[1]
  208. - II * B.sinhlambda[k][beta] * B.chain.si_n_anis_over_2[1])
  209. , complex<DP> (B.chain.Nsites));
  210. Hm[index_a][index_b] = Fn_K_0_G_0 - (1.0 - 2.0 * (B.base.Mdown % 2)) * // MODIF from XXZ
  211. Prod_powerN * Fn_K_1_G_2;
  212. } // if (B.chain.Str_L == 1)
  213. else {
  214. if (b <= B.chain.Str_L[k] - 1) Hm[index_a][index_b] = Fn_K(A, j, alpha, a, B, k, beta, b);
  215. else if (b == B.chain.Str_L[k]) {
  216. Vect_CX ln_FunctionF(B.chain.Str_L[k] + 2);
  217. for (int i = 0; i < B.chain.Str_L[k] + 2; ++i) ln_FunctionF[i] = ln_Fn_F (B, k, beta, i);
  218. Vect_CX ln_FunctionG(B.chain.Str_L[k] + 2);
  219. for (int i = 0; i < B.chain.Str_L[k] + 2; ++i) ln_FunctionG[i] = ln_Fn_G (A, B, k, beta, i);
  220. sum1 = 0.0;
  221. sum1 += Fn_K (A, j, alpha, a, B, k, beta, 0)
  222. * exp(ln_FunctionG[0] + ln_FunctionG[1] - ln_FunctionF[0] - ln_FunctionF[1]);
  223. sum1 += (1.0 - 2.0 * (B.base.Mdown % 2)) * // MODIF from XXZ
  224. Fn_K (A, j, alpha, a, B, k, beta, B.chain.Str_L[k])
  225. * exp(ln_FunctionG[B.chain.Str_L[k]] + ln_FunctionG[B.chain.Str_L[k] + 1]
  226. - ln_FunctionF[B.chain.Str_L[k]] - ln_FunctionF[B.chain.Str_L[k] + 1]);
  227. for (int jsum = 1; jsum < B.chain.Str_L[k]; ++jsum)
  228. sum1 -= Fn_L (A, j, alpha, a, B, k, beta, jsum) *
  229. exp(ln_FunctionG[jsum] + ln_FunctionG[jsum + 1] - ln_FunctionF[jsum] - ln_FunctionF[jsum + 1]);
  230. prod_num = exp(II * im_ln_Fn_F_B_0[k][beta] + ln_FunctionF[1]
  231. - ln_FunctionG[B.chain.Str_L[k]] + logabssinzeta);
  232. for (int jsum = 2; jsum <= B.chain.Str_L[k]; ++jsum)
  233. prod_num *= exp(ln_FunctionG[jsum] - real(ln_Fn_F(B, k, beta, jsum - 1)) + logabssinzeta);
  234. // include all string contributions F_B_0 in this term
  235. Hm[index_a][index_b] = prod_num * sum1;
  236. } // else if (b == B.chain.Str_L[k])
  237. } // else
  238. index_b++;
  239. }}} // sums over k, beta, b
  240. // now define the elements Hm[a][M]
  241. Hm[index_a][B.base.Mdown] = one_over_A_sinhlambda_sq_plus_sinzetaover2sq;
  242. index_a++;
  243. }}} // sums over j, alpha, a
  244. complex<DP> ln_ME_sq = log(1.0 * A.chain.Nsites) + real(ln_prod1 - ln_prod2) - real(ln_prod3) + real(ln_prod4)
  245. + 2.0 * real(lndet_LU_CX_dstry(Hm)) + logabssinzeta - A.lnnorm - B.lnnorm;
  246. return(0.5 * ln_ME_sq); // Return ME, not MEsq
  247. }
  248. } // namespace ABACUS