From 94f581284a7876d3215b02aa63004564600e4f8c Mon Sep 17 00:00:00 2001
From: Giovanni La Mura <giovanni.lamura@inaf.it>
Date: Thu, 26 Oct 2023 17:51:36 +0200
Subject: [PATCH] Introduce FORTRAN/C++ correspondence logs

---
 src/include/sph_subs.h | 133 +++++++++++++++++++++++++++++++++++++++++
 src/sphere/sph.f       |   1 +
 src/sphere/sphere.cpp  |   7 ++-
 3 files changed, 138 insertions(+), 3 deletions(-)

diff --git a/src/include/sph_subs.h b/src/include/sph_subs.h
index 411a8ec0..081b13ac 100644
--- a/src/include/sph_subs.h
+++ b/src/include/sph_subs.h
@@ -49,6 +49,139 @@ struct C2 {
 	double vsz[2];
 };
 
+/*! \brief Conjugate of a double precision complex number
+ *
+ * \param z: `std::complex\<double\>` The input complex number.
+ * \return result: `std::complex\<double\>` The conjugate of the input number.
+ */
+std::complex<double> dconjg(std::complex<double> z) {
+	double zreal = z.real();
+	double zimag = z.imag();
+	return std::complex<double>(zreal, -zimag);
+}
+
+/*! \brief C++ porting of CG1
+ *
+ * \param li: `int`
+ * \param mu: `int`
+ * \param l: `int`
+ * \param m: `int`
+ * \return result: `double`
+ */
+double cg1(int lmpml, int mu, int l, int m) {
+	double result = 0.0;
+	double xd, xn;
+	if (lmpml == -1) { // Interpreted as GOTO 30
+		xd = 2.0 * l * (2 * l - 1);
+		if (mu == -1) {
+			xn = 1.0 * (l - 1 - m) * (l - m);
+		} else if (mu == 0) {
+			xn = 2.0 * (l - m) * (l + m);
+		} else if (mu == 1) {
+			xn = 1.0 * (l - 1 + m) * (l + m);
+		} else {
+			throw 111; // Need an exception for unpredicted indices.
+		}
+		result = sqrt(xn / xd);
+	} else if (lmpml == 0) { // Interpreted as GOTO 5
+		bool goto10 = (m != 0) || (mu != 0);
+		if (!goto10) {
+			result = 0.0;
+			return result;
+		}
+		if (mu != 0) {
+			xd = 2.0 * l * (l + 1);
+			if (mu == -1) {
+				xn = 1.0 * (l - m) * (l + m + 1);
+				result = -sqrt(xn / xd);
+			} else if (mu == 1) { // mu > 0
+				xn = 1.0 * (l + m) * (l - m + 1);
+				result = sqrt(xn / xd);
+			} else {
+				throw 111; // Need an exception for unpredicted indices.
+			}
+		} else { // mu = 0
+			xd = 1.0 * l * (l + 1);
+			xn = -1.0 * m;
+			result = xn / sqrt(xd);
+		}
+	} else if (lmpml == 1) { // Interpreted as GOTO 60
+		xd = 2.0 * (l * 2 + 3) * (l + 1);
+		if (mu == -1) {
+			xn = 1.0 * (l + 1 + m) * (l + 2 + m);
+			result = sqrt(xn / xd);
+		} else if (mu == 0) {
+			xn = 2.0 * (l + 1 - m) * (l + 1 + m);
+			result = -sqrt(xn / xd);
+		} else if (mu == 1) {
+			xn = 1.0 * (l + 1 - m) * (l + 2 - m);
+			result = sqrt(xn / xd);
+		} else { // mu was not recognized.
+			throw 111; // Need an exception for unpredicted indices.
+		}
+	} else { // lmpml was not recognized
+		throw 111; // Need an exception for unpredicted indices.
+	}
+	return result;
+}
+
+/*! \brief C++ porting of APS
+ *
+ * \param zpv: `double ****`
+ * \param li: `int`
+ * \param nsph: `int`
+ * \param c1: `C1 *`
+ * \param sqk: `double`
+ * \param gaps: `double *`
+ */
+void aps(double ****zpv, int li, int nsph, C1 *c1, double sqk, double *gaps) {
+	std::complex<double> cc0 = std::complex<double>(0.0, 0.0);
+	std::complex<double> summ, sume, suem, suee, sum;
+	double half_pi = acos(0.0);
+	double cofs = half_pi * 2.0 / sqk;
+	for (int i40 = 1; i40 <= nsph; i40++) {
+		int iogi = c1->iog[i40 - 1];
+		if (iogi >= i40) {
+			sum = cc0;
+			for (int l30 = 1; l30 <= li; l30++) {
+				int ltpo = l30 + l30 + 1;
+				for (int ilmp = 1; ilmp <= 3; ilmp++) {
+					bool goto30 = (l30 == 1 && ilmp == 1) || (l30 == li && ilmp == 3);
+					if (!goto30) {
+						int lmpml = ilmp - 2;
+						int lmp = l30 + lmpml;
+						double cofl = sqrt(1.0 * (ltpo * (lmp + lmp + 1)));
+						summ = zpv[l30 - 1][ilmp - 1][0][0] /
+								(
+										dconjg(c1->rmi[l30 - 1][i40 - 1]) *
+										c1->rmi[lmp - 1][i40 - 1]
+								);
+						sume = zpv[l30 - 1][ilmp - 1][0][1] /
+								(
+										dconjg(c1->rmi[l30 - 1][i40 - 1]) *
+										c1->rei[lmp - 1][i40 - 1]
+								);
+						suem = zpv[l30 - 1][ilmp - 1][1][0] /
+								(
+										dconjg(c1->rei[l30 - 1][i40 - 1]) *
+										c1->rmi[lmp - 1][i40 - 1]
+								);
+						suee = zpv[l30 - 1][ilmp - 1][1][1] /
+								(
+										dconjg(c1->rei[l30 - 1][i40 - 1]) *
+										c1->rei[lmp - 1][i40 - 1]
+								);
+						sum += (cg1(lmpml, 0, l30, -1) * (summ - sume - suem + suee) +
+								cg1(lmpml, 0, l30, 1) * (summ + sume + suem + suee)) * cofl;
+					}
+				}
+			}
+		}
+		gaps[i40 - 1] = sum.real() * cofs;
+		printf("DEBUG: gaps[%d] = %lE\n", i40, gaps[i40 - 1]);
+	}
+}
+
 /*! \brief C++ porting of DIEL
  *
  * \param npntmo: `int`
diff --git a/src/sphere/sph.f b/src/sphere/sph.f
index 69d8e3b3..77a6df97 100644
--- a/src/sphere/sph.f
+++ b/src/sphere/sph.f
@@ -645,6 +645,7 @@ CCC  1RMI(LI,NSPH),REI(LI,NSPH),GAPS(NSPH)
      1CG1(LMPML,0,L,1)*(SUMM+SUME+SUEM+SUEE))*COFL
    30 CONTINUE
       GAPS(I)=DREAL(SUM)*COFS
+      PRINT *,"DEBUG: GAPS(",I,") =",GAPS(I)
    40 CONTINUE
       RETURN
       END
diff --git a/src/sphere/sphere.cpp b/src/sphere/sphere.cpp
index 3f87c90d..e422603d 100644
--- a/src/sphere/sphere.cpp
+++ b/src/sphere/sphere.cpp
@@ -9,9 +9,8 @@ using namespace std;
 
 //! \brief C++ implementation of SPH
 void sphere() {
-	//complex<double> dc0[5];
-	//complex<double> dc0m[2][4];
 	complex<double> arg, s0, tfsas;
+	double gaps[2];
 	printf("INFO: making legacy configuration ...\n");
 	ScattererConfiguration *conf = ScattererConfiguration::from_dedfb("../../test_data/sphere/DEDFB");
 	conf->write_formatted("c_OEDFB");
@@ -204,7 +203,6 @@ void sphere() {
 					int iogi = sconf->iog_vec[i132 - 1];
 					if (iogi != i132) {
 						for (int l123 = 1; l123 <= gconf->l_max; l123++) {
-							// QUESTION: aren't RMI and REI still empty?
 							c1->rmi[l123 - 1][i132 - 1] = c1->rmi[l123 - 1][iogi - 1];
 							c1->rei[l123 - 1][i132 - 1] = c1->rei[l123 - 1][iogi - 1];
 						}
@@ -265,6 +263,9 @@ void sphere() {
 				double cs0 = 0.25 * vk * vk * vk / half_pi;
 				sscr0(tfsas, nsph, gconf->l_max, vk, exri, c1);
 				printf("DEBUG: TFSAS = (%lE,%lE)\n", tfsas.real(), tfsas.imag());
+				double sqk = vk * vk * sconf->exdc;
+				aps(zpv, gconf->l_max, nsph, c1, sqk, gaps);
+				// Would call RABAS
 			} //jxi488
 			tppoan.close();
 		} else { // In case TPPOAN could not be opened. Should never happen.
-- 
GitLab