From b23ab06c34d8deb13dc9426c12739760a4bb4d22 Mon Sep 17 00:00:00 2001 From: Benedikt Kloss Date: Mon, 15 Jan 2024 20:15:30 -0500 Subject: [PATCH] Move solver_funcs to separate directory. Adapt dmrg and dmrg-x to new solver interface. --- src/ITensorNetworks.jl | 6 +- src/solvers/applyexp.jl | 27 +++++++ src/solvers/dmrg_x_solver.jl | 19 +++++ src/solvers/eigsolve.jl | 40 ++++++++++ src/solvers/exponentiate.jl | 42 ++++++++++ src/treetensornetworks/solvers/dmrg.jl | 29 +------ src/treetensornetworks/solvers/dmrg_x.jl | 14 +--- src/treetensornetworks/solvers/tdvp.jl | 80 +------------------ src/treetensornetworks/solvers/update_step.jl | 3 +- .../test_solvers/test_tdvp.jl | 13 +-- 10 files changed, 145 insertions(+), 128 deletions(-) create mode 100644 src/solvers/applyexp.jl create mode 100644 src/solvers/dmrg_x_solver.jl create mode 100644 src/solvers/eigsolve.jl create mode 100644 src/solvers/exponentiate.jl diff --git a/src/ITensorNetworks.jl b/src/ITensorNetworks.jl index 5b660bc9..81a2a3d7 100644 --- a/src/ITensorNetworks.jl +++ b/src/ITensorNetworks.jl @@ -106,6 +106,11 @@ include("tensornetworkoperators.jl") include(joinpath("ITensorsExt", "itensorutils.jl")) include(joinpath("Graphs", "abstractgraph.jl")) include(joinpath("Graphs", "abstractdatagraph.jl")) +include(joinpath("solvers","eigsolve.jl")) +include(joinpath("solvers","exponentiate.jl")) +include(joinpath("treetensornetworks", "solvers", "applyexp.jl")) #this defines the primitive before the solver function +include(joinpath("solvers","applyexp.jl")) +include(joinpath("solvers","dmrg_x_solver.jl")) include(joinpath("treetensornetworks", "abstracttreetensornetwork.jl")) include(joinpath("treetensornetworks", "ttn.jl")) include(joinpath("treetensornetworks", "opsum_to_ttn.jl")) @@ -114,7 +119,6 @@ include(joinpath("treetensornetworks", "projttns", "projttn.jl")) include(joinpath("treetensornetworks", "projttns", "projttnsum.jl")) include(joinpath("treetensornetworks", "projttns", "projttn_apply.jl")) include(joinpath("treetensornetworks", "solvers", "solver_utils.jl")) -include(joinpath("treetensornetworks", "solvers", "applyexp.jl")) include(joinpath("treetensornetworks", "solvers", "update_step.jl")) include(joinpath("treetensornetworks", "solvers", "alternating_update.jl")) include(joinpath("treetensornetworks", "solvers", "tdvp.jl")) diff --git a/src/solvers/applyexp.jl b/src/solvers/applyexp.jl new file mode 100644 index 00000000..55a7052d --- /dev/null +++ b/src/solvers/applyexp.jl @@ -0,0 +1,27 @@ +function applyexp_solver() + function solver( + init; + psi_ref!, + PH_ref!, + region, + sweep_regions, + sweep_step, + solver_krylovdim=30, + solver_outputlevel=0, + solver_tol=1E-8, + substep, + time_step, + normalize, + ) + H=PH_ref![] + #applyexp tol is absolute, compute from tol_per_unit_time: + tol = abs(time_step) * tol_per_unit_time + psi, exp_info = applyexp( + H, time_step, init; tol, maxiter=solver_krylovdim, outputlevel=solver_outputlevel + ) + return psi, (; info=exp_info) + end + return solver +end + + diff --git a/src/solvers/dmrg_x_solver.jl b/src/solvers/dmrg_x_solver.jl new file mode 100644 index 00000000..33b1ad2a --- /dev/null +++ b/src/solvers/dmrg_x_solver.jl @@ -0,0 +1,19 @@ +function dmrg_x_solver( + init; + psi_ref!, + PH_ref!, + normalize=nothing, + region, + sweep_regions, + sweep_step, + half_sweep, + step_kwargs... + ) + H = contract(PH_ref![], ITensor(1.0)) + D, U = eigen(H; ishermitian=true) + u = uniqueind(U, H) + max_overlap, max_ind = findmax(abs, array(dag(init) * U)) + U_max = U * dag(onehot(u => max_ind)) + # TODO: improve this to return the energy estimate too + return U_max, NamedTuple() +end \ No newline at end of file diff --git a/src/solvers/eigsolve.jl b/src/solvers/eigsolve.jl new file mode 100644 index 00000000..da3ba5c1 --- /dev/null +++ b/src/solvers/eigsolve.jl @@ -0,0 +1,40 @@ + +function eigsolve_solver(; + solver_which_eigenvalue=:SR, #TODO: settle on pattern to pass solver kwargs + ishermitian=true, + solver_tol=1e-14, + solver_krylovdim=3, + solver_maxiter=1, + solver_verbosity=0, + ) + + function solver( + init; + psi_ref!, + PH_ref!, + normalize, + region, + sweep_regions, + sweep_step, + sweep_kwargs... + # slurp solver_kwargs? #TODO: homogenize how the solver kwargs are passed + ) + howmany = 1 + which = solver_which_eigenvalue + vals, vecs, info = eigsolve( + PH_ref![], + init, + howmany, + which; + ishermitian, + tol=solver_tol, + krylovdim=solver_krylovdim, + maxiter=solver_maxiter, + verbosity=solver_verbosity, + ) + phi = vecs[1] + return phi, (; solver_info=info, energies=vals) + end + return solver + end + diff --git a/src/solvers/exponentiate.jl b/src/solvers/exponentiate.jl new file mode 100644 index 00000000..bb2e98e2 --- /dev/null +++ b/src/solvers/exponentiate.jl @@ -0,0 +1,42 @@ +function exponentiate_solver() + function solver( + init; + psi_ref!, + PH_ref!, + ishermitian=true, + issymmetric=true, + region, + sweep_regions, + sweep_step, + solver_krylovdim=30, + solver_maxiter=100, + solver_outputlevel=0, + solver_tol=1E-12, + substep, + normalize, + time_step, + ) + #H=copy(PH_ref![]) + H=PH_ref![] ###since we are not changing H we don't need the copy + # let's test whether given region and sweep regions we can find out what the previous and next region were + # this will be needed in subspace expansion + region_ind=sweep_step + next_region=region_ind==length(sweep_regions) ? nothing : first(sweep_regions[region_ind+1]) + previous_region=region_ind==1 ? nothing : first(sweep_regions[region_ind-1]) + + phi, exp_info = KrylovKit.exponentiate( + H, + time_step, + init; + ishermitian, + issymmetric, + tol=solver_tol, + krylovdim=solver_krylovdim, + maxiter=solver_maxiter, + verbosity=solver_outputlevel, + eager=true, + ) + return phi, (; info=exp_info) + end + return solver +end diff --git a/src/treetensornetworks/solvers/dmrg.jl b/src/treetensornetworks/solvers/dmrg.jl index c67b3b86..8f24fd63 100644 --- a/src/treetensornetworks/solvers/dmrg.jl +++ b/src/treetensornetworks/solvers/dmrg.jl @@ -1,31 +1,3 @@ -function eigsolve_solver(; - solver_which_eigenvalue=:SR, - ishermitian=true, - solver_tol=1e-14, - solver_krylovdim=3, - solver_maxiter=1, - solver_verbosity=0, -) - function solver(H, init; normalize=nothing, region=nothing, half_sweep=nothing) - howmany = 1 - which = solver_which_eigenvalue - vals, vecs, info = eigsolve( - H, - init, - howmany, - which; - ishermitian, - tol=solver_tol, - krylovdim=solver_krylovdim, - maxiter=solver_maxiter, - verbosity=solver_verbosity, - ) - psi = vecs[1] - return psi, (; solver_info=info, energies=vals) - end - return solver -end - """ Overload of `ITensors.dmrg`. """ @@ -51,6 +23,7 @@ function dmrg( ), H, init; + reverse_step=false, kwargs..., ) end diff --git a/src/treetensornetworks/solvers/dmrg_x.jl b/src/treetensornetworks/solvers/dmrg_x.jl index 30a97f3a..67ab42c9 100644 --- a/src/treetensornetworks/solvers/dmrg_x.jl +++ b/src/treetensornetworks/solvers/dmrg_x.jl @@ -1,16 +1,4 @@ -function dmrg_x_solver( - PH, init; normalize=nothing, region=nothing, half_sweep=nothing, reverse_step=nothing -) - H = contract(PH, ITensor(1.0)) - D, U = eigen(H; ishermitian=true) - u = uniqueind(U, H) - max_overlap, max_ind = findmax(abs, array(dag(init) * U)) - U_max = U * dag(onehot(u => max_ind)) - # TODO: improve this to return the energy estimate too - return U_max, NamedTuple() -end - function dmrg_x(PH, init::AbstractTTN; kwargs...) - psi = alternating_update(dmrg_x_solver, PH, init; kwargs...) + psi = alternating_update(ITensorNetworks.dmrg_x_solver, PH, init; kwargs...) return psi end diff --git a/src/treetensornetworks/solvers/tdvp.jl b/src/treetensornetworks/solvers/tdvp.jl index dfc553c6..863a36df 100644 --- a/src/treetensornetworks/solvers/tdvp.jl +++ b/src/treetensornetworks/solvers/tdvp.jl @@ -1,72 +1,3 @@ -function exponentiate_solver() - function solver( - init; - psi_ref!, - PH_ref!, - ishermitian=true, - issymmetric=true, - region, - sweep_regions, - sweep_step, - solver_krylovdim=30, - solver_maxiter=100, - solver_outputlevel=0, - solver_tol=1E-12, - substep, - normalize, - time_step, - ) - #H=copy(PH_ref![]) - H=PH_ref![] ###since we are not changing H we don't need the copy - # let's test whether given region and sweep regions we can find out what the previous and next region were - # this will be needed in subspace expansion - region_ind=sweep_step - next_region=region_ind==length(sweep_regions) ? nothing : first(sweep_regions[region_ind+1]) - previous_region=region_ind==1 ? nothing : first(sweep_regions[region_ind-1]) - - phi, exp_info = KrylovKit.exponentiate( - H, - time_step, - init; - ishermitian, - issymmetric, - tol=solver_tol, - krylovdim=solver_krylovdim, - maxiter=solver_maxiter, - verbosity=solver_outputlevel, - eager=true, - ) - return phi, (; info=exp_info) - end - return solver -end - -function applyexp_solver() - function solver( - init; - psi_ref!, - PH_ref!, - region, - sweep_regions, - sweep_step, - solver_krylovdim=30, - solver_outputlevel=0, - solver_tol=1E-8, - substep, - time_step, - normalize, - ) - H=PH_ref![] - #applyexp tol is absolute, compute from tol_per_unit_time: - tol = abs(time_step) * tol_per_unit_time - psi, exp_info = applyexp( - H, time_step, init; tol, maxiter=solver_krylovdim, outputlevel=solver_outputlevel - ) - return psi, (; info=exp_info) - end - return solver -end - function _compute_nsweeps(nsteps, t, time_step, order) nsweeps_per_step = order / 2 nsweeps = 1 @@ -183,15 +114,6 @@ Optional keyword arguments: * `observer` - object implementing the Observer interface which can perform measurements and stop early * `write_when_maxdim_exceeds::Int` - when the allowed maxdim exceeds this value, begin saving tensors to disk to free memory in large calculations """ -function tdvp(H, t::Number, init::AbstractTTN; solver_backend="exponentiate", kwargs...) - if solver_backend == "exponentiate" - solver = exponentiate_solver - elseif solver_backend == "applyexp" - solver = applyexp_solver - else - error( - "solver_backend=$solver_backend not recognized (options are \"applyexp\" or \"exponentiate\")", - ) - end +function tdvp(H, t::Number, init::AbstractTTN; solver=exponentiate_solver, kwargs...) return tdvp(solver(), H, t, init; kwargs...) end diff --git a/src/treetensornetworks/solvers/update_step.jl b/src/treetensornetworks/solvers/update_step.jl index 30e3f29e..a62f7b70 100644 --- a/src/treetensornetworks/solvers/update_step.jl +++ b/src/treetensornetworks/solvers/update_step.jl @@ -44,7 +44,8 @@ function update_step( step_printer=step_printer, (step_observer!)=observer(), sweep::Int=1, - sweep_regions=default_sweep_regions(nsite, psi), + reverse_step::Bool=false, + sweep_regions=default_sweep_regions(nsite, psi;reverse_step), kwargs..., ) PH = copy(PH) diff --git a/test/test_treetensornetworks/test_solvers/test_tdvp.jl b/test/test_treetensornetworks/test_solvers/test_tdvp.jl index efa76f02..1d50fba1 100644 --- a/test/test_treetensornetworks/test_solvers/test_tdvp.jl +++ b/test/test_treetensornetworks/test_solvers/test_tdvp.jl @@ -4,6 +4,7 @@ using KrylovKit: exponentiate using Observers using Random using Test +#exponentiate_solver=ITensorNetworks.exponentiate_solver #ToDo: how to best handle importing etc. @testset "MPS TDVP" begin @testset "Basic TDVP" begin @@ -31,7 +32,7 @@ using Test # #Different backend solvers, default solver_backend = "applyexp" ψ1_exponentiate_backend = tdvp( - H, -0.1im, ψ0; nsteps=1, cutoff, nsite=1, solver_backend="exponentiate" + H, -0.1im, ψ0; nsteps=1, cutoff, nsite=1, solver=exponentiate_solver ) @test ψ1 ≈ ψ1_exponentiate_backend rtol = 1e-7 @@ -48,7 +49,7 @@ using Test #Different backend solvers, default solver_backend = "applyexp" ψ2_exponentiate_backend = tdvp( - H, +0.1im, ψ1; nsteps=1, cutoff, solver_backend="exponentiate" + H, +0.1im, ψ1; nsteps=1, cutoff, solver=exponentiate_solver ) @test ψ2 ≈ ψ2_exponentiate_backend rtol = 1e-7 @@ -84,7 +85,7 @@ using Test #Different backend solvers, default solver_backend = "applyexp" ψ1_exponentiate_backend = tdvp( - Hs, -0.1im, ψ0; nsteps=1, cutoff, nsite=1, solver_backend="exponentiate" + Hs, -0.1im, ψ0; nsteps=1, cutoff, nsite=1, solver=exponentiate_solver ) @test ψ1 ≈ ψ1_exponentiate_backend rtol = 1e-7 @@ -101,7 +102,7 @@ using Test #Different backend solvers, default solver_backend = "applyexp" ψ2_exponentiate_backend = tdvp( - Hs, +0.1im, ψ1; nsteps=1, cutoff, solver_backend="exponentiate" + Hs, +0.1im, ψ1; nsteps=1, cutoff, solver=exponentiate_solver ) @test ψ2 ≈ ψ2_exponentiate_backend rtol = 1e-7 @@ -249,7 +250,7 @@ using Test solver_tol=1e-12, solver_maxiter=500, solver_krylovdim=25, - solver_backend="exponentiate", + solver=exponentiate_solver, ) # TODO: What should `expect` output? Right now # it outputs a dictionary. @@ -385,7 +386,7 @@ using Test reverse_step, normalize=true, solver_krylovdim=15, - solver_backend="exponentiate", + solver=ITensorNetworks.exponentiate_solver, ) end