From 0544d6cb4ec502847f7a28086b690d5384243737 Mon Sep 17 00:00:00 2001 From: arnab82 Date: Sat, 25 May 2024 21:58:06 -0400 Subject: [PATCH 1/3] pt2 function added for cebugging --- examples/pt2_test/_testdata_cmf_h8.jld2 | Bin 0 -> 41262 bytes examples/pt2_test/_testdata_cmf_h9.jld2 | Bin 0 -> 61554 bytes examples/pt2_test/h8.jl | 65 +++++ examples/pt2_test/h9.jl | 65 +++++ examples/tetracene_dimer/rhf.ipynb | 327 ++++++++++++++++++++++++ src/tpsci_pt2_energy.jl | 3 + src/tucker_pt.jl | 130 +++++++++- 7 files changed, 589 insertions(+), 1 deletion(-) create mode 100644 examples/pt2_test/_testdata_cmf_h8.jld2 create mode 100644 examples/pt2_test/_testdata_cmf_h9.jld2 create mode 100644 examples/pt2_test/h8.jl create mode 100644 examples/pt2_test/h9.jl create mode 100644 examples/tetracene_dimer/rhf.ipynb diff --git a/examples/pt2_test/_testdata_cmf_h8.jld2 b/examples/pt2_test/_testdata_cmf_h8.jld2 new file mode 100644 index 0000000000000000000000000000000000000000..334db8f06034492461dcf4b1da3f9c8089557dbd GIT binary patch literal 41262 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 ********\n", + "method = RHF\n", + "initial guess = minao\n", + "damping factor = 0\n", + "level_shift factor = 0\n", + "DIIS = \n", + "diis_start_cycle = 1\n", + "diis_space = 8\n", + "SCF conv_tol = 1e-09\n", + "SCF conv_tol_grad = None\n", + "SCF max_cycles = 200\n", + "direct_scf = True\n", + "direct_scf_tol = 1e-13\n", + "chkfile to save SCF result = /var/folders/jm/nd85t1m57dv98qh_jtqr2clh0000gn/T/tmpdnsov0fw\n", + "max_memory 4000 MB (current use 0 MB)\n", + "Set gradient conv threshold to 3.16228e-05\n", + "Initial guess from minao.\n", + "init E= -1373.8028556562\n", + " HOMO = -0.155360683388173 LUMO = -0.0462263741041608\n", + "cycle= 1 E= -1360.24317894862 delta_E= 13.6 |g|= 0.719 |ddm|= 11.3\n", + " HOMO = -0.143564650973908 LUMO = 0.138616575609906\n", + "cycle= 2 E= -1360.45923594589 delta_E= -0.216 |g|= 0.214 |ddm|= 1.43\n", + " HOMO = -0.16857409289756 LUMO = 0.12156954691401\n", + "cycle= 3 E= -1360.47370291225 delta_E= -0.0145 |g|= 0.126 |ddm|= 0.515\n", + " HOMO = -0.161685188896432 LUMO = 0.130732383404331\n", + "cycle= 4 E= -1360.4797695373 delta_E= -0.00607 |g|= 0.0139 |ddm|= 0.195\n", + " HOMO = -0.161052562766623 LUMO = 0.132842939686475\n", + "cycle= 5 E= -1360.47997484479 delta_E= -0.000205 |g|= 0.007 |ddm|= 0.0363\n", + " HOMO = -0.161719238589572 LUMO = 0.13333514441787\n", + "cycle= 6 E= -1360.48004918981 delta_E= -7.43e-05 |g|= 0.0028 |ddm|= 0.0275\n", + " HOMO = -0.162116206364872 LUMO = 0.13340292370013\n", + "cycle= 7 E= -1360.48005930686 delta_E= -1.01e-05 |g|= 0.000758 |ddm|= 0.0121\n", + " HOMO = -0.162172259276158 LUMO = 0.133406742943205\n", + "cycle= 8 E= -1360.48005985105 delta_E= -5.44e-07 |g|= 0.000404 |ddm|= 0.00231\n", + " HOMO = -0.162160553391006 LUMO = 0.133425925701453\n", + "cycle= 9 E= -1360.480060043 delta_E= -1.92e-07 |g|= 0.000209 |ddm|= 0.00137\n", + " HOMO = -0.162173349764571 LUMO = 0.133407030042586\n", + "cycle= 10 E= -1360.48006011093 delta_E= -6.79e-08 |g|= 7.63e-05 |ddm|= 0.0011\n", + " HOMO = -0.16217164410856 LUMO = 0.133406153044998\n", + "cycle= 11 E= -1360.48006011748 delta_E= -6.56e-09 |g|= 2.19e-05 |ddm|= 0.000359\n", + " HOMO = -0.162171319204681 LUMO = 0.133405720724885\n", + "cycle= 12 E= -1360.48006011796 delta_E= -4.82e-10 |g|= 6.26e-06 |ddm|= 9.42e-05\n", + " HOMO = -0.162171197951934 LUMO = 0.133405667622305\n", + "Extra cycle E= -1360.480060118 delta_E= -3.27e-11 |g|= 4.12e-06 |ddm|= 1.48e-05\n", + "converged SCF energy = -1360.480060118\n" + ] + } + ], + "source": [ + "from functools import reduce\n", + "import numpy as np\n", + "import scipy\n", + "\n", + "import pyscf\n", + "from pyscf import fci\n", + "from pyscf import gto, scf, ao2mo, lo, tdscf, cc\n", + "\n", + "\n", + "def tda_denisty_matrix(td, state_id):\n", + " '''\n", + " Taking the TDA amplitudes as the CIS coefficients, calculate the density\n", + " matrix (in AO basis) of the excited states\n", + " '''\n", + " cis_t1 = td.xy[state_id][0]\n", + " dm_oo =-np.einsum('ia,ka->ik', cis_t1.conj(), cis_t1)\n", + " dm_vv = np.einsum('ia,ic->ac', cis_t1, cis_t1.conj())\n", + "\n", + " # The ground state density matrix in mo_basis\n", + " mf = td._scf\n", + " dm = np.diag(mf.mo_occ)\n", + "\n", + " # Add CIS contribution\n", + " nocc = cis_t1.shape[0]\n", + " # Note that dm_oo and dm_vv correspond to spin-up contribution. \"*2\" to\n", + " # include the spin-down contribution\n", + " dm[:nocc,:nocc] += dm_oo * 2\n", + " dm[nocc:,nocc:] += dm_vv * 2\n", + "\n", + " # Transform density matrix to AO basis\n", + " mo = mf.mo_coeff\n", + " dm = np.einsum('pi,ij,qj->pq', mo, dm, mo.conj())\n", + " return dm\n", + "\n", + "mol = gto.Mole()\n", + "mol.atom = '''\n", + "C 3.86480 -1.02720 1.27180\n", + "C 3.60170 -0.34020 0.07040\n", + "C 2.93000 -1.06360 2.30730\n", + "C 4.54410 -0.28950 -0.97720\n", + "C 2.32170 0.33090 -0.09430\n", + "C 3.20030 -1.73210 3.54060\n", + "C 1.65210 -0.39240 2.14450\n", + "C 4.27330 0.38740 -2.16700\n", + "C 2.05960 1.01870 -1.29550\n", + "C 1.37970 0.28170 0.95340\n", + "C 2.27260 -1.73270 4.55160\n", + "C 0.71620 -0.41900 3.22490\n", + "C 2.99590 1.05910 -2.32920\n", + "C 5.20900 0.41800 -3.24760\n", + "C 1.01870 -1.06880 4.39470\n", + "C 2.72630 1.73410 -3.55840\n", + "C 4.90670 1.07300 -4.41500\n", + "C 3.65410 1.73950 -4.56900\n", + "C -3.86650 1.01870 -1.29550\n", + "C -2.93020 1.05910 -2.32920\n", + "C -3.60440 0.33090 -0.09430\n", + "C -3.19980 1.73410 -3.55840\n", + "C -1.65280 0.38740 -2.16700\n", + "C -2.32430 -0.34020 0.07040\n", + "C -4.54630 0.28170 0.95340\n", + "C -2.27200 1.73950 -4.56900\n", + "C -0.71710 0.41800 -3.24760\n", + "C -1.38200 -0.28950 -0.97720\n", + "C -2.06130 -1.02720 1.27180\n", + "C -4.27400 -0.39240 2.14450\n", + "C -1.01930 1.07300 -4.41500\n", + "C -2.99610 -1.06360 2.30730\n", + "C -5.20980 -0.41900 3.22490\n", + "C -2.72580 -1.73210 3.54060\n", + "C -4.90730 -1.06880 4.39470\n", + "C -3.65350 -1.73270 4.55160\n", + "H 4.82300 -1.53290 1.39770\n", + "H 5.49910 -0.80290 -0.85660\n", + "H 4.15900 -2.23700 3.66390\n", + "H 1.10180 1.52560 -1.42170\n", + "H 0.42460 0.79440 0.83100\n", + "H 2.50000 -2.24040 5.48840\n", + "H -0.23700 0.09640 3.10140\n", + "H 6.16210 -0.09790 -3.12730\n", + "H 0.29870 -1.07700 5.21470\n", + "H 1.76850 2.24120 -3.67870\n", + "H 5.62580 1.08320 -5.23570\n", + "H 3.42730 2.25190 -5.50340\n", + "H -4.82430 1.52560 -1.42170\n", + "H -4.15760 2.24120 -3.67870\n", + "H -5.50150 0.79440 0.83100\n", + "H -2.49880 2.25190 -5.50340\n", + "H 0.23610 -0.09790 -3.12730\n", + "H -0.42700 -0.80290 -0.85660\n", + "H -1.10300 -1.53290 1.39770\n", + "H -0.30030 1.08320 -5.23570\n", + "H -6.16310 0.09640 3.10140\n", + "H -1.76710 -2.23700 3.66390\n", + "H -5.62740 -1.07700 5.21470\n", + "H -3.42610 -2.24040 5.48840\n", + "'''\n", + "\n", + "\n", + "mol.basis = 'sto-3g'\n", + "mol.spin = 0\n", + "mol.build()\n", + "\n", + "#mf = scf.ROHF(mol).x2c()\n", + "mf = scf.RHF(mol)\n", + "mf.verbose = 4\n", + "mf.get_init_guess(mol, key='minao')\n", + "mf.conv_tol = 1e-9\n", + "#mf.level_shift = .1\n", + "#mf.diis_start_cycle = 4\n", + "#mf.diis_space = 10\n", + "mf.run(max_cycle=200)\n", + "\n", + "\n", + "n_triplets = 2\n", + "n_singlets = 2\n", + "\n", + "avg_rdm1 = mf.make_rdm1()\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "\n" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\n", + "\n", + "******** for ********\n", + "nstates = 2 singlet\n", + "deg_eia_thresh = 1.000e-03\n", + "wfnsym = None\n", + "conv_tol = 1e-09\n", + "eigh lindep = 1e-12\n", + "eigh level_shift = 0\n", + "eigh max_space = 50\n", + "eigh max_cycle = 100\n", + "chkfile = /var/folders/jm/nd85t1m57dv98qh_jtqr2clh0000gn/T/tmpdnsov0fw\n", + "max_memory 4000 MB (current use 0 MB)\n", + "\n", + "\n", + "Excited State energies (eV)\n", + "[4.37757462 4.50864348]\n", + "\n", + "** Singlet excitation energies and oscillator strengths **\n", + "Excited State 1: 4.37757 eV 283.23 nm f=0.7064\n", + " 117 -> 123 -0.10590\n", + " 118 -> 124 -0.10581\n", + " 119 -> 122 -0.45930\n", + " 120 -> 121 -0.49153\n", + "Excited State 2: 4.50864 eV 274.99 nm f=0.0000\n", + " 119 -> 121 0.47837\n", + " 120 -> 122 0.46727\n", + "\n", + "** Transition electric dipole moments (AU) **\n", + "state X Y Z Dip. S. Osc.\n", + " 1 2.3095 -1.1112 0.1355 6.5870 0.7064\n", + " 2 -0.0015 0.0006 0.0007 0.0000 0.0000\n", + "\n", + "** Transition velocity dipole moments (imaginary part, AU) **\n", + "state X Y Z Dip. S. Osc.\n", + " 1 -0.1016 0.0712 -0.0291 0.0162 0.0673\n", + " 2 0.0001 0.0000 0.0001 0.0000 0.0000\n", + "\n", + "** Transition magnetic dipole moments (imaginary part, AU) **\n", + "state X Y Z\n", + " 1 -0.0003 -0.0000 0.0022\n", + " 2 -0.0411 -0.1064 0.4355\n" + ] + } + ], + "source": [ + "# compute singlets\n", + "mytd = tdscf.TDA(mf)\n", + "mytd.singlet = True \n", + "mytd = mytd.run(nstates=n_singlets)\n", + "mytd.analyze()\n", + "for i in range(mytd.nroots):\n", + " avg_rdm1 += tda_denisty_matrix(mytd, i)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "# compute triplets \n", + "mytd = tdscf.TDA(mf)\n", + "mytd.singlet = False \n", + "mytd = mytd.run(nstates=n_triplets)\n", + "mytd.analyze()\n", + "for i in range(mytd.nroots):\n", + " avg_rdm1 += tda_denisty_matrix(mytd, i)\n", + "\n", + "# normalize\n", + "avg_rdm1 = avg_rdm1 / (n_singlets + n_triplets + 1)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "S = mf.get_ovlp()\n", + "F = mf.get_fock()\n", + "np.save(\"fock_mat18\", F)\n", + "np.save(\"overlap_mat18\", S)\n", + "np.save(\"density_mat18\", mf.make_rdm1())\n", + "np.save(\"rhf_mo_coeffs18\", mf.mo_coeff)\n", + "np.save(\"cis_sa_density_mat18\", avg_rdm1)" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": ".venv", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.9.6" + } + }, + "nbformat": 4, + "nbformat_minor": 2 +} diff --git a/src/tpsci_pt2_energy.jl b/src/tpsci_pt2_energy.jl index 2d3db0d..9e5c9fa 100644 --- a/src/tpsci_pt2_energy.jl +++ b/src/tpsci_pt2_energy.jl @@ -129,6 +129,9 @@ function compute_pt2_energy(ci_vector_in::TPSCIstate{T,N,R}, cluster_ops, cluste tmp = Int(round(length(jobs_vec)/100)) + if tmp == 0 + tmp += 1 + end verbose < 1 || println(" |----------------------------------------------------------------------------------------------------|") verbose < 1 || println(" |0% 100%|") verbose < 1 || print(" |") diff --git a/src/tucker_pt.jl b/src/tucker_pt.jl index a35b468..c253a42 100644 --- a/src/tucker_pt.jl +++ b/src/tucker_pt.jl @@ -735,7 +735,7 @@ function compute_pt2_energy(ref::BSTstate{T,N,R}, cluster_ops, clustered_ham; #for (jobi,job) in collect(enumerate(jobs_vec)) fock_sig = job[1] tid = Threads.threadid() - e2_thread[tid] .+= _pt2_job(fock_sig, job[2], ref_vec, cluster_ops, clustered_ham, clustered_ham_0, + e2_thread[tid] .+= _pt2_job2(fock_sig, job[2], ref_vec, cluster_ops, clustered_ham, clustered_ham_0, nbody, verbose, thresh_foi, max_number, E0, F0, prescreen, compress_twice) if verbose > 1 if jobi%tmp == 0 @@ -929,3 +929,131 @@ end +function _pt2_job2(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_ham, clustered_ham_0, + nbody, verbose, thresh, max_number, E0, F0, prescreen, compress_twice) where {T,N,R} + + ecorr = [] + for jobi in job + + terms, ket_fock, ket_tconfigs = jobi + for (ket_tconfig, ket_tuck) in ket_tconfigs + sig = BSTstate(ket.clusters, ket.p_spaces, ket.q_spaces, T=T, R=R) + add_fockconfig!(sig, sig_fock) + data = OrderedDict{TuckerConfig{N},Vector{Tucker{T,N,R}}}() + + for term in terms + + length(term.clusters) <= nbody || continue + + + + + + # + # find the sig TuckerConfigs reached by applying current Hamiltonian term to ket_tconfig. + # + # For example: + # + # [(p'q), I, I, (r's), I ] * |P,Q,P,Q,P> --> |X, Q, P, X, P> where X = {P,Q} + # + # This term will couple to 4 distinct tucker blocks (assuming each of the active clusters + # have both non-zero P and Q spaces within the current fock sector, "sig_fock". + # + # We will loop over all these destination TuckerConfig's by creating the cartesian product of their + # available spaces, this list of which we will keep in "available". + # + + available = [] # list of lists of index ranges, the cartesian product is the set needed + # for current term, expand index ranges for active clusters + for ci in term.clusters + tmp = [] + if haskey(ket.p_spaces[ci.idx], sig_fock[ci.idx]) + push!(tmp, ket.p_spaces[ci.idx][sig_fock[ci.idx]]) + end + if haskey(ket.q_spaces[ci.idx], sig_fock[ci.idx]) + push!(tmp, ket.q_spaces[ci.idx][sig_fock[ci.idx]]) + end + push!(available, tmp) + end + + + # + # Now loop over cartesian product of available subspaces (those in X above) and + # create the target TuckerConfig and then evaluate the associated terms + for prod in Iterators.product(available...) + sig_tconfig = [ket_tconfig.config...] + for cidx in 1:length(term.clusters) + ci = term.clusters[cidx] + sig_tconfig[ci.idx] = prod[cidx] + end + sig_tconfig = TuckerConfig(sig_tconfig) + + # + # the `term` has now coupled our ket TuckerConfig, to a sig TuckerConfig + # let's compute the matrix element block, then compress, then add it to any existing compressed + # coefficient tensor for that sig TuckerConfig. + # + # Both the Compression and addition takes a fair amount of work. + + + check_term(term, sig_fock, sig_tconfig, ket_fock, ket_tconfig) || continue + + + if prescreen + bound = calc_bound(term, cluster_ops, + sig_fock, sig_tconfig, + ket_fock, ket_tconfig, ket_tuck, + prescreen=thresh) + bound == true || continue + end + + sig_tuck = form_sigma_block_expand(term, cluster_ops, + sig_fock, sig_tconfig, + ket_fock, ket_tconfig, ket_tuck, + max_number=max_number, + prescreen=thresh) + + + if length(sig_tuck) == 0 + continue + end + if norm(sig_tuck) < thresh + continue + end + + + #compress new addition + sig_tuck = compress(sig_tuck, thresh=thresh) + + length(sig_tuck) > 0 || continue + + #add to current sigma vector + if haskey(sig[sig_fock], sig_tconfig) + + if haskey(data, sig_tconfig) + push!(data[sig_tconfig], sig_tuck) + else + data[sig_tconfig] = [sig[sig_fock][sig_tconfig], sig_tuck] + end + else + sig[sig_fock][sig_tconfig] = sig_tuck + end + + end + + end + for (tconfig, tucks) in data + if compress_twice + sig[sig_fock][tconfig] = compress(nonorth_add(tucks), thresh=thresh) + else + sig[sig_fock][tconfig] = nonorth_add(tucks) + end + end + # Compute PT2 energy for this job + v_pt, e_pt, ecor = compute_pt1_wavefunction(sig, ket, cluster_ops, clustered_ham, clustered_ham_0, E0, F0, verbose=0) + push!(ecorr, ecor) + end + end + return sum(ecorr) + # return ecorr +end \ No newline at end of file From d9da4daa27e257b49c232c32102202f579fa7267 Mon Sep 17 00:00:00 2001 From: arnab82 Date: Sat, 25 May 2024 22:07:02 -0400 Subject: [PATCH 2/3] pt2 spt working only for ground state --- examples/pt2_test/h8.jl | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/examples/pt2_test/h8.jl b/examples/pt2_test/h8.jl index 942e8d4..77792e6 100644 --- a/examples/pt2_test/h8.jl +++ b/examples/pt2_test/h8.jl @@ -17,11 +17,10 @@ cluster_ops = FermiCG.compute_cluster_ops(cluster_bases, ints); FermiCG.add_cmf_operators!(cluster_ops, cluster_bases, ints, d1.a, d1.b); -nroots=3 +nroots=1 # TPSCI # -nroots=3 ci_vector = FermiCG.TPSCIstate(clusters, ref_fock, R=nroots) ci_vector = FermiCG.add_spin_focksectors(ci_vector) @@ -49,7 +48,7 @@ for ci in clusters push!(p_spaces, ssi) end -ci_vector = BSTstate(clusters, p_spaces, cluster_bases, R=3) +ci_vector = BSTstate(clusters, p_spaces, cluster_bases, R=1) na = 4 nb = 4 From c95c82cd20709804b9ca7a7f9b35007bcc91ab37 Mon Sep 17 00:00:00 2001 From: arnab82 Date: Thu, 30 May 2024 23:30:34 -0400 Subject: [PATCH 3/3] another version of pt2 correction added where we loop over tuckerconfig --- examples/pt2_test/h8.jl | 3 +- examples/pt2_test/h9.jl | 6 +- src/tucker_pt.jl | 316 +++++++++++++++++++++++++++++++--------- 3 files changed, 250 insertions(+), 75 deletions(-) diff --git a/examples/pt2_test/h8.jl b/examples/pt2_test/h8.jl index 77792e6..4863f79 100644 --- a/examples/pt2_test/h8.jl +++ b/examples/pt2_test/h8.jl @@ -4,9 +4,10 @@ using FermiCG using Printf using Test using JLD2 -ref_fock = FockConfig(init_fspace) + @load "_testdata_cmf_h8.jld2" +ref_fock = FockConfig(init_fspace) # Do TPS M=20 cluster_bases = FermiCG.compute_cluster_eigenbasis_spin(ints, clusters, d1, [3,3], ref_fock, max_roots=M, verbose=1); diff --git a/examples/pt2_test/h9.jl b/examples/pt2_test/h9.jl index a233c79..33b9214 100644 --- a/examples/pt2_test/h9.jl +++ b/examples/pt2_test/h9.jl @@ -29,7 +29,7 @@ ci_vector = FermiCG.add_spin_focksectors(ci_vector) display(ci_vector) etpsci, vtpsci = FermiCG.tps_ci_direct(ci_vector, cluster_ops, clustered_ham); -ept1 = FermiCG.compute_pt2_energy(vtpsci, cluster_ops, clustered_ham, thresh_foi=1e-12) +@time ept1 = FermiCG.compute_pt2_energy(vtpsci, cluster_ops, clustered_ham, thresh_foi=1e-12) @@ -57,8 +57,8 @@ FermiCG.fill_p_space!(ci_vector, na, nb) FermiCG.eye!(ci_vector) ebst, vbst = FermiCG.ci_solve(ci_vector, cluster_ops, clustered_ham) -ept2 = FermiCG.compute_pt2_energy(vbst, cluster_ops, clustered_ham, thresh_foi=1e-64) - +@time ept2 = FermiCG.compute_pt2_energy(vbst, cluster_ops, clustered_ham, thresh_foi=1e-64,prescreen = true,compress_twice = true) +@time ept2 = FermiCG.compute_pt2_energy2(vbst, cluster_ops, clustered_ham, thresh_foi=1e-64,prescreen = true,compress_twice = true) println(" PT2 - tpsci") display(ept1) println(" PT2 - bst") diff --git a/src/tucker_pt.jl b/src/tucker_pt.jl index c253a42..f701f3d 100644 --- a/src/tucker_pt.jl +++ b/src/tucker_pt.jl @@ -735,7 +735,7 @@ function compute_pt2_energy(ref::BSTstate{T,N,R}, cluster_ops, clustered_ham; #for (jobi,job) in collect(enumerate(jobs_vec)) fock_sig = job[1] tid = Threads.threadid() - e2_thread[tid] .+= _pt2_job2(fock_sig, job[2], ref_vec, cluster_ops, clustered_ham, clustered_ham_0, + e2_thread[tid] .+= _pt2_job(fock_sig, job[2], ref_vec, cluster_ops, clustered_ham, clustered_ham_0, nbody, verbose, thresh_foi, max_number, E0, F0, prescreen, compress_twice) if verbose > 1 if jobi%tmp == 0 @@ -927,28 +927,190 @@ function _pt2_job(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_ha return ecorr end +""" + compute_pt2_energy2(ref::BSTstate{T,N,R}, cluster_ops, clustered_ham; + H0 = "Hcmf", + nbody = 4, + thresh_foi = 1e-6, + max_number = nothing, + opt_ref = true, + ci_tol = 1e-6, + verbose = true) where {T,N,R} + +Directly compute the PT2 energy, in parallel, with each thread handling a single `FockConfig` at a time. +""" +function compute_pt2_energy2(ref::BSTstate{T,N,R}, cluster_ops, clustered_ham; + H0 = "Hcmf", + nbody = 4, + thresh_foi = 1e-6, + max_number = nothing, + opt_ref = true, + ci_tol = 1e-6, + verbose = 1, + prescreen = false, + compress_twice = false) where {T,N,R} + println() + println(" |...................................BST-PT2............................................") + verbose < 1 || println(" H0 : ", H0 ) + verbose < 1 || println(" nbody : ", nbody ) + verbose < 1 || println(" thresh_foi : ", thresh_foi ) + verbose < 1 || println(" max_number : ", max_number ) + verbose < 1 || println(" opt_ref : ", opt_ref ) + verbose < 1 || println(" ci_tol : ", ci_tol ) + verbose < 1 || println(" verbose : ", verbose ) + verbose < 1 || @printf("\n") + verbose < 1 || @printf(" %-50s", "Length of Reference: ") + verbose < 1 || @printf("%10i\n", length(ref)) + + lk = ReentrantLock() + + # + # Solve variationally in reference space + ref_vec = deepcopy(ref) + clusters = ref_vec.clusters + + E0 = zeros(T,R) + + if opt_ref + @printf(" %-50s\n", "Solve zeroth-order problem: ") + time = @elapsed E0, ref_vec = ci_solve(ref_vec, cluster_ops, clustered_ham, conv_thresh=ci_tol) + @printf(" %-50s%10.6f seconds\n", "Diagonalization time: ",time) + else + @printf(" %-50s", "Compute zeroth-order energy: ") + flush(stdout) + @time E0 = compute_expectation_value(ref_vec, cluster_ops, clustered_ham) + end + + # + # get E0 = <0|H0|0> + clustered_ham_0 = extract_1body_operator(clustered_ham, op_string = H0) + @printf(" %-50s", "Compute <0|H0|0>: ") + @time F0 = compute_expectation_value(ref_vec, cluster_ops, clustered_ham_0) + + if verbose > 0 + @printf(" %5s %12s %12s\n", "Root", "<0|H|0>", "<0|F|0>") + for r in 1:R + @printf(" %5s %12.8f %12.8f\n",r, E0[r], F0[r]) + end + end + + # + # define batches (FockConfigs present in resolvant) + jobs = Dict{FockConfig{N},Vector{Tuple}}() + for (fock_ket, configs_ket) in ref_vec.data + for (ftrans, terms) in clustered_ham + fock_x = ftrans + fock_ket + + # + # check to make sure this fock config doesn't have negative or too many electrons in any cluster + all(f[1] >= 0 for f in fock_x) || continue + all(f[2] >= 0 for f in fock_x) || continue + all(f[1] <= length(clusters[fi]) for (fi,f) in enumerate(fock_x)) || continue + all(f[2] <= length(clusters[fi]) for (fi,f) in enumerate(fock_x)) || continue + + job_input = (terms, fock_ket, configs_ket) + if haskey(jobs, fock_x) + push!(jobs[fock_x], job_input) + else + jobs[fock_x] = [job_input] + end + end + end + + + jobs_vec = [] + for (fock_x, job) in jobs + push!(jobs_vec, (fock_x, job)) + end + + println(" Number of jobs: ", length(jobs_vec)) + println(" Number of threads: ", Threads.nthreads()) + BLAS.set_num_threads(1) + flush(stdout) + + #ham_0s = Vector{ClusteredOperator}() + #for t in Threads.nthreads() + # push!(ham_0s, extract_1body_operator(clustered_ham, op_string = H0) ) + #end + + + e2_thread = Vector{Vector{Float64}}() + for tid in 1:Threads.nthreads() + push!(e2_thread, zeros(T,R)) + end + + tmp = ceil(length(jobs_vec)/100) + verbose < 2 || println(" |----------------------------------------------------------------------------------------------------|") + verbose < 2 || println(" |0% 100%|") + verbose < 2 || print(" |") + #@profilehtml @Threads.threads for job in jobs_vec + nprinted = 0 + alloc = @allocated t = @elapsed begin + + @Threads.threads for (jobi,job) in collect(enumerate(jobs_vec)) + #for (jobi,job) in collect(enumerate(jobs_vec)) + fock_sig = job[1] + tid = Threads.threadid() + e2_thread[tid] .+= _pt2_job2(fock_sig, job[2], ref_vec, cluster_ops, clustered_ham, clustered_ham_0, + nbody, verbose, thresh_foi, max_number, E0, F0, prescreen, compress_twice) + if verbose > 1 + if jobi%tmp == 0 + begin + lock(lk) + try + print("-") + nprinted += 1 + flush(stdout) + finally + unlock(lk) + end + end + end + end + end + end + flush(stdout) + verbose < 2 || for i in nprinted+1:100 + print("-") + end + verbose < 2 || println("|") + flush(stdout) + + @printf(" %-48s%10.1f s Allocated: %10.1e GB\n", "Time spent computing E2: ",t,alloc*1e-9) + ecorr = sum(e2_thread) + + E2 = zeros(R) + for r in 1:R + E2[r] = E0[r] + ecorr[r] + @printf(" State %3i: %-35s%14.8f\n", r, "E(PT2) corr: ", ecorr[r]) + end + + @printf(" %5s %12s %12s\n", "Root", "E(0)", "E(2)") + for r in 1:R + @printf(" %5s %12.8f %12.8f\n", r, E0[r], E2[r]) + end + println(" ......................................................................................|") + + return E2 +end function _pt2_job2(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_ham, clustered_ham_0, nbody, verbose, thresh, max_number, E0, F0, prescreen, compress_twice) where {T,N,R} - ecorr = [] + tconfigs_to_process = Dict{TuckerConfig{N}, Vector{Vector{Any}}}() + + ecorr = Vector{T}([0.0 for i in 1:R]) + for jobi in job terms, ket_fock, ket_tconfigs = jobi - for (ket_tconfig, ket_tuck) in ket_tconfigs - sig = BSTstate(ket.clusters, ket.p_spaces, ket.q_spaces, T=T, R=R) - add_fockconfig!(sig, sig_fock) - data = OrderedDict{TuckerConfig{N},Vector{Tucker{T,N,R}}}() - - for term in terms - length(term.clusters) <= nbody || continue - + for term in terms - - - + length(term.clusters) <= nbody || continue + + for (ket_tconfig, ket_tuck) in ket_tconfigs # # find the sig TuckerConfigs reached by applying current Hamiltonian term to ket_tconfig. # @@ -956,7 +1118,7 @@ function _pt2_job2(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_h # # [(p'q), I, I, (r's), I ] * |P,Q,P,Q,P> --> |X, Q, P, X, P> where X = {P,Q} # - # This term will couple to 4 distinct tucker blocks (assuming each of the active clusters + # This this term, will couple to 4 distinct tucker blocks (assuming each of the active clusters # have both non-zero P and Q spaces within the current fock sector, "sig_fock". # # We will loop over all these destination TuckerConfig's by creating the cartesian product of their @@ -964,6 +1126,7 @@ function _pt2_job2(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_h # available = [] # list of lists of index ranges, the cartesian product is the set needed + # # for current term, expand index ranges for active clusters for ci in term.clusters tmp = [] @@ -987,73 +1150,84 @@ function _pt2_job2(sig_fock, job, ket::BSTstate{T,N,R}, cluster_ops, clustered_h sig_tconfig[ci.idx] = prod[cidx] end sig_tconfig = TuckerConfig(sig_tconfig) - - # - # the `term` has now coupled our ket TuckerConfig, to a sig TuckerConfig - # let's compute the matrix element block, then compress, then add it to any existing compressed - # coefficient tensor for that sig TuckerConfig. - # - # Both the Compression and addition takes a fair amount of work. - - - check_term(term, sig_fock, sig_tconfig, ket_fock, ket_tconfig) || continue - - - if prescreen - bound = calc_bound(term, cluster_ops, - sig_fock, sig_tconfig, - ket_fock, ket_tconfig, ket_tuck, - prescreen=thresh) - bound == true || continue + # println(sig_tconfig) + # println([term, ket_fock, ket_tconfig]) + if haskey(tconfigs_to_process, sig_tconfig) + push!(tconfigs_to_process[sig_tconfig], [term, ket_fock, ket_tconfig]) + else + tconfigs_to_process[sig_tconfig] = [[term, ket_fock, ket_tconfig]] end + end + end + end + end - sig_tuck = form_sigma_block_expand(term, cluster_ops, - sig_fock, sig_tconfig, - ket_fock, ket_tconfig, ket_tuck, - max_number=max_number, - prescreen=thresh) + for (sig_tconfig, terms_to_process) in tconfigs_to_process + + i=0 + for (term, ket_fock, ket_tconfig) in terms_to_process + ket_tuck = ket[ket_fock][ket_tconfig] + # + # the `term` has now coupled our ket TuckerConfig, to a sig TuckerConfig + # let's compute the matrix element block, then compress, then add it to any existing compressed + # coefficient tensor for that sig TuckerConfig. + # + # Both the Compression and addition takes a fair amount of work. - if length(sig_tuck) == 0 - continue - end - if norm(sig_tuck) < thresh - continue - end + check_term(term, sig_fock, sig_tconfig, ket_fock, ket_tconfig) || continue - #compress new addition - sig_tuck = compress(sig_tuck, thresh=thresh) - length(sig_tuck) > 0 || continue + if prescreen + bound = calc_bound(term, cluster_ops, + sig_fock, sig_tconfig, + ket_fock, ket_tconfig, ket_tuck, + prescreen=thresh) + bound == true || continue + end - #add to current sigma vector - if haskey(sig[sig_fock], sig_tconfig) + sig_tuck = form_sigma_block_expand(term, cluster_ops, + sig_fock, sig_tconfig, + ket_fock, ket_tconfig, ket_tuck, + max_number=max_number, + prescreen=thresh) + #compress new addition + sig_tuck = compress(sig_tuck, thresh=thresh) + # println(sig_tuck) + if i==0 + curr_tuck = sig_tuck + else + curr_tuck=nonorth_add([curr_tuck, sig_tuck]) + end + curr_tuck=nonorth_add([curr_tuck, sig_tuck]) + i+=1 + end - if haskey(data, sig_tconfig) - push!(data[sig_tconfig], sig_tuck) - else - data[sig_tconfig] = [sig[sig_fock][sig_tconfig], sig_tuck] - end - else - sig[sig_fock][sig_tconfig] = sig_tuck - end + if length(curr_tuck) == 0 + continue + end + if norm(curr_tuck) < thresh + continue + end - end - - end - for (tconfig, tucks) in data - if compress_twice - sig[sig_fock][tconfig] = compress(nonorth_add(tucks), thresh=thresh) - else - sig[sig_fock][tconfig] = nonorth_add(tucks) - end - end - # Compute PT2 energy for this job - v_pt, e_pt, ecor = compute_pt1_wavefunction(sig, ket, cluster_ops, clustered_ham, clustered_ham_0, E0, F0, verbose=0) - push!(ecorr, ecor) + + #compress new addition + curr_tuck = compress(curr_tuck, thresh=thresh) + if compress_twice + curr_tuck = compress(curr_tuck, thresh=thresh) end + + sig = BSTstate(ket.clusters, ket.p_spaces, ket.q_spaces, T=T, R=R) + add_fockconfig!(sig, sig_fock) + sig[sig_fock][sig_tconfig] = curr_tuck + + # Compute PT2 energy for this job + _, _, ecorr_i = compute_pt1_wavefunction(sig, ket, cluster_ops, clustered_ham, clustered_ham_0, E0, F0, verbose=0) + # println(ecorr_i) + ecorr += ecorr_i end - return sum(ecorr) - # return ecorr + + + return ecorr end \ No newline at end of file