From 200d2c5f726c030b0a69d6507d5032317445b903 Mon Sep 17 00:00:00 2001 From: vfisikop Date: Thu, 29 Feb 2024 16:27:36 +0200 Subject: [PATCH] Remove submodules and add volesti and externals directly in the repo. commit hash from volesti 5bf9188fb11fddab17de98fabc505e993216ca5f --- .gitmodules | 3 - inst/AUTHORS | 9 +- inst/COPYRIGHTS | 7 +- src/Makevars | 10 +- src/Makevars.win | 10 +- src/external/PackedCSparse/FloatArray.h | 307 ++ src/external/PackedCSparse/FloatArrayAVX2.h | 222 ++ src/external/PackedCSparse/PackedChol.h | 308 ++ src/external/PackedCSparse/SparseMatrix.h | 230 ++ src/external/PackedCSparse/add.h | 102 + src/external/PackedCSparse/chol.h | 247 ++ src/external/PackedCSparse/leverage.h | 65 + src/external/PackedCSparse/leverageJL.h | 148 + src/external/PackedCSparse/multiply.h | 142 + src/external/PackedCSparse/outerprod.h | 86 + src/external/PackedCSparse/projinv.h | 90 + src/external/PackedCSparse/qd/COPYING | 16 + src/external/PackedCSparse/qd/Makefile | 15 + src/external/PackedCSparse/qd/NEWS | 181 ++ src/external/PackedCSparse/qd/README | 437 +++ src/external/PackedCSparse/qd/bits.cc | 85 + src/external/PackedCSparse/qd/bits.h | 32 + src/external/PackedCSparse/qd/c_dd.cc | 314 ++ src/external/PackedCSparse/qd/c_dd.h | 98 + src/external/PackedCSparse/qd/c_qd.cc | 450 +++ src/external/PackedCSparse/qd/c_qd.h | 119 + src/external/PackedCSparse/qd/dd_const.cc | 40 + src/external/PackedCSparse/qd/dd_inline.h | 621 ++++ src/external/PackedCSparse/qd/dd_real.cc | 1303 ++++++++ src/external/PackedCSparse/qd/dd_real.h | 293 ++ src/external/PackedCSparse/qd/fpu.cc | 124 + src/external/PackedCSparse/qd/fpu.h | 39 + src/external/PackedCSparse/qd/inline.h | 143 + src/external/PackedCSparse/qd/qd.pdf | Bin 0 -> 194502 bytes src/external/PackedCSparse/qd/qd_config.h | 92 + src/external/PackedCSparse/qd/qd_const.cc | 62 + src/external/PackedCSparse/qd/qd_inline.h | 1047 +++++++ src/external/PackedCSparse/qd/qd_real.cc | 2624 +++++++++++++++++ src/external/PackedCSparse/qd/qd_real.h | 296 ++ src/external/PackedCSparse/qd/util.cc | 22 + src/external/PackedCSparse/qd/util.h | 4 + src/external/PackedCSparse/transpose.h | 90 + .../Spectra/include/Spectra/GenEigsBase.h | 479 +++ .../Spectra/GenEigsComplexShiftSolver.h | 156 + .../include/Spectra/GenEigsRealShiftSolver.h | 90 + .../Spectra/include/Spectra/GenEigsSolver.h | 160 + .../Spectra/include/Spectra/LinAlg/Arnoldi.h | 283 ++ .../Spectra/include/Spectra/LinAlg/BKLDLT.h | 522 ++++ .../include/Spectra/LinAlg/DoubleShiftQR.h | 378 +++ .../Spectra/include/Spectra/LinAlg/Lanczos.h | 170 ++ .../include/Spectra/LinAlg/TridiagEigen.h | 219 ++ .../Spectra/LinAlg/UpperHessenbergEigen.h | 317 ++ .../Spectra/LinAlg/UpperHessenbergQR.h | 670 +++++ .../include/Spectra/MatOp/DenseCholesky.h | 110 + .../Spectra/MatOp/DenseGenComplexShiftSolve.h | 104 + .../include/Spectra/MatOp/DenseGenMatProd.h | 82 + .../Spectra/MatOp/DenseGenRealShiftSolve.h | 90 + .../include/Spectra/MatOp/DenseSymMatProd.h | 75 + .../Spectra/MatOp/DenseSymShiftSolve.h | 94 + .../include/Spectra/MatOp/SparseCholesky.h | 111 + .../include/Spectra/MatOp/SparseGenMatProd.h | 76 + .../Spectra/MatOp/SparseGenRealShiftSolve.h | 95 + .../Spectra/MatOp/SparseRegularInverse.h | 102 + .../include/Spectra/MatOp/SparseSymMatProd.h | 75 + .../Spectra/MatOp/SparseSymShiftSolve.h | 97 + .../Spectra/MatOp/internal/ArnoldiOp.h | 155 + .../MatOp/internal/SymGEigsCholeskyOp.h | 77 + .../Spectra/MatOp/internal/SymGEigsRegInvOp.h | 74 + .../Spectra/include/Spectra/SymEigsBase.h | 447 +++ .../include/Spectra/SymEigsShiftSolver.h | 205 ++ .../Spectra/include/Spectra/SymEigsSolver.h | 174 ++ .../Spectra/include/Spectra/SymGEigsSolver.h | 334 +++ .../Spectra/include/Spectra/Util/CompInfo.h | 37 + .../Spectra/include/Spectra/Util/GEigsMode.h | 34 + .../include/Spectra/Util/SelectionRule.h | 277 ++ .../include/Spectra/Util/SimpleRandom.h | 93 + .../Spectra/include/Spectra/Util/TypeTraits.h | 73 + .../include/Spectra/contrib/LOBPCGSolver.h | 501 ++++ .../Spectra/contrib/PartialSVDSolver.h | 203 ++ src/external/minimum_ellipsoid/bnmin_main.h | 87 + src/external/minimum_ellipsoid/khach.h | 220 ++ src/external/minimum_ellipsoid/mcpoint.h | 477 +++ src/volesti | 1 - src/volesti/include/SDPAFormatManager.h | 271 ++ .../include/cartesian_geom/cartesian_kernel.h | 39 + src/volesti/include/cartesian_geom/point.h | 238 ++ src/volesti/include/convex_bodies/ball.h | 176 ++ .../convex_bodies/ballintersectconvex.h | 401 +++ src/volesti/include/convex_bodies/barriers.h | 52 + .../include/convex_bodies/convex_body.h | 109 + .../correlation_matrices/corre_matrix.hpp | 161 + .../correlation_spectrahedron.hpp | 266 ++ .../correlation_spectrahedron_MT.hpp | 180 ++ src/volesti/include/convex_bodies/ellipsoid.h | 281 ++ src/volesti/include/convex_bodies/hpolytope.h | 927 ++++++ .../include/convex_bodies/orderpolytope.h | 755 +++++ .../include/convex_bodies/spectrahedra/LMI.h | 251 ++ .../spectrahedra/spectrahedron.h | 492 ++++ .../convex_bodies/vpolyintersectvpoly.h | 429 +++ src/volesti/include/convex_bodies/vpolytope.h | 648 ++++ .../convex_bodies/zonoIntersecthpoly.h | 272 ++ src/volesti/include/convex_bodies/zpolytope.h | 619 ++++ .../include/diagnostics/diagnostics.hpp | 24 + .../diagnostics/effective_sample_size.hpp | 123 + .../ess_updater_autocovariance.hpp | 69 + .../diagnostics/ess_window_updater.hpp | 154 + src/volesti/include/diagnostics/geweke.hpp | 79 + .../include/diagnostics/interval_psrf.hpp | 79 + .../include/diagnostics/multivariate_psrf.hpp | 55 + .../include/diagnostics/print_diagnostics.hpp | 56 + src/volesti/include/diagnostics/raftery.hpp | 142 + .../raftery_subroutines/empquant.hpp | 32 + .../raftery_subroutines/indtest.hpp | 43 + .../diagnostics/raftery_subroutines/mcest.hpp | 28 + .../raftery_subroutines/mctest.hpp | 63 + .../diagnostics/raftery_subroutines/ppnd.hpp | 56 + .../diagnostics/raftery_subroutines/thin.hpp | 32 + .../include/diagnostics/thin_samples.hpp | 37 + .../include/diagnostics/univariate_psrf.hpp | 67 + .../boost_random_number_generator.hpp | 95 + .../generators/convex_bodies_generator.h | 124 + .../generators/h_polytopes_generator.h | 61 + .../generators/known_polytope_generators.h | 361 +++ .../generators/order_polytope_generator.h | 63 + .../include/generators/sdp_generator.h | 151 + .../generators/v_polytopes_generators.h | 175 ++ .../generators/z_polytopes_generators.h | 145 + .../integration/simple_MC_integration.hpp | 276 ++ src/volesti/include/lp_oracles/misc_lp.h | 145 + src/volesti/include/lp_oracles/solve_lp.h | 296 ++ src/volesti/include/lp_oracles/vpolyoracles.h | 427 +++ src/volesti/include/lp_oracles/zpolyoracles.h | 221 ++ .../matrix_operations/DenseProductMatrix.h | 105 + .../matrix_operations/EigenDenseMatrix.h | 76 + .../matrix_operations/EigenvaluesProblems.h | 450 +++ src/volesti/include/misc/linear_extensions.h | 92 + src/volesti/include/misc/misc.h | 268 ++ src/volesti/include/misc/poset.h | 131 + src/volesti/include/misc/print_table.hpp | 384 +++ .../include/nlp_oracles/nlp_hpolyoracles.hpp | 572 ++++ .../include/nlp_oracles/nlp_oracles.hpp | 22 + .../include/nlp_oracles/nlp_vpolyoracles.hpp | 314 ++ src/volesti/include/ode_solvers/basis.hpp | 186 ++ .../include/ode_solvers/collocation.hpp | 298 ++ src/volesti/include/ode_solvers/euler.hpp | 130 + .../ode_solvers/generalized_leapfrog.hpp | 172 ++ .../include/ode_solvers/implicit_midpoint.hpp | 177 ++ .../ode_solvers/integral_collocation.hpp | 289 ++ src/volesti/include/ode_solvers/leapfrog.hpp | 221 ++ .../include/ode_solvers/ode_solvers.hpp | 64 + .../include/ode_solvers/oracle_functors.hpp | 397 +++ .../ode_solvers/oracle_functors_rcpp.hpp | 159 + .../ode_solvers/randomized_midpoint.hpp | 226 ++ .../ode_solvers/richardson_extrapolation.hpp | 188 ++ .../include/ode_solvers/runge_kutta.hpp | 172 ++ .../optimization/simulated_annealing.hpp | 152 + .../include/optimization/sliding_window.hpp | 66 + .../preprocess/crhmc/analytic_center.h | 173 ++ .../preprocess/crhmc/constraint_problem.h | 81 + .../include/preprocess/crhmc/crhmc_input.h | 172 ++ .../include/preprocess/crhmc/crhmc_problem.h | 703 +++++ .../include/preprocess/crhmc/crhmc_utils.h | 406 +++ .../include/preprocess/crhmc/lewis_center.h | 188 ++ src/volesti/include/preprocess/crhmc/opts.h | 62 + .../preprocess/crhmc/two_sided_barrier.h | 173 ++ .../crhmc/weighted_two_sided_barrier.h | 166 ++ .../estimate_L_smooth_parameter.hpp | 73 + .../include/preprocess/max_inscribed_ball.hpp | 216 ++ .../preprocess/max_inscribed_ellipsoid.hpp | 236 ++ .../max_inscribed_ellipsoid_rounding.hpp | 77 + ...n_sampling_covering_ellipsoid_rounding.hpp | 123 + .../include/preprocess/svd_rounding.hpp | 176 ++ .../random_walks/boltzmann_hmc_walk.hpp | 216 ++ .../random_walks/boundary_cdhr_walk.hpp | 91 + .../random_walks/boundary_rdhr_walk.hpp | 85 + .../include/random_walks/compute_diameter.hpp | 225 ++ .../crhmc/additional_units/auto_tuner.hpp | 64 + .../additional_units/dynamic_regularizer.hpp | 59 + .../additional_units/dynamic_step_size.hpp | 118 + .../crhmc/additional_units/dynamic_weight.hpp | 76 + .../include/random_walks/crhmc/crhmc_walk.hpp | 245 ++ .../random_walks/crhmc/hamiltonian.hpp | 249 ++ .../random_walks/ellipsoid_walks/README.md | 7 + .../ellipsoid_walks/dikin_walker.h | 157 + .../ellipsoid_walks/john_walker.h | 204 ++ .../ellipsoid_walks/math_functions.h | 54 + .../ellipsoid_walks/vaidya_walker.h | 171 ++ ...ial_hamiltonian_monte_carlo_exact_walk.hpp | 300 ++ .../gaussian_accelerated_billiard_walk.hpp | 248 ++ .../random_walks/gaussian_ball_walk.hpp | 107 + .../random_walks/gaussian_cdhr_walk.hpp | 151 + ...ian_hamiltonian_monte_carlo_exact_walk.hpp | 275 ++ .../include/random_walks/gaussian_helpers.hpp | 48 + .../random_walks/gaussian_rdhr_walk.hpp | 118 + .../hamiltonian_monte_carlo_walk.hpp | 179 ++ .../include/random_walks/langevin_walk.hpp | 153 + .../random_walks/multithread_walks.hpp | 739 +++++ .../include/random_walks/nuts_hmc_walk.hpp | 380 +++ .../include/random_walks/random_walks.hpp | 34 + .../uniform_accelerated_billiard_walk.hpp | 309 ++ ...orm_accelerated_billiard_walk_parallel.hpp | 311 ++ .../random_walks/uniform_ball_walk.hpp | 96 + .../random_walks/uniform_billiard_walk.hpp | 200 ++ .../random_walks/uniform_cdhr_walk.hpp | 97 + .../random_walks/uniform_dikin_walk.hpp | 102 + .../random_walks/uniform_john_walk.hpp | 94 + .../random_walks/uniform_rdhr_walk.hpp | 92 + .../random_walks/uniform_vaidya_walk.hpp | 95 + .../include/root_finders/mp_solve_wrapper.hpp | 75 + .../include/root_finders/newton_raphson.hpp | 49 + .../quadratic_polynomial_solvers.hpp | 42 + .../include/root_finders/root_finders.hpp | 28 + src/volesti/include/sampling/ellipsoid.hpp | 82 + src/volesti/include/sampling/mmcs.hpp | 128 + .../include/sampling/parallel_mmcs.hpp | 203 ++ .../sampling/random_point_generators.hpp | 395 +++ .../random_point_generators_multithread.hpp | 256 ++ .../sampling/sample_correlation_matrices.hpp | 153 + src/volesti/include/sampling/sampling.hpp | 603 ++++ src/volesti/include/sampling/simplex.hpp | 392 +++ src/volesti/include/sampling/sphere.hpp | 132 + src/volesti/include/sos/NonSymmetricIPM.h | 333 +++ src/volesti/include/sos/NonSymmetricIPM.hpp | 781 +++++ src/volesti/include/sos/README.md | 104 + .../sos/barriers/DualSOSConeStandardBarrier.h | 49 + .../barriers/DualSOSConeStandardBarrier.hpp | 81 + .../include/sos/barriers/FullSpaceBarrier.h | 46 + .../include/sos/barriers/FullSpaceBarrier.hpp | 66 + .../sos/barriers/InterpolantDualSOSBarrier.h | 113 + .../barriers/InterpolantDualSOSBarrier.hpp | 493 ++++ src/volesti/include/sos/barriers/LHSCB.h | 89 + src/volesti/include/sos/barriers/LHSCB.hpp | 89 + .../include/sos/barriers/LPStandardBarrier.h | 47 + .../sos/barriers/LPStandardBarrier.hpp | 43 + .../include/sos/barriers/ProductBarrier.h | 113 + .../include/sos/barriers/ProductBarrier.hpp | 173 ++ src/volesti/include/sos/barriers/SumBarrier.h | 59 + .../include/sos/barriers/SumBarrier.hpp | 101 + .../include/sos/barriers/ZeroSpaceBarrier.h | 44 + .../include/sos/barriers/ZeroSpaceBarrier.hpp | 46 + src/volesti/include/sos/config/config.json | 21 + src/volesti/include/sos/gsoc_history.md | 15 + src/volesti/include/sos/include/cxxtimer.hpp | 184 ++ .../include/matplotlib-cpp/matplotlibcpp.h | 2366 +++++++++++++++ src/volesti/include/sos/plot_saved.png | Bin 0 -> 132753 bytes src/volesti/include/sos/utils.cpp | 6 + src/volesti/include/sos/utils.h | 331 +++ src/volesti/include/volume/copulas.h | 244 ++ src/volesti/include/volume/exact_vols.h | 101 + src/volesti/include/volume/math_helpers.hpp | 46 + src/volesti/include/volume/rotating.hpp | 64 + .../include/volume/sampling_policies.hpp | 50 + .../include/volume/volume_cooling_balls.hpp | 853 ++++++ .../volume/volume_cooling_gaussians.hpp | 490 +++ .../include/volume/volume_cooling_hpoly.hpp | 385 +++ .../volume/volume_sequence_of_balls.hpp | 266 ++ 256 files changed, 54098 insertions(+), 20 deletions(-) delete mode 100644 .gitmodules create mode 100644 src/external/PackedCSparse/FloatArray.h create mode 100644 src/external/PackedCSparse/FloatArrayAVX2.h create mode 100644 src/external/PackedCSparse/PackedChol.h create mode 100644 src/external/PackedCSparse/SparseMatrix.h create mode 100644 src/external/PackedCSparse/add.h create mode 100644 src/external/PackedCSparse/chol.h create mode 100644 src/external/PackedCSparse/leverage.h create mode 100644 src/external/PackedCSparse/leverageJL.h create mode 100644 src/external/PackedCSparse/multiply.h create mode 100644 src/external/PackedCSparse/outerprod.h create mode 100644 src/external/PackedCSparse/projinv.h create mode 100644 src/external/PackedCSparse/qd/COPYING create mode 100644 src/external/PackedCSparse/qd/Makefile create mode 100644 src/external/PackedCSparse/qd/NEWS create mode 100644 src/external/PackedCSparse/qd/README create mode 100644 src/external/PackedCSparse/qd/bits.cc create mode 100644 src/external/PackedCSparse/qd/bits.h create mode 100644 src/external/PackedCSparse/qd/c_dd.cc create mode 100644 src/external/PackedCSparse/qd/c_dd.h create mode 100644 src/external/PackedCSparse/qd/c_qd.cc create mode 100644 src/external/PackedCSparse/qd/c_qd.h create mode 100644 src/external/PackedCSparse/qd/dd_const.cc create mode 100644 src/external/PackedCSparse/qd/dd_inline.h create mode 100644 src/external/PackedCSparse/qd/dd_real.cc create mode 100644 src/external/PackedCSparse/qd/dd_real.h create mode 100644 src/external/PackedCSparse/qd/fpu.cc create mode 100644 src/external/PackedCSparse/qd/fpu.h create mode 100644 src/external/PackedCSparse/qd/inline.h create mode 100644 src/external/PackedCSparse/qd/qd.pdf create mode 100644 src/external/PackedCSparse/qd/qd_config.h create mode 100644 src/external/PackedCSparse/qd/qd_const.cc create mode 100644 src/external/PackedCSparse/qd/qd_inline.h create mode 100644 src/external/PackedCSparse/qd/qd_real.cc create mode 100644 src/external/PackedCSparse/qd/qd_real.h create mode 100644 src/external/PackedCSparse/qd/util.cc create mode 100644 src/external/PackedCSparse/qd/util.h create mode 100644 src/external/PackedCSparse/transpose.h create mode 100644 src/external/Spectra/include/Spectra/GenEigsBase.h create mode 100644 src/external/Spectra/include/Spectra/GenEigsComplexShiftSolver.h create mode 100644 src/external/Spectra/include/Spectra/GenEigsRealShiftSolver.h create mode 100644 src/external/Spectra/include/Spectra/GenEigsSolver.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/Arnoldi.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/BKLDLT.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/DoubleShiftQR.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/Lanczos.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/TridiagEigen.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergEigen.h create mode 100644 src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergQR.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseCholesky.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseGenComplexShiftSolve.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseGenMatProd.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseGenRealShiftSolve.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseSymMatProd.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/DenseSymShiftSolve.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseCholesky.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseGenMatProd.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseGenRealShiftSolve.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseRegularInverse.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseSymMatProd.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/SparseSymShiftSolve.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/internal/ArnoldiOp.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsCholeskyOp.h create mode 100644 src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsRegInvOp.h create mode 100644 src/external/Spectra/include/Spectra/SymEigsBase.h create mode 100644 src/external/Spectra/include/Spectra/SymEigsShiftSolver.h create mode 100644 src/external/Spectra/include/Spectra/SymEigsSolver.h create mode 100644 src/external/Spectra/include/Spectra/SymGEigsSolver.h create mode 100644 src/external/Spectra/include/Spectra/Util/CompInfo.h create mode 100644 src/external/Spectra/include/Spectra/Util/GEigsMode.h create mode 100644 src/external/Spectra/include/Spectra/Util/SelectionRule.h create mode 100644 src/external/Spectra/include/Spectra/Util/SimpleRandom.h create mode 100644 src/external/Spectra/include/Spectra/Util/TypeTraits.h create mode 100644 src/external/Spectra/include/Spectra/contrib/LOBPCGSolver.h create mode 100644 src/external/Spectra/include/Spectra/contrib/PartialSVDSolver.h create mode 100644 src/external/minimum_ellipsoid/bnmin_main.h create mode 100644 src/external/minimum_ellipsoid/khach.h create mode 100644 src/external/minimum_ellipsoid/mcpoint.h delete mode 160000 src/volesti create mode 100644 src/volesti/include/SDPAFormatManager.h create mode 100644 src/volesti/include/cartesian_geom/cartesian_kernel.h create mode 100644 src/volesti/include/cartesian_geom/point.h create mode 100644 src/volesti/include/convex_bodies/ball.h create mode 100644 src/volesti/include/convex_bodies/ballintersectconvex.h create mode 100644 src/volesti/include/convex_bodies/barriers.h create mode 100644 src/volesti/include/convex_bodies/convex_body.h create mode 100755 src/volesti/include/convex_bodies/correlation_matrices/corre_matrix.hpp create mode 100755 src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron.hpp create mode 100755 src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron_MT.hpp create mode 100644 src/volesti/include/convex_bodies/ellipsoid.h create mode 100644 src/volesti/include/convex_bodies/hpolytope.h create mode 100644 src/volesti/include/convex_bodies/orderpolytope.h create mode 100644 src/volesti/include/convex_bodies/spectrahedra/LMI.h create mode 100644 src/volesti/include/convex_bodies/spectrahedra/spectrahedron.h create mode 100644 src/volesti/include/convex_bodies/vpolyintersectvpoly.h create mode 100644 src/volesti/include/convex_bodies/vpolytope.h create mode 100644 src/volesti/include/convex_bodies/zonoIntersecthpoly.h create mode 100644 src/volesti/include/convex_bodies/zpolytope.h create mode 100644 src/volesti/include/diagnostics/diagnostics.hpp create mode 100644 src/volesti/include/diagnostics/effective_sample_size.hpp create mode 100644 src/volesti/include/diagnostics/ess_updater_autocovariance.hpp create mode 100644 src/volesti/include/diagnostics/ess_window_updater.hpp create mode 100644 src/volesti/include/diagnostics/geweke.hpp create mode 100644 src/volesti/include/diagnostics/interval_psrf.hpp create mode 100644 src/volesti/include/diagnostics/multivariate_psrf.hpp create mode 100644 src/volesti/include/diagnostics/print_diagnostics.hpp create mode 100644 src/volesti/include/diagnostics/raftery.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/empquant.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/indtest.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/mcest.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/mctest.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/ppnd.hpp create mode 100644 src/volesti/include/diagnostics/raftery_subroutines/thin.hpp create mode 100644 src/volesti/include/diagnostics/thin_samples.hpp create mode 100644 src/volesti/include/diagnostics/univariate_psrf.hpp create mode 100644 src/volesti/include/generators/boost_random_number_generator.hpp create mode 100644 src/volesti/include/generators/convex_bodies_generator.h create mode 100644 src/volesti/include/generators/h_polytopes_generator.h create mode 100644 src/volesti/include/generators/known_polytope_generators.h create mode 100644 src/volesti/include/generators/order_polytope_generator.h create mode 100644 src/volesti/include/generators/sdp_generator.h create mode 100644 src/volesti/include/generators/v_polytopes_generators.h create mode 100644 src/volesti/include/generators/z_polytopes_generators.h create mode 100644 src/volesti/include/integration/simple_MC_integration.hpp create mode 100644 src/volesti/include/lp_oracles/misc_lp.h create mode 100644 src/volesti/include/lp_oracles/solve_lp.h create mode 100644 src/volesti/include/lp_oracles/vpolyoracles.h create mode 100644 src/volesti/include/lp_oracles/zpolyoracles.h create mode 100644 src/volesti/include/matrix_operations/DenseProductMatrix.h create mode 100644 src/volesti/include/matrix_operations/EigenDenseMatrix.h create mode 100644 src/volesti/include/matrix_operations/EigenvaluesProblems.h create mode 100644 src/volesti/include/misc/linear_extensions.h create mode 100644 src/volesti/include/misc/misc.h create mode 100644 src/volesti/include/misc/poset.h create mode 100644 src/volesti/include/misc/print_table.hpp create mode 100644 src/volesti/include/nlp_oracles/nlp_hpolyoracles.hpp create mode 100644 src/volesti/include/nlp_oracles/nlp_oracles.hpp create mode 100644 src/volesti/include/nlp_oracles/nlp_vpolyoracles.hpp create mode 100644 src/volesti/include/ode_solvers/basis.hpp create mode 100644 src/volesti/include/ode_solvers/collocation.hpp create mode 100644 src/volesti/include/ode_solvers/euler.hpp create mode 100644 src/volesti/include/ode_solvers/generalized_leapfrog.hpp create mode 100644 src/volesti/include/ode_solvers/implicit_midpoint.hpp create mode 100644 src/volesti/include/ode_solvers/integral_collocation.hpp create mode 100644 src/volesti/include/ode_solvers/leapfrog.hpp create mode 100644 src/volesti/include/ode_solvers/ode_solvers.hpp create mode 100644 src/volesti/include/ode_solvers/oracle_functors.hpp create mode 100644 src/volesti/include/ode_solvers/oracle_functors_rcpp.hpp create mode 100644 src/volesti/include/ode_solvers/randomized_midpoint.hpp create mode 100644 src/volesti/include/ode_solvers/richardson_extrapolation.hpp create mode 100644 src/volesti/include/ode_solvers/runge_kutta.hpp create mode 100644 src/volesti/include/optimization/simulated_annealing.hpp create mode 100644 src/volesti/include/optimization/sliding_window.hpp create mode 100644 src/volesti/include/preprocess/crhmc/analytic_center.h create mode 100644 src/volesti/include/preprocess/crhmc/constraint_problem.h create mode 100644 src/volesti/include/preprocess/crhmc/crhmc_input.h create mode 100644 src/volesti/include/preprocess/crhmc/crhmc_problem.h create mode 100644 src/volesti/include/preprocess/crhmc/crhmc_utils.h create mode 100644 src/volesti/include/preprocess/crhmc/lewis_center.h create mode 100644 src/volesti/include/preprocess/crhmc/opts.h create mode 100644 src/volesti/include/preprocess/crhmc/two_sided_barrier.h create mode 100644 src/volesti/include/preprocess/crhmc/weighted_two_sided_barrier.h create mode 100644 src/volesti/include/preprocess/estimate_L_smooth_parameter.hpp create mode 100644 src/volesti/include/preprocess/max_inscribed_ball.hpp create mode 100644 src/volesti/include/preprocess/max_inscribed_ellipsoid.hpp create mode 100644 src/volesti/include/preprocess/max_inscribed_ellipsoid_rounding.hpp create mode 100644 src/volesti/include/preprocess/min_sampling_covering_ellipsoid_rounding.hpp create mode 100644 src/volesti/include/preprocess/svd_rounding.hpp create mode 100644 src/volesti/include/random_walks/boltzmann_hmc_walk.hpp create mode 100644 src/volesti/include/random_walks/boundary_cdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/boundary_rdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/compute_diameter.hpp create mode 100644 src/volesti/include/random_walks/crhmc/additional_units/auto_tuner.hpp create mode 100644 src/volesti/include/random_walks/crhmc/additional_units/dynamic_regularizer.hpp create mode 100644 src/volesti/include/random_walks/crhmc/additional_units/dynamic_step_size.hpp create mode 100644 src/volesti/include/random_walks/crhmc/additional_units/dynamic_weight.hpp create mode 100644 src/volesti/include/random_walks/crhmc/crhmc_walk.hpp create mode 100644 src/volesti/include/random_walks/crhmc/hamiltonian.hpp create mode 100644 src/volesti/include/random_walks/ellipsoid_walks/README.md create mode 100644 src/volesti/include/random_walks/ellipsoid_walks/dikin_walker.h create mode 100644 src/volesti/include/random_walks/ellipsoid_walks/john_walker.h create mode 100644 src/volesti/include/random_walks/ellipsoid_walks/math_functions.h create mode 100644 src/volesti/include/random_walks/ellipsoid_walks/vaidya_walker.h create mode 100644 src/volesti/include/random_walks/exponential_hamiltonian_monte_carlo_exact_walk.hpp create mode 100644 src/volesti/include/random_walks/gaussian_accelerated_billiard_walk.hpp create mode 100644 src/volesti/include/random_walks/gaussian_ball_walk.hpp create mode 100644 src/volesti/include/random_walks/gaussian_cdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/gaussian_hamiltonian_monte_carlo_exact_walk.hpp create mode 100644 src/volesti/include/random_walks/gaussian_helpers.hpp create mode 100644 src/volesti/include/random_walks/gaussian_rdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/hamiltonian_monte_carlo_walk.hpp create mode 100644 src/volesti/include/random_walks/langevin_walk.hpp create mode 100644 src/volesti/include/random_walks/multithread_walks.hpp create mode 100644 src/volesti/include/random_walks/nuts_hmc_walk.hpp create mode 100644 src/volesti/include/random_walks/random_walks.hpp create mode 100644 src/volesti/include/random_walks/uniform_accelerated_billiard_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_accelerated_billiard_walk_parallel.hpp create mode 100644 src/volesti/include/random_walks/uniform_ball_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_billiard_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_cdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_dikin_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_john_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_rdhr_walk.hpp create mode 100644 src/volesti/include/random_walks/uniform_vaidya_walk.hpp create mode 100644 src/volesti/include/root_finders/mp_solve_wrapper.hpp create mode 100644 src/volesti/include/root_finders/newton_raphson.hpp create mode 100644 src/volesti/include/root_finders/quadratic_polynomial_solvers.hpp create mode 100644 src/volesti/include/root_finders/root_finders.hpp create mode 100644 src/volesti/include/sampling/ellipsoid.hpp create mode 100644 src/volesti/include/sampling/mmcs.hpp create mode 100644 src/volesti/include/sampling/parallel_mmcs.hpp create mode 100644 src/volesti/include/sampling/random_point_generators.hpp create mode 100644 src/volesti/include/sampling/random_point_generators_multithread.hpp create mode 100644 src/volesti/include/sampling/sample_correlation_matrices.hpp create mode 100644 src/volesti/include/sampling/sampling.hpp create mode 100644 src/volesti/include/sampling/simplex.hpp create mode 100644 src/volesti/include/sampling/sphere.hpp create mode 100644 src/volesti/include/sos/NonSymmetricIPM.h create mode 100644 src/volesti/include/sos/NonSymmetricIPM.hpp create mode 100644 src/volesti/include/sos/README.md create mode 100644 src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.h create mode 100644 src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/FullSpaceBarrier.h create mode 100644 src/volesti/include/sos/barriers/FullSpaceBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.h create mode 100644 src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/LHSCB.h create mode 100644 src/volesti/include/sos/barriers/LHSCB.hpp create mode 100644 src/volesti/include/sos/barriers/LPStandardBarrier.h create mode 100644 src/volesti/include/sos/barriers/LPStandardBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/ProductBarrier.h create mode 100644 src/volesti/include/sos/barriers/ProductBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/SumBarrier.h create mode 100644 src/volesti/include/sos/barriers/SumBarrier.hpp create mode 100644 src/volesti/include/sos/barriers/ZeroSpaceBarrier.h create mode 100644 src/volesti/include/sos/barriers/ZeroSpaceBarrier.hpp create mode 100644 src/volesti/include/sos/config/config.json create mode 100644 src/volesti/include/sos/gsoc_history.md create mode 100644 src/volesti/include/sos/include/cxxtimer.hpp create mode 100644 src/volesti/include/sos/include/matplotlib-cpp/matplotlibcpp.h create mode 100644 src/volesti/include/sos/plot_saved.png create mode 100644 src/volesti/include/sos/utils.cpp create mode 100644 src/volesti/include/sos/utils.h create mode 100644 src/volesti/include/volume/copulas.h create mode 100644 src/volesti/include/volume/exact_vols.h create mode 100644 src/volesti/include/volume/math_helpers.hpp create mode 100644 src/volesti/include/volume/rotating.hpp create mode 100644 src/volesti/include/volume/sampling_policies.hpp create mode 100644 src/volesti/include/volume/volume_cooling_balls.hpp create mode 100644 src/volesti/include/volume/volume_cooling_gaussians.hpp create mode 100644 src/volesti/include/volume/volume_cooling_hpoly.hpp create mode 100644 src/volesti/include/volume/volume_sequence_of_balls.hpp diff --git a/.gitmodules b/.gitmodules deleted file mode 100644 index 42627744..00000000 --- a/.gitmodules +++ /dev/null @@ -1,3 +0,0 @@ -[submodule "src/volesti"] - path = src/volesti - url = https://github.com/GeomScale/volesti.git diff --git a/inst/AUTHORS b/inst/AUTHORS index 956eb325..213c3a8f 100644 --- a/inst/AUTHORS +++ b/inst/AUTHORS @@ -1,4 +1,5 @@ -1. Bojan Nikolic . We have modified the implementations of Khachiyan's Algorithm by B. Nikolic from bnmin1-1.11 package for the Computation of Minimum Volume Enclosing Ellipsoids in /src/external/minimum_ellipsoid. - -2. Kjell Konis , Stefan I. Larimore and Timothy A. Davis , Kjell Eikland, Michel Berkelaar, Richard Stallman, Authors of lpsolve package , in /src/external/lpsolve. - +1. Kjell Konis , Stefan I. Larimore and Timothy A. Davis , Kjell Eikland, Michel Berkelaar, Richard Stallman, Authors of lpsolve package , in /src/external/lpSolve. +2. Bojan Nikolic . We have modified the implementations of Khachiyan's Algorithm by B. Nikolic from bnmin1-1.11 package for the Computation of Minimum Volume Enclosing Ellipsoids in /src/external/minimum_ellipsoid. +3. Authors of Spectra are described in detail here: https://github.com/yixuan/spectra/blob/master/AUTHORS.md +4. Author of PackedCSparse is Yin Tat Lee as described in https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/tree/master/code/solver/PackedCSparse +5. David H. Bailey author of qd library \ No newline at end of file diff --git a/inst/COPYRIGHTS b/inst/COPYRIGHTS index 5f83b029..a0c8033f 100644 --- a/inst/COPYRIGHTS +++ b/inst/COPYRIGHTS @@ -1,6 +1,9 @@ All files in src/external are taken from -(i) lpsolve (https://cran.r-project.org/package=lpSolve) version 5.6.20, -(ii) minimum_ellipsoid (or bnmin1) (https://www.mrao.cam.ac.uk/~bn204/oof/bnmin1.html) version 1.11 +1. lpsolve (https://cran.r-project.org/package=lpSolve) version 5.6.20, +2. minimum_ellipsoid (or bnmin1) (https://www.mrao.cam.ac.uk/~bn204/oof/bnmin1.html) version 1.11 +3. Spectra (https://github.com/yixuan/spectra/tree/master) version 0.8.1 +4. PackedCSparse (https://github.com/ConstrainedSampler/PolytopeSamplerMatlab) +5. qd (https://www.davidhbailey.com/dhbsoftware/) version 2.3.22 Copyrights and modification details are explicitly described at the beginning of each of those files. diff --git a/src/Makevars b/src/Makevars index e0ef376b..eb27e4d5 100644 --- a/src/Makevars +++ b/src/Makevars @@ -1,16 +1,16 @@ -PKG_CPPFLAGS=-Ivolesti/external -Iexternal/lpSolve/src -Ivolesti/external/minimum_ellipsoid -Ivolesti/include -Ivolesti/include/convex_bodies/spectrahedra +PKG_CPPFLAGS=-Iexternal -Iexternal/lpSolve/src -Iexternal/minimum_ellipsoid -Ivolesti/include -Ivolesti/include/convex_bodies/spectrahedra PKG_CXXFLAGS= -DBOOST_NO_AUTO_PTR -DDISABLE_NLP_ORACLES -PKG_LIBS=-Lexternal/lpSolve/src -llp_solve -Lvolesti/external/PackedCSparse/qd -lqd $(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS) +PKG_LIBS=-Lexternal/lpSolve/src -llp_solve -Lexternal/PackedCSparse/qd -lqd $(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS) -$(SHLIB): external/lpSolve/src/liblp_solve.a volesti/external/PackedCSparse/qd/libqd.a +$(SHLIB): external/lpSolve/src/liblp_solve.a external/PackedCSparse/qd/libqd.a external/lpSolve/src/liblp_solve.a: @(cd external/lpSolve/src && $(MAKE) liblp_solve.a \ CC="$(CC)" CPPFLAGS="$(CPPFLAGS)" CFLAGS="$(CFLAGS)" \ CPICFLAGS="$(CPICFLAGS)" AR="$(AR)" RANLIB="$(RANLIB)") -volesti/external/PackedCSparse/qd/libqd.a: - @(cd volesti/external/PackedCSparse/qd && $(MAKE) libqd.a \ +external/PackedCSparse/qd/libqd.a: + @(cd external/PackedCSparse/qd && $(MAKE) libqd.a \ CC="$(CC)" CPPFLAGS="$(CPPFLAGS)" CFLAGS="$(CFLAGS)" \ CPICFLAGS="$(CPICFLAGS)" AR="$(AR)") diff --git a/src/Makevars.win b/src/Makevars.win index 554a2bce..2a06a4d5 100644 --- a/src/Makevars.win +++ b/src/Makevars.win @@ -1,9 +1,9 @@ -PKG_CPPFLAGS=-Ivolesti/external -Iexternal/lpSolve/src -Ivolesti/external/minimum_ellipsoid -Ivolesti/include -Ivolesti/include/convex_bodies/spectrahedra +PKG_CPPFLAGS=-Iexternal -Iexternal/lpSolve/src -Iexternal/minimum_ellipsoid -Ivolesti/include -Ivolesti/include/convex_bodies/spectrahedra PKG_CXXFLAGS= -lm -ldl -DBOOST_NO_AUTO_PTR -DDISABLE_NLP_ORACLES -PKG_LIBS=-Lexternal/lpSolve/src -llp_solve -Lvolesti/external/PackedCSparse/qd -lqd $(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS) +PKG_LIBS=-Lexternal/lpSolve/src -llp_solve -Lexternal/PackedCSparse/qd -lqd $(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS) -$(SHLIB): external/lpSolve/src/liblp_solve.a volesti/external/PackedCSparse/qd/libqd.a +$(SHLIB): external/lpSolve/src/liblp_solve.a external/PackedCSparse/qd/libqd.a external/lpSolve/src/liblp_solve.a: @(cd external/lpSolve/src && $(MAKE) liblp_solve.a \ @@ -11,7 +11,7 @@ external/lpSolve/src/liblp_solve.a: CFLAGS="$(CFLAGS)" CPICFLAGS="$(CPICFLAGS)" AR="$(AR)" \ RANLIB="$(RANLIB)") -volesti/external/PackedCSparse/qd/libqd.a: - @(cd volesti/external/PackedCSparse/qd && $(MAKE) libqd.a \ +external/PackedCSparse/qd/libqd.a: + @(cd external/PackedCSparse/qd && $(MAKE) libqd.a \ CC="$(CC)" CPPFLAGS="$(CPPFLAGS)" CFLAGS="$(CFLAGS)" \ CPICFLAGS="$(CPICFLAGS)" AR="$(AR)") \ No newline at end of file diff --git a/src/external/PackedCSparse/FloatArray.h b/src/external/PackedCSparse/FloatArray.h new file mode 100644 index 00000000..28d1c5da --- /dev/null +++ b/src/external/PackedCSparse/FloatArray.h @@ -0,0 +1,307 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include +#include +#include +namespace PackedCSparse { + template + struct BaseImpl + { + T x[k]; + + BaseImpl() {}; + + BaseImpl(const T& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = rhs; + } + + BaseImpl operator+(const BaseImpl& rhs) const + { + BaseImpl lhs; + for (size_t i = 0; i < k; i++) + lhs.x[i] = x[i] + rhs.x[i]; + return lhs; + } + + BaseImpl operator-(const BaseImpl& rhs) const + { + BaseImpl lhs; + for (size_t i = 0; i < k; i++) + lhs.x[i] = x[i] - rhs.x[i]; + return lhs; + } + + BaseImpl operator*(const BaseImpl& rhs) const + { + BaseImpl lhs; + for (size_t i = 0; i < k; i++) + lhs.x[i] = x[i] * rhs.x[i]; + return lhs; + } + + BaseImpl operator/(const BaseImpl& rhs) const + { + BaseImpl lhs; + for (size_t i = 0; i < k; i++) + lhs.x[i] = x[i] / rhs.x[i]; + return lhs; + } + + BaseImpl& operator+=(const BaseImpl& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] += rhs.x[i]; + return *this; + } + + BaseImpl& operator-=(const BaseImpl& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] -= rhs.x[i]; + return *this; + } + + BaseImpl& operator*=(const BaseImpl& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] *= rhs.x[i]; + return *this; + } + + BaseImpl& operator/=(const BaseImpl& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] /= rhs.x[i]; + return *this; + } + + explicit operator bool() const + { + bool ret = false; + for (size_t i = 0; i < k; i++) + ret = ret || bool(x[i]); + return ret; + } + + static T get(const BaseImpl& a, size_t index) + { + return a.x[index]; + } + + static void set(BaseImpl& a, size_t index, const T& value) + { + a.x[index] = value; + } + + static BaseImpl abs(const BaseImpl& a) + { + BaseImpl out; + for (size_t i = 0; i < k; i++) + out.x[i] = std::abs(a.x[i]); + return out; + } + + static BaseImpl log(const BaseImpl& a) + { + BaseImpl out; + for (size_t i = 0; i < k; i++) + out.x[i] = std::log(a.x[i]); + return out; + } + + static void fmadd(BaseImpl& a, const BaseImpl& b, const BaseImpl& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] += b.x[i] * c.x[i]; + } + + static void fnmadd(BaseImpl& a, const BaseImpl& b, const BaseImpl& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] -= b.x[i] * c.x[i]; + } + + static void fmadd(BaseImpl& a, const BaseImpl& b, const T& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] += b.x[i] * c; + } + + static void fnmadd(BaseImpl& a, const BaseImpl& b, const T& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] -= b.x[i] * c; + } + + static BaseImpl clipped_sqrt(const BaseImpl& a, const T nonpos_output) + { + BaseImpl out; + for (size_t i = 0; i < k; i++) + { + T r = a.x[i]; + if (r > 0) + out.x[i] = sqrt(r); + else + out.x[i] = nonpos_output; + } + return out; + } + + static BaseImpl sign(std::mt19937_64& gen) + { + BaseImpl out; + unsigned long long seed = gen(); + for (size_t i = 0; i < k; i++) + { + out.x[i] = T((2 * ((seed >> i) & 1)) - 1.0); + if ((i & 63) == 63) seed = gen(); + } + return out; + } + + }; + + template + struct BaseScalarImpl + { + static T get(const T& x, size_t index) + { + return x; + } + + static void set(T& x, size_t index, T& value) + { + x = value; + } + + static T abs(const T &x) + { + return ::abs(x); + } + + static T log(const T &x) + { + return ::log(x); + } + + static void fmadd(T& a, const T& b, const T& c) + { + a += b * c; + } + + static void fnmadd(T& a, const T& b, const T& c) + { + a -= b * c; + } + + static T clipped_sqrt(const T& x, const T& nonpos_output) + { + if (x > 0.0) + return sqrt(x); + else + return nonpos_output; + } + + static T sign(std::mt19937_64& gen) + { + unsigned long long seed = gen(); + return T((2 * (seed & 1)) - 1.0); + } + }; + + template + struct FloatTypeSelector + { + using type = typename std::conditional>::type; + using funcImpl = typename std::conditional, BaseImpl>::type; + }; + + #ifdef __AVX2__ + #include "FloatArrayAVX2.h" + #else + template + struct FloatTypeSelector + { + using type = typename std::conditional< k == 1, double, BaseImpl>::type; + using funcImpl = typename std::conditional< k == 1, BaseScalarImpl, BaseImpl>::type; + }; + + template + struct FloatTypeSelector, l> + { + using type = BaseImpl; + using funcImpl = BaseImpl; + }; + #endif + + template + struct FloatTypeSelector, l> + { + using type = BaseImpl; + using funcImpl = BaseImpl; + }; + + template + using FloatArray = typename FloatTypeSelector::type; + + template + using FloatArrayFunc = typename FloatTypeSelector::funcImpl; + + template + auto get(const T& a, size_t index) -> decltype(FloatArrayFunc::get(a, index)) + { + return FloatArrayFunc::get(a, index); + } + + template + void set(T1& a, size_t index, T2 value) + { + FloatArrayFunc::set(a, index, value); + } + + template + void fmadd(T1& a, const T2& b, const T3& c) + { + FloatArrayFunc::fmadd(a, b, c); + } + + template + void fnmadd(T1& a, const T2& b, const T3& c) + { + FloatArrayFunc::fnmadd(a, b, c); + } + + template + T1 clipped_sqrt(const T1& a, const T2 b) + { + return FloatArrayFunc::clipped_sqrt(a, b); + } + + template + T abs(const T& a) + { + return FloatArrayFunc::abs(a); + } + + template + T log(const T& a) + { + return FloatArrayFunc::log(a); + } + + template + T sign(std::mt19937_64& gen) + { + return FloatArrayFunc::sign(gen); + } +} diff --git a/src/external/PackedCSparse/FloatArrayAVX2.h b/src/external/PackedCSparse/FloatArrayAVX2.h new file mode 100644 index 00000000..3ae7592a --- /dev/null +++ b/src/external/PackedCSparse/FloatArrayAVX2.h @@ -0,0 +1,222 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +template + struct m256dArray +{ + __m256d x[k]; + + m256dArray() {}; + + m256dArray(const double rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = _mm256_set1_pd(rhs); + } + + template + m256dArray(const m256dArray& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = rhs.x[i % k2]; + } + + m256dArray operator+(const m256dArray& rhs) const + { + m256dArray out; + for (size_t i = 0; i < k; i++) + out.x[i] = _mm256_add_pd(x[i], rhs.x[i]); + return out; + } + + m256dArray operator-(const m256dArray& rhs) const + { + m256dArray out; + for (size_t i = 0; i < k; i++) + out.x[i] = _mm256_sub_pd(x[i], rhs.x[i]); + return out; + } + + m256dArray operator*(const m256dArray& rhs) const + { + m256dArray out; + for (size_t i = 0; i < k; i++) + out.x[i] = _mm256_mul_pd(x[i], rhs.x[i]); + return out; + } + + m256dArray operator/(const m256dArray& rhs) const + { + m256dArray out; + for (size_t i = 0; i < k; i++) + out.x[i] = _mm256_div_pd(x[i], rhs.x[i]); + return out; + } + + m256dArray& operator+=(const m256dArray& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = _mm256_add_pd(x[i], rhs.x[i]); + return *this; + } + + m256dArray& operator-=(const m256dArray& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = _mm256_sub_pd(x[i], rhs.x[i]); + return *this; + } + + m256dArray& operator*=(const m256dArray& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = _mm256_mul_pd(x[i], rhs.x[i]); + return *this; + } + + m256dArray& operator/=(const m256dArray& rhs) + { + for (size_t i = 0; i < k; i++) + x[i] = _mm256_div_pd(x[i], rhs.x[i]); + return *this; + } + + explicit operator bool() const + { + bool ret = false; + __m256d z = _mm256_set1_pd(0.0); + for (size_t i = 0; i < k; i++) + { + __m256d c = _mm256_cmp_pd(x[i], z, _CMP_EQ_OQ); + ret = ret || (_mm256_movemask_pd(c) != 0xf); + } + return ret; + } + + static double get(const m256dArray& x, size_t index) + { + double y[4]; + _mm256_store_pd(y, x.x[index / 4]); + return y[index & 3]; + } + + static void set(m256dArray& x, size_t index, double value) + { + __m256d v = _mm256_broadcast_sd(&value); + switch (index & 3) + { + case 0: x.x[index / 4] = _mm256_blend_pd(x.x[index / 4], v, 1); break; + case 1: x.x[index / 4] = _mm256_blend_pd(x.x[index / 4], v, 2); break; + case 2: x.x[index / 4] = _mm256_blend_pd(x.x[index / 4], v, 4); break; + default: x.x[index / 4] = _mm256_blend_pd(x.x[index / 4], v, 8); break; + } + } + + static m256dArray abs(const m256dArray& x) + { + const __m256d mask = _mm256_castsi256_pd(_mm256_set1_epi64x(0x7FFFFFFFFFFFFFFF)); + + m256dArray out; + for (size_t i = 0; i < k; i++) + out.x[i] = _mm256_and_pd(x.x[i], mask); + return out; + } + + static m256dArray log(const m256dArray& x) + { + // gcc does not support _mm256_log_pd + // Do it sequentially instead + + //m256dArray out; + //for (size_t i = 0; i < k; i++) + // out.x[i] = _mm256_log_pd(x.x[i]); + + m256dArray out; + for (size_t i = 0; i < 4*k; i++) + set(out, i, std::log(get(x,i))); + return out; + } + + static void fmadd(m256dArray& a, const m256dArray& b, const double& c) + { + auto cx = _mm256_set1_pd(c); + for (size_t i = 0; i < k; i++) + a.x[i] = _mm256_fmadd_pd(b.x[i], cx, a.x[i]); + } + + static void fnmadd(m256dArray& a, const m256dArray& b, const double& c) + { + auto cx = _mm256_set1_pd(c); + for (size_t i = 0; i < k; i++) + a.x[i] = _mm256_fnmadd_pd(b.x[i], cx, a.x[i]); + } + + static void fmadd(m256dArray& a, const m256dArray& b, const m256dArray& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] = _mm256_fmadd_pd(b.x[i], c.x[i], a.x[i]); + } + + static void fnmadd(m256dArray& a, const m256dArray& b, const m256dArray& c) + { + for (size_t i = 0; i < k; i++) + a.x[i] = _mm256_fnmadd_pd(b.x[i], c.x[i], a.x[i]); + } + + static m256dArray clipped_sqrt(const m256dArray& x, const double nonpos_output) + { + m256dArray out; + + const __m256d large = { nonpos_output, nonpos_output, nonpos_output, nonpos_output }; + const __m256d zero = _mm256_setzero_pd(); + for (size_t i = 0; i < k; i++) + { + __m256d xi = x.x[i]; + __m256d mask = _mm256_cmp_pd(xi, zero, _CMP_LE_OS); // mask = (rhs.x[i]<= 0) ? -1 : 0 + out.x[i] = _mm256_blendv_pd(_mm256_sqrt_pd(xi), large, mask); + } + return out; + } + + static m256dArray sign(std::mt19937_64& gen) + { + m256dArray out; + const __m256i bits = _mm256_set_epi64x(1, 2, 4, 8); + const __m256d zero = _mm256_setzero_pd(); + const __m256d pos = _mm256_set_pd(1.0, 1.0, 1.0, 1.0); + const __m256d neg = _mm256_set_pd(-1.0, -1.0, -1.0, -1.0); + + unsigned long long seed = gen(); + for (size_t i = 0; i < k; i++) + { + __m256i s = _mm256_set1_epi64x((seed >> (4 * i)) & 15); + __m256i xi = _mm256_and_si256(s, bits); + __m256d x = _mm256_castsi256_pd(xi); + __m256d mask = _mm256_cmp_pd(x, zero, _CMP_EQ_OQ); // mask = (rhs.x[i] == 0) ? -1 : 0 + out.x[i] = _mm256_blendv_pd(pos, neg, mask); + if ((i & 63) == 63) seed = gen(); + } + return out; + } +}; + +template + struct FloatTypeSelector +{ + static_assert(k == 1 || k % 4 == 0, "Array assumes k = 1 or a multiple of 4"); + using type = typename std::conditional< k == 1, double, m256dArray>::type; + using funcImpl = typename std::conditional< k == 1, BaseScalarImpl, m256dArray>::type; +}; + +template + struct FloatTypeSelector, l> +{ + using type = m256dArray; + using funcImpl = m256dArray; +}; diff --git a/src/external/PackedCSparse/PackedChol.h b/src/external/PackedCSparse/PackedChol.h new file mode 100644 index 00000000..97d10220 --- /dev/null +++ b/src/external/PackedCSparse/PackedChol.h @@ -0,0 +1,308 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include "Eigen/Eigen" +#include "SparseMatrix.h" +#include "chol.h" +#include "leverage.h" +#include "leverageJL.h" +#include "multiply.h" +#include "qd/dd_real.h" +#include +#include +using namespace PackedCSparse; + +template +void get_slice(Tout *out, Tin *in, size_t n, size_t idx) { + for (size_t j = 0; j < n; j++) + out[j] = to_double(get(in[j], idx)); +} + +template +void set_slice(Tout *out, Tin *in, size_t n, size_t idx) { + for (size_t j = 0; j < n; j++) + set(out[j], idx, to_double(in[j])); +} + +template struct PackedChol { + using Tx = double; + using Tx2 = FloatArray; + using Te = dd_real; + + // parameters + SparseMatrix A; + SparseMatrix At; + UniqueAlignedPtr w; + Tx accuracyThreshold = 1e-6; + std::vector + exactIdx; // k size array. Indices we perform high precision calculation + std::vector + numExact; // number of times we perform high precision decompose (length + // k+1, the last one records how many times we do decompose) + bool decomposed = false; + + // preprocess info for different CSparse operations (PackedDouble) + MultiplyOutput H; // cache for H = A W A' + CholOutput L; // cache for L = chol(H) + LeverageOutput diagP; // cache for L = chol(H) + LeverageJLOutput diagPJL; // cache for L = chol(H) + + // preprocess info for different CSparse operations (dd_real) + MultiplyOutput H_exact; // cache for H = A W A' + CholOutput L_exact; // cache for L = chol(H) + LeverageOutput diagP_exact; // cache for L = chol(H) + LeverageJLOutput diagPJL_exact; // cache for L = chol(H) + SparseMatrix Le[k]; // store output of L_exact + + PackedChol(const SparseMatrix &A_) { + A = std::move(A_.clone()); + At = transpose(A); + w.reset(pcs_aligned_new(A.n)); + numExact.resize(k + 1); + } + + void setSeed(unsigned long long seed) { + diagPJL.gen.seed(seed); + diagPJL_exact.gen.seed(seed); + } + + bool allExact() { return exactIdx.size() == k; } + + bool hasExact() { return exactIdx.size() > 0; } + + template Tx2 decompose(const Tv2_ *w_in) { + + Tx2 acc = Tx2(0.0); + + // record w + Ti n = A.n; + + for (Ti j = 0; j < n; j++) { + w[j] = w_in[j]; + } + // compute chol + ++numExact[k]; + if (accuracyThreshold > 0.0 || + !decomposed) // the first time we call, always run the double chol. + { + multiply(H, A, w.get(), At); + chol(L, H); + decomposed = true; + + exactIdx.clear(); + acc = estimateAccuracy(); + for (size_t i = 0; i < k; i++) { + if (get(acc, i) >= + accuracyThreshold) // >= is important for the case accuracyThreshold + // = 0.0, we need to compute everything exactly + exactIdx.push_back(i); + } + } else if (!allExact()) { + exactIdx.clear(); + for (size_t i = 0; i < k; i++) + exactIdx.push_back(i); + } + + if (hasExact()) { + Te *w_exact = new Te[n]; + + for (size_t i : exactIdx) { + ++numExact[i]; + get_slice(w_exact, w.get(), n, i); + multiply(H_exact, A, w_exact, At); + chol(L_exact, H_exact); + + // copy result to Le[i] + if (!Le[i].initialized()) + Le[i] = std::move(L_exact.template clone()); + else { + Ti nz = L_exact.nnz(); + for (Ti s = 0; s < nz; ++s) + Le[i].x[s] = (L_exact.x[s]); + } + } + + delete[] w_exact; + } + return acc; + } + + Tx2 logdet() { + pcs_assert(decomposed, "logdet: Need to call decompose first."); + + Ti m = A.m; + Tx2 ret = Tx2(0); + + if (!allExact()) { + Ti *Lp = L.p.get(); + Tx2 *Lx = L.x.get(); + for (Ti j = 0; j < m; j++) + ret += log(Lx[Lp[j]]); + } + + if (hasExact()) { + for (size_t i : exactIdx) { + Te ret_e = 0.0; + Ti *Lp = Le[i].p.get(); + Te *Lx = Le[i].x.get(); + + for (Ti j = 0; j < m; j++) + ret_e += log(Lx[Lp[j]]); + + set(ret, i, to_double(ret_e)); + } + } + + return ret * Tx2(2.0); + } + + void diagL(Tx2 *out) { + pcs_assert(decomposed, "diagL: Need to call decompose first."); + + Ti m = A.m; + + if (!allExact()) { + Ti *Li = L.i.get(), *Lp = L.p.get(); + Tx2 *Lx = L.x.get(); + for (Ti j = 0; j < m; j++) + out[j] = Lx[Lp[j]]; + } + + if (hasExact()) { + for (size_t i : exactIdx) { + Ti *Lp = Le[i].p.get(); + Te *Lx = Le[i].x.get(); + + for (Ti j = 0; j < m; j++) + set(out[j], i, to_double(Lx[Lp[j]])); + } + } + } + + SparseMatrix getL(Ti i) { + pcs_assert(decomposed, "getL: Need to call decompose first."); + + Ti m = L.m, n = L.n, nz = L.nnz(); + SparseMatrix out(m, n, nz); + + Ti *outp = out.p.get(), *Lp = L.p.get(); + Ti *outi = out.i.get(), *Li = L.i.get(); + + for (Ti s = 0; s <= n; s++) + outp[s] = Lp[s]; + + for (Ti s = 0; s < nz; s++) + outi[s] = Li[s]; + + bool isExact = false; + for (size_t i_ : exactIdx) { + if (i_ == i) + isExact = true; + } + + double *outx = out.x.get(); + if (isExact) { + Te *Lx = Le[i].x.get(); + for (Ti s = 0; s < nz; s++) + outx[s] = to_double(Lx[s]); + } else { + Tx2 *Lx = L.x.get(); + for (Ti s = 0; s < nz; s++) + outx[s] = get(Lx[s], i); + } + + return std::move(out); + } + + void solve(Tx2 *b, Tx2 *out) { + pcs_assert(decomposed, "solve: Need to call decompose first."); + + if (!allExact()) { + lsolve(L, b, out); + ltsolve(L, out, out); + } + + if (hasExact()) { + Ti m = A.m; + Te *b_exact = new Te[m]; + Te *out_exact = new Te[m]; + + for (size_t i : exactIdx) { + get_slice(b_exact, b, m, i); + lsolve(Le[i], b_exact, out_exact); + ltsolve(Le[i], out_exact, out_exact); + set_slice(out, out_exact, m, i); + } + + delete[] b_exact; + delete[] out_exact; + } + }; + + void leverageScoreComplement(Tx2 *out) { + pcs_assert(decomposed, + "leverageScoreComplement: Need to call decompose first."); + + Ti n = A.n, m = A.m; + + if (!allExact()) { + Tx2 T1 = Tx2(1.0), T2 = Tx2(2.0); + leverage(diagP, L, A, At); + + Tx2 *tau = diagP.x.get(); + for (Ti j = 0; j < n; j++) + out[j] = T1 - tau[j] * w[j]; + } + + if (hasExact()) { + Te T1 = Te(1.0), T2 = Te(2.0); + for (size_t i : exactIdx) { + leverage(diagP_exact, Le[i], A, At); + + Te *tau = diagP_exact.x.get(); + for (Ti j = 0; j < n; j++) + set(out[j], i, to_double(T1 - tau[j] * get(w[j], i))); + } + } + } + + void leverageScoreComplementJL(Tx2 *out, size_t JL_k) { + pcs_assert(decomposed, + "leverageScoreComplementJL: Need to call decompose first."); + + Ti m = A.m, n = A.n; + + if (!allExact()) { + Tx2 T1 = Tx2(1.0), T2 = Tx2(2.0); + leverageJL(diagPJL, L, A, At, JL_k); + + Tx2 *tau = diagPJL.x.get(); + for (Ti j = 0; j < n; j++) + out[j] = T1 - tau[j] * w[j]; + } + + if (hasExact()) { + Te T1 = Te(1.0), T2 = Te(2.0); + for (size_t i : exactIdx) { + leverageJL(diagPJL_exact, Le[i], A, At, JL_k); + + Te *tau = diagPJL_exact.x.get(); + for (Ti j = 0; j < n; j++) + set(out[j], i, to_double(T1 - tau[j] * get(w[j], i))); + } + } + } + + Tx2 estimateAccuracy() { + pcs_assert(decomposed, "estimateAccuracy: Need to call decompose first."); + + return cholAccuracy(diagPJL, L, A, At, w.get()); + } +}; diff --git a/src/external/PackedCSparse/SparseMatrix.h b/src/external/PackedCSparse/SparseMatrix.h new file mode 100644 index 00000000..5dca4286 --- /dev/null +++ b/src/external/PackedCSparse/SparseMatrix.h @@ -0,0 +1,230 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis +#pragma once +#include +#include +#include "FloatArray.h" + +namespace PackedCSparse { + static void pcs_assert(bool value, const char* message) + { + if (value == false) + throw std::logic_error(message); + } + + template + T* pcs_aligned_new(size_t size) + { + int alignment = 64; // size of memory cache line + int offset = alignment - 1 + sizeof(void*); + void* p1 = (void*)new char[size * sizeof(T) + offset]; + void** p2 = (void**)(((size_t)(p1)+offset) & ~(alignment - 1)); + p2[-1] = p1; + return (T*)p2; + } + + template + struct AlignedDeleter + { + void operator()(T* p) const + { + delete[](char*)(((void**)p)[-1]); + } + }; + + template + using UniqueAlignedPtr = std::unique_ptr>; + + template + using UniquePtr = std::unique_ptr; + + // Tx = Type for entries, Ti = Type for indices. + // if Tx == bool, the matrix stores only sparsity information + template + struct SparseMatrix + { + Ti m = 0; /* number of rows */ + Ti n = 0; /* number of columns */ + UniquePtr p; /* column pointers (size n+1) */ + UniquePtr i; /* row indices, size nnz */ + UniqueAlignedPtr x; /* numerical values, size nnz */ + + SparseMatrix() = default; + + SparseMatrix(Ti m_, Ti n_, Ti nzmax_) + { + initialize(m_, n_, nzmax_); + } + + bool initialized() const + { + return p && i; + } + + void initialize(Ti m_, Ti n_, Ti nzmax) + { + if (nzmax < 1) nzmax = 1; + m = m_; n = n_; + p.reset(new Ti[n + 1]); + i.reset(new Ti[nzmax]); + if (!std::is_same::value) + x.reset(pcs_aligned_new(nzmax)); + } + + Ti nnz() const + { + return p[n]; + } + + template + SparseMatrix clone() const + { + SparseMatrix C(m, n, nnz()); + Ti* Ap = p.get(), * Ai = i.get(); Tx* Ax = x.get(); + Ti2* Cp = C.p.get(), * Ci = C.i.get(); Tx2* Cx = C.x.get(); + + for (Ti s = 0; s <= n; s++) + Cp[s] = Ti2(Ap[s]); + + Ti nz = nnz(); + for (Ti s = 0; s < nz; s++) + Ci[s] = Ti2(Ai[s]); + + if (Cx) + { + for (Ti s = 0; s < nz; s++) + Cx[s] = Ax? Tx2(Ax[s]): Tx2(1.0); + } + + return C; + } + }; + + template + struct DenseVector + { + Ti n = 0; /* number of columns */ + UniqueAlignedPtr x; /* numerical values, size nnz */ + + DenseVector() = default; + + DenseVector(Ti n_) + { + initialize(n_); + } + + bool initialized() const + { + return bool(x); + } + + void initialize(Ti n_) + { + n = n_; + x.reset(pcs_aligned_new(n_)); + } + }; + + + // basic functions + template + SparseMatrix speye(Ti n, Tx* d = nullptr) + { + SparseMatrix D(n, n, n); + + for (Ti k = 0; k < n; k++) + { + D.i[k] = k; + D.p[k] = k; + } + D.p[n] = n; + + Tx Tx1 = Tx(1.0); + for (Ti k = 0; k < n; k++) + D.x[k] = (d ? d[k] : Tx1); + return D; + } + + // Solve L out = x + // Input: L in Tx^{n by n}, x in Tx2^{n} + // Output: out in Tx2^{n}. + // If out is provided, we will output to out. Else, output to x. + template + void lsolve(const SparseMatrix& L, Tx2* x, Tx2* out = nullptr) + { + pcs_assert(L.initialized(), "lsolve: bad inputs."); + pcs_assert(L.n == L.m, "lsolve: dimensions mismatch."); + + Ti n = L.n, * Lp = L.p.get(), * Li = L.i.get(); Tx* Lx = L.x.get(); + + if (!out) out = x; + if (x != out) std::copy(x, x + n, out); + + for (Ti j = 0; j < n; j++) + { + Tx2 out_j = out[j] / Lx[Lp[j]]; + out[j] = out_j; + + Ti p_start = Lp[j] + 1, p_end = Lp[j + 1]; + for (Ti p = p_start; p < p_end; p++) + { //out[Li[p]] -= Lx[p] * out[j]; + fnmadd(out[Li[p]], out_j, Lx[p]); + } + } + } + + // Solve L' out = x + // Input: L in Tx^{n by n}, x in Tx2^{n} + // Output: out in Tx2^{n}. + // If out is provided, we will output to out. Else, output to x. + template + void ltsolve(const SparseMatrix& L, Tx2* x, Tx2* out = nullptr) + { + pcs_assert(L.initialized(), "ltsolve: bad inputs."); + pcs_assert(L.n == L.m, "ltsolve: dimensions mismatch."); + + Ti n = L.n, * Lp = L.p.get(), * Li = L.i.get(); Tx* Lx = L.x.get(); + + if (!out) out = x; + if (x != out) std::copy(x, x + n, out); + + for (Ti j = n - 1; j != -1; j--) + { + Tx2 out_j = out[j]; + + Ti p_start = Lp[j] + 1, p_end = Lp[j + 1]; + for (Ti p = p_start; p < p_end; p++) + { //out[j] -= Lx[p] * out[Li[p]]; + fnmadd(out_j, out[Li[p]], Lx[p]); + } + + out[j] = out_j / Tx2(Lx[Lp[j]]); + } + } + + // Update y <-- y + A x + // Input: A in Tx^{n by n}, x, y in Tx2^{n} + template + void gaxpy(const SparseMatrix& A, const Tx2* x, Tx2* y) + { + pcs_assert(A.initialized(), "gaxpy: bad inputs."); + Ti m = A.m, n = A.n, * Ap = A.p.get(), * Ai = A.i.get(); Tx* Ax = A.x.get(); + + for (Ti j = 0; j < n; j++) + { + Tx2 x_j = x[j]; + + Ti p_start = Ap[j], p_end = Ap[j + 1]; + for (Ti p = p_start; p < p_end; p++) + { //y[Ai[p]] += Ax[p] * x[j]; + fmadd(y[Ai[p]], x_j, Ax[p]); + } + } + } +}; diff --git a/src/external/PackedCSparse/add.h b/src/external/PackedCSparse/add.h new file mode 100644 index 00000000..16195722 --- /dev/null +++ b/src/external/PackedCSparse/add.h @@ -0,0 +1,102 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include +#include "SparseMatrix.h" + +// Problem: +// Compute M = A + B + +// Algorithm: +// M = 0 +// M(A != 0) += A(A != 0) +// M(B != 0) += B(A != 0) + +namespace PackedCSparse { + template + struct AddOutput : SparseMatrix + { + UniquePtr forwardA; + UniquePtr forwardB; + + template + void initialize(const SparseMatrix& A, const SparseMatrix& B) + { + pcs_assert(A.initialized() && B.initialized(), "add: bad inputs."); + pcs_assert(A.n == B.n && A.m == B.m, "add: dimensions mismatch."); + + Ti m = A.m, n = A.n; + Ti Anz = A.nnz(); Ti* Ap = A.p.get(), * Ai = A.i.get(); + Ti Bnz = B.nnz(); Ti* Bp = B.p.get(), * Bi = B.i.get(); + this->m = A.m; this->n = A.n; + + std::vector Ci; + Ti* Cp = new Ti[n + 1]; + forwardA.reset(new Ti[Anz]); + forwardB.reset(new Ti[Bnz]); + + Cp[0] = 0; + for (Ti i = 0; i < n; i++) + { + Ti s1 = Ap[i], s2 = Bp[i], end1 = Ap[i + 1], end2 = Bp[i + 1]; + while ((s1 < end1) || (s2 < end2)) + { + Ti q = Ti(Ci.size()); + Ti i1 = (s1 < end1) ? Ai[s1] : m; + Ti i2 = (s2 < end2) ? Bi[s2] : m; + Ti min_i = std::min(i1, i2); + Ci.push_back(min_i); + + if (i1 == min_i) + forwardA[s1++] = q; + + if (i2 == min_i) + forwardB[s2++] = q; + } + Cp[i + 1] = Ti(Ci.size()); + } + + this->p.reset(Cp); + this->i.reset(new Ti[Ci.size()]); + this->x.reset(pcs_aligned_new(Ci.size())); + std::copy(Ci.begin(), Ci.end(), this->i.get()); + } + }; + + template + void add(AddOutput& o, const SparseMatrix& A, const SparseMatrix& B) + { + if (!o.initialized()) + o.initialize(A, B); + + Ti m = o.m, n = o.n; + Ti Anz = A.nnz(); Ti* Ap = A.p.get(), * Ai = A.i.get(); Tx* Ax = A.x.get(); + Ti Bnz = B.nnz(); Ti* Bp = B.p.get(), * Bi = B.i.get(); Tx* Bx = B.x.get(); + Ti Cnz = o.nnz(); Ti* Cp = o.p.get(), * Ci = o.i.get(); Tx2* Cx = o.x.get(); + Ti* forwardA = o.forwardA.get(), *forwardB = o.forwardB.get(); + + for (Ti s = 0; s < Cnz; s++) + Cx[s] = 0; + + for (Ti s = 0; s < Anz; s++) + Cx[forwardA[s]] = Ax[s]; + + for (Ti s = 0; s < Bnz; s++) + Cx[forwardB[s]] += Bx[s]; + } + + template + AddOutput add(const SparseMatrix& A, const SparseMatrix& B) + { + AddOutput o; + add(o, A, B); + return o; + } +} diff --git a/src/external/PackedCSparse/chol.h b/src/external/PackedCSparse/chol.h new file mode 100644 index 00000000..d6001ae2 --- /dev/null +++ b/src/external/PackedCSparse/chol.h @@ -0,0 +1,247 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include +#include +#include "SparseMatrix.h" +#include "transpose.h" + +// Problem: +// Compute chol(A) + +// Algorithm: +// We need to study this later as this is the bottleneck. +// Document it as a lyx. +// chol_up_looking: +// Compute L row by row +// This is faster when it is compute bound. +// +// chol_left_looking: +// Compute L col by col +// This is faster when it is memory bound. + + +namespace PackedCSparse { + template + struct CholOutput : SparseMatrix + { + TransposeOutput Lt; // sparsity pattern of the Lt + UniquePtr diag; // the index for diagonal element. Ax[diag[k]] is A_kk + UniquePtr c; // c[i] = index the last nonzero on column i in the current L + UniqueAlignedPtr w; // the row of L we are computing + + // The cost of this is roughly 3 times larger than chol + // One can optimize it by using other data structure + void initialize(const SparseMatrix& A) + { + pcs_assert(A.initialized(), "chol: bad inputs."); + pcs_assert(A.n == A.m, "chol: dimensions mismatch."); + + Ti n = A.n, * Ap = A.p.get(), * Ai = A.i.get(); + + // initialize + this->diag.reset(new Ti[n]); + this->c.reset(new Ti[n]); + this->w.reset(pcs_aligned_new(n)); + + // compute the sparsity pattern of L and diag + using queue = std::priority_queue, std::greater>; + queue q; // sol'n of the current row of L + Ti* mark = new Ti[n]; // used to prevent same indices push to q twice + std::vector* cols = new std::vector[n]; // stores the non-zeros of each col of L + Ti nz = 0, Anz = Ap[n]; + + for (Ti i = 0; i < n; i++) + mark[i] = -1; + + // for each row of A + for (Ti i = 0; i < n; i++) + { // push i-th row of A, called a_12, into mark + Ti s; + for (s = Ap[i]; s < Ap[i + 1]; s++) + { + Ti j = Ai[s]; + if (j >= i) break; + + q.push(j); + mark[j] = i; + } + if (s >= Anz) // this case happens only if the diagonal is 0. No cholesky in this case. + this->diag[i] = 0; + else + this->diag[i] = s; + + // Solve L_11 l_12 = a_12 + while (!q.empty()) + { + Ti j = q.top(); + + for (Ti k : cols[j]) + { + if (mark[k] != i) + { + q.push(k); + mark[k] = i; + } + } + q.pop(); + + // update j col + cols[j].push_back(i); + ++nz; + } + + // diag + cols[i].push_back(i); + ++nz; + } + delete[] mark; + + // write it as the compress form + SparseMatrix::initialize(n, n, nz); + + Ti s_start = 0; Ti s = 0; + for (Ti i = 0; i < n; i++) + { + this->p[i] = s_start; + for (Ti k : cols[i]) + this->i[s++] = k; + s_start += Ti(cols[i].size()); + } + this->p[n] = s_start; + delete[] cols; + + this->Lt = transpose(*this); + + // initialize w to 0 + Tx Tv0 = Tx(0); + for (Ti k = 0; k < n; k++) + w[k] = Tv0; + } + }; + + template + void chol(CholOutput& o, const SparseMatrix& A) + { + if (!o.initialized()) + o.initialize(A); + + //chol_up_looking(o, A); + chol_left_looking(o, A); + } + + template + void chol_up_looking(CholOutput& o, const SparseMatrix& A) + { + Ti *Ap = A.p.get(), * Ai = A.i.get(); Tx* Ax = A.x.get(); + Ti nzmax = o.nzmax; Ti n = A.n; + Ti *Lp = o.p.get(); Ti* Li = o.i.get(); + Ti *Ltp = o.Lt.p.get(); Ti* Lti = o.Lt.i.get(); + + Tx T0 = Tx(0); + Tx* Lx = o.x.get(); Tx* w = o.w.get(); Ti* c = o.c.get(); + Ti* diag = o.diag.get(); + + Ti* Lti_ptr = Lti; + for (Ti k = 0; k < n; ++k) + { + c[k] = Lp[k]; + + Ti s_end = diag[k]; + for (Ti s = Ap[k]; s < s_end; ++s) + w[Ai[s]] = Ax[s]; + + // Solve L_11 l_12 = a_12 + Tx d = Ax[s_end]; Ti i; + for (; (i = *(Lti_ptr++)) < k;) + { + Ti dLi = Lp[i], ci = c[i]++; + Tx Lki = w[i] / Lx[dLi]; + w[i] = T0; // maintain x = 0 for the (k+1) iteration + + for (Ti q = dLi + 1; q < ci; ++q) + fnmadd(w[Li[q]], Lx[q], Lki); + + d -= Lki * Lki; + Lx[ci] = Lki; + } + + // l_22 = sqrt(a22 - ) + Lx[c[k]++] = clipped_sqrt(d); + } + } + + template + void chol_left_looking(CholOutput& o, const SparseMatrix& A) + { + Ti* Ap = A.p.get(), * Ai = A.i.get(); Tx* Ax = A.x.get(); + Ti nzmax = o.nnz(); Ti n = A.n; + Ti* Lp = o.p.get(); Ti* Li = o.i.get(); + Ti* Ltp = o.Lt.p.get(); Ti* Lti = o.Lt.i.get(); + + Tx T0 = Tx(0), T1 = Tx(1); + Tx* Lx = o.x.get(); + Tx* w = o.w.get(); Ti* c = o.c.get(); + Ti* diag = o.diag.get(); + + for (Ti j = 0; j < n; ++j) + { + c[j] = Lp[j]; + + // x = A_{j:n, j} + { + Ti is_start = diag[j], is_end = Ap[j + 1]; + for (Ti is = is_start; is < is_end; ++is) + w[Ai[is]] = Ax[is]; + } + + // for each p in L_{j, 1:j-1} + Ti ps_start = Ltp[j], ps_end = Ltp[j + 1] - 1; + for (Ti ps = ps_start; ps < ps_end; ++ps) + { + Ti p = Lti[ps]; + Ti cp = c[p]++; + Tx Ljp = Lx[cp]; + + // for each i in L_{j:n,p} + Ti is_start = cp, is_end = Lp[p + 1]; + for (Ti is = is_start; is < is_end; ++is) + { + Ti i = Li[is]; + fnmadd(w[i], Lx[is], Ljp); + } + } + + Tx Ljj = clipped_sqrt(w[j], 1e128); + Lx[c[j]++] = Ljj; + Tx inv_Ljj = T1 / Ljj; + w[j] = T0; + + // for each i in L_{:,j} + { + Ti is_start = Lp[j] + 1, is_end = Lp[j + 1]; + for (Ti is = is_start; is < is_end; ++is) + { + Ti i = Li[is]; + Lx[is] = w[i] * inv_Ljj; + w[i] = T0; + } + } + } + } + + template + CholOutput chol(const SparseMatrix& A) + { + CholOutput o; + chol(o, A); + return o; + } +} diff --git a/src/external/PackedCSparse/leverage.h b/src/external/PackedCSparse/leverage.h new file mode 100644 index 00000000..69549308 --- /dev/null +++ b/src/external/PackedCSparse/leverage.h @@ -0,0 +1,65 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include "SparseMatrix.h" +#include "projinv.h" +#include "outerprod.h" + +// Problem: +// Compute M = diag(A' inv(LL') A) + +namespace PackedCSparse { + template + struct LeverageOutput : DenseVector + { + ProjinvOutput Hinv; // Hinv = inv(H)|_L + OuterprodOutput tau; // tau = diag(A' Hinv A) + + template + void initialize(const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At) + { + pcs_assert(L.initialized() && A.initialized() && At.initialized(), "leverage: bad inputs."); + pcs_assert(L.m == L.n && L.n == A.m && L.n == At.n && A.n == At.m, "leverage: dimensions mismatch."); + DenseVector::initialize(A.n); + Hinv.initialize(L); + tau.initialize(A, Hinv, At); + } + }; + + template + void leverage(LeverageOutput& o, const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At) + { + if (!o.initialized()) + o.initialize(L, A, At); + + Tx T1 = Tx(1.0), T2 = Tx(2.0); + projinv(o.Hinv, L); + + Ti m = A.m, n = A.n; + Ti* Sp = o.Hinv.p.get(); Tx* Sv = o.Hinv.x.get(); + for (Ti k = 0; k < m; ++k) + Sv[Sp[k]] /= T2; + + outerprod(o.tau, A, o.Hinv, At); + + Tx* x = o.x.get(), * tau = o.tau.x.get(); + for (Ti j = 0; j < n; j++) + x[j] = T2 * tau[j]; + } + + + template + LeverageOutput leverage(const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At) + { + LeverageOutput o; + leverage(o, L, A, At); + return o; + } +} diff --git a/src/external/PackedCSparse/leverageJL.h b/src/external/PackedCSparse/leverageJL.h new file mode 100644 index 00000000..a5001bcd --- /dev/null +++ b/src/external/PackedCSparse/leverageJL.h @@ -0,0 +1,148 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include +#include "SparseMatrix.h" + +// Problem: +// Approximate M = diag(A' inv(LL') A) +namespace PackedCSparse { + const size_t JLPackedSize = 4; + + template + struct LeverageJLOutput : DenseVector + { + UniqueAlignedPtr d; // random direction d + UniqueAlignedPtr L_d; // random direction d + UniqueAlignedPtr AtL_d; // A' L^{-1} d + Ti m = 0; + std::mt19937_64 gen; + + template + void initialize(const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At) + { + pcs_assert(L.initialized() && A.initialized() && At.initialized(), "leverageJL: bad inputs."); + pcs_assert(L.m == L.n && L.n == A.m && L.n == At.n && A.n == At.m, "leverageJL: dimensions mismatch."); + this->n = A.n; this->m = A.m; + this->x.reset(pcs_aligned_new(this->n)); + this->d.reset(pcs_aligned_new(this->m * 2 * JLPackedSize)); + this->L_d.reset(pcs_aligned_new(this->m * 2 * JLPackedSize)); + this->AtL_d.reset(pcs_aligned_new(this->n * 2 * JLPackedSize)); + } + }; + + // compute sum_{j=1}^{k} (A' L^{-T} u_j) .* (A' L^{-T} u_j) + template + void projectionJL(LeverageJLOutput& o, const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At) + { + Ti m = A.m, n = A.n; + Tx T0 = Tx(0.0), T1 = Tx(1.0); + Tx* d = o.d.get(), * L_d = o.L_d.get(), * AtL_d = o.AtL_d.get(), * x = o.x.get(); + + for (Ti i = 0; i < m * k; i++) + d[i] = sign(o.gen); + + for (Ti i = 0; i < n * k; i++) + AtL_d[i] = T0; + + ltsolve(L, (BaseImpl*)d, (BaseImpl*)L_d); + gaxpy(At, (BaseImpl*)L_d, (BaseImpl*)AtL_d); + + for (Ti i = 0; i < n; i++) + { + Tx ret_i = T0; + for (Ti j = 0; j < k; j++) + ret_i += AtL_d[i * k + j] * AtL_d[i * k + j]; + + x[i] += ret_i; + } + } + + template + void leverageJL(LeverageJLOutput& o, const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At, size_t k) + { + if (!o.initialized()) + o.initialize(L, A, At); + + Ti n = A.n; Tx* x = o.x.get(); + for (Ti i = 0; i < n; i++) + x[i] = Tx(0.0); + + constexpr size_t k_step = JLPackedSize; + for(size_t i = 1; i <= k / k_step; i++) + projectionJL(o, L, A, At); + + for (size_t i = 1; i <= k % k_step; i++) + projectionJL<1>(o, L, A, At); + + Tx ratio = Tx(1 / double(k)); + for (Ti i = 0; i < n; i++) + x[i] *= ratio; + } + + template + LeverageJLOutput leverageJL(const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At, size_t k) + { + LeverageJLOutput o; + leverageJL(o, L, A, At, k); + return o; + } + + + // compute (A' L^{-T} u_j) .* (A' L^{-T} v_j) for j = 1, 2, ... k + template + Tx cholAccuracy(LeverageJLOutput& o, const SparseMatrix& L, const SparseMatrix& A, const SparseMatrix& At, const Tx* w) + { + if (!o.initialized()) + o.initialize(L, A, At); + + constexpr Ti k = JLPackedSize; + constexpr Ti k_ = 2 * k; + + + Ti m = A.m, n = A.n; + Tx T0 = Tx(0.0), T1 = Tx(1.0); + Tx* d = o.d.get(), * L_d = o.L_d.get(), * AtL_d = o.AtL_d.get(), * x = o.x.get(); + + std::uniform_real_distribution distribution(-sqrt(3.0),sqrt(3.0)); + for (Ti i = 0; i < m * k_; i++) + d[i] = Tx(distribution(o.gen)); // roughly uniform distribution with variance 1 + + for (Ti i = 0; i < n * k_; i++) + AtL_d[i] = T0; + + ltsolve(L, (BaseImpl*)d, (BaseImpl*)L_d); + gaxpy(At, (BaseImpl*)L_d, (BaseImpl*)AtL_d); + + Tx result[k]; + for (Ti j = 0; j < k; j++) + result[j] = Tx(0.0); + + for (Ti i = 0; i < m; i++) + { + Tx* d = o.d.get() + i * (2 * k); + for (Ti j = 0; j < k; j++) + result[j] -= d[j] * d[j + k]; + } + + for (Ti i = 0; i < n; i++) + { + Tx w_i = w[i]; + for (Ti j = 0; j < k; j++) + result[j] += AtL_d[i * k_ + j] * AtL_d[i * k_ + j + k] * w_i; + } + + Tx est = Tx(0.0); + for (Ti j = 0; j < k; j++) + est += result[j] * result[j]; + + return clipped_sqrt(est/Tx(double(k)), 0.0); + } +} diff --git a/src/external/PackedCSparse/multiply.h b/src/external/PackedCSparse/multiply.h new file mode 100644 index 00000000..cfd53375 --- /dev/null +++ b/src/external/PackedCSparse/multiply.h @@ -0,0 +1,142 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include +#include +#include "SparseMatrix.h" + +// Problem: +// Compute M = A diag(w) B + +// Algorithm: +// Compute M col by col + +namespace PackedCSparse { + template + struct MultiplyOutput : SparseMatrix + { + UniqueAlignedPtr c; + + template + void initialize(const SparseMatrix& A, const SparseMatrix& B) + { + pcs_assert(A.initialized() && B.initialized(), "multiply: bad inputs."); + pcs_assert(A.n == B.m, "multiply: dimensions mismatch."); + + Ti m = A.m, n = B.n; + Ti* Ap = A.p.get(), * Ai = A.i.get(); + Ti* Bp = B.p.get(), * Bi = B.i.get(); + + this->c.reset(pcs_aligned_new(m)); + + Ti* last_j = new Ti[m]; + for (Ti i = 0; i < m; i++) + { + last_j[i] = -1; + this->c[i] = Tx(0.0); + } + + Ti* Cp = new Ti[size_t(n)+1]; + std::vector Ci; + + Cp[0] = 0; + for (Ti j1 = 0; j1 < n; j1++) + { + for (Ti p1 = Bp[j1]; p1 < Bp[j1 + 1]; p1++) + { + Ti j2 = Bi[p1]; + for (Ti p2 = Ap[j2]; p2 < Ap[j2 + 1]; p2++) + { + Ti i = Ai[p2]; + if (last_j[i] != j1) + { + last_j[i] = j1; + Ci.push_back(i); + } + } + } + Cp[j1 + 1] = Ti(Ci.size()); + } + delete[] last_j; + + for (Ti j = 0; j < n; j++) + std::sort(Ci.begin() + Cp[j], Ci.begin() + Cp[j + 1]); + + this->m = m; this->n = n; + this->x.reset(pcs_aligned_new(Ci.size())); + this->p.reset(Cp); + this->i.reset(new Ti[Ci.size()]); + std::copy(Ci.begin(), Ci.end(), this->i.get()); + } + }; + + template + void multiply_general(MultiplyOutput& o, const SparseMatrix& A, const Tx2* w, const SparseMatrix& B) + { + if (!o.initialized()) + o.initialize(A, B); + + Ti m = o.m, n = o.n; + Ti* Ap = A.p.get(), * Ai = A.i.get(); Tx* Ax = A.x.get(); + Ti* Bp = B.p.get(), * Bi = B.i.get(); Tx* Bx = B.x.get(); + Ti* Cp = o.p.get(), * Ci = o.i.get(); Tx2* Cx = o.x.get(); + Tx2* c = o.c.get(); // initialized to 0 + + const Tx2 T0 = Tx2(0); + for (Ti j1 = 0; j1 < n; j1++) + { + for (Ti p1 = Bp[j1]; p1 < Bp[j1 + 1]; p1++) + { + Ti j2 = Bi[p1]; + Tx2 beta = has_weight? (Tx2(Bx[p1]) * w[j2]) : Tx2(Bx[p1]); + + for (Ti p2 = Ap[j2]; p2 < Ap[j2 + 1]; p2++) + { + //x[Ai[p2]] += beta * Ax[p2]; + fmadd(c[Ai[p2]], beta, Ax[p2]); + } + } + + for (Ti p1 = Cp[j1]; p1 < Cp[j1 + 1]; p1++) + { + Cx[p1] = c[Ci[p1]]; + c[Ci[p1]] = T0; // ensure c is 0 after the call + } + } + } + + template + void multiply(MultiplyOutput& o, const SparseMatrix& A, const SparseMatrix& B) + { + multiply_general(o, A, nullptr, B); + } + + template + void multiply(MultiplyOutput& o, const SparseMatrix& A, const Tx2* w, const SparseMatrix& B) + { + multiply_general(o, A, w, B); + } + + template + MultiplyOutput multiply(const SparseMatrix& A, const Tx2* w, const SparseMatrix& B) + { + MultiplyOutput o; + multiply(o, A, w, B); + return o; + } + + template + MultiplyOutput multiply(const SparseMatrix& A, const SparseMatrix& B) + { + MultiplyOutput o; + multiply(o, A, B); + return o; + } +} diff --git a/src/external/PackedCSparse/outerprod.h b/src/external/PackedCSparse/outerprod.h new file mode 100644 index 00000000..6f0e98e3 --- /dev/null +++ b/src/external/PackedCSparse/outerprod.h @@ -0,0 +1,86 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis +#pragma once +#include "SparseMatrix.h" + +// Problem: +// Compute x = diag(At S Bt) + +// Algorithm: +// Note that x = diag(B St A) = grad_H Tr(St A H B) +// We run autodiff on the function Tr(St A H B). +// Hence, the algorithm is essentially same as multiply(A, B) with the same runtime. + +namespace PackedCSparse { + template + struct OuterprodOutput : DenseVector + { + UniqueAlignedPtr s_col; + UniquePtr s_mark; + + template + void initialize(const SparseMatrix& A, const SparseMatrix& S, const SparseMatrix& B) + { + pcs_assert(A.initialized() && B.initialized() && S.initialized(), "outerprod: bad inputs."); + pcs_assert(A.m == S.m && S.n == B.n, "outerprod: dimensions mismatch."); + + DenseVector::initialize(A.n); + s_col.reset(pcs_aligned_new(S.m)); + s_mark.reset(new Ti[S.m]); + } + }; + + template + void outerprod(OuterprodOutput& o, const SparseMatrix& A, const SparseMatrix& S, const SparseMatrix& B) + { + if (!o.initialized()) + o.initialize(A, S, B); + + Ti Sn = S.n, Sm = S.m, An = A.n; + Ti* Ap = A.p.get(), * Ai = A.i.get(); Tx2* Ax = A.x.get(); + Ti* Bp = B.p.get(), * Bi = B.i.get(); Tx2* Bx = B.x.get(); + Ti* Sp = S.p.get(), * Si = S.i.get(); Tx* Sx = S.x.get(); + Tx* s_col = o.s_col.get(); + Ti* s_mark = o.s_mark.get(); + Tx* x = o.x.get(); + + std::fill(s_mark, s_mark + Sm, Ti(-1)); + std::fill(x, x + An, Tx(0.0)); + + for (Ti j = 0; j < Sn; j++) + { + for (Ti p = Sp[j]; p < Sp[j + 1]; p++) + { + s_col[Si[p]] = Sx[p]; + s_mark[Si[p]] = j; + } + + for (Ti p = Bp[j]; p < Bp[j + 1]; p++) + { + Ti i = Bi[p]; Tx b = Bx[p]; + for (Ti q = Ap[i]; q < Ap[i + 1]; q++) + { + Tx a = Ax[q]; Ti a_i = Ai[q]; + if (s_mark[a_i] == j) + { //x[i] += s_col[a_i] * a * b; + fmadd(x[i], s_col[a_i], a * b); + } + } + } + } + } + + template + OuterprodOutput outerprod(const SparseMatrix& A, const SparseMatrix& S, const SparseMatrix& B) + { + OuterprodOutput o; + outerprod(o, A, S, B); + return o; + } +} diff --git a/src/external/PackedCSparse/projinv.h b/src/external/PackedCSparse/projinv.h new file mode 100644 index 00000000..a319a47e --- /dev/null +++ b/src/external/PackedCSparse/projinv.h @@ -0,0 +1,90 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis + +#pragma once +#include "SparseMatrix.h" + +// Problem: +// Compute inv(L L') restricted on L + +// Algorithm: +// We need to study this later as this is the bottleneck. +// Document it as a lyx. + +namespace PackedCSparse { + template + struct ProjinvOutput : SparseMatrix + { + TransposeOutput Lt; // sparsity pattern of the Lt + UniqueAlignedPtr w; // the row of L we are computing + UniquePtr c; // c[i] = index the last nonzero on column i in the current L + + void initialize(const SparseMatrix& L) + { + pcs_assert(L.initialized(), "chol: bad inputs."); + pcs_assert(L.n == L.m, "chol: dimensions mismatch."); + + // Copy the sparsity of L + SparseMatrix::operator=(std::move(L.clone())); + + // allocate workspaces + Ti n = L.n; + w.reset(pcs_aligned_new(n)); + c.reset(new Ti[n]); + Lt = transpose(L); + } + }; + + template + void projinv(ProjinvOutput& o, const SparseMatrix& L) + { + if (!o.initialized()) + o.initialize(L); + + Tx* Sx = o.x.get(); Ti n = o.n; + Ti* Li = L.i.get(), * Lp = L.p.get(); Tx* Lv = L.x.get(); + Ti* Lti = o.Lt.i.get(), * Ltp = o.Lt.p.get(); + Tx* w = o.w.get(); + Ti* c = o.c.get(); + Tx T0 = Tx(0), T1 = Tx(1); + + for (Ti k = 0; k < n; k++) + c[k] = Lp[k + 1] - 1; + + for (Ti k = n - 1; k != -1; k--) + { + for (Ti p = Lp[k] + 1; p < Lp[k + 1]; p++) + w[Li[p]] = Sx[p]; + + Tx sum = T1 / Lv[Lp[k]]; + for (Ti p = Ltp[k + 1] - 1; p != Ltp[k] - 1; p--) + { + Ti i = Lti[p], Lpi = Lp[i]; + + for (Ti q = Lp[i + 1] - 1; q != Lpi; q--) + fnmadd(sum, Lv[q], w[Li[q]]); + //sum -= Lv[q] * w[Li[q]]; + + sum = sum / Lv[Lpi]; + w[i] = sum; + Sx[c[i]] = sum; + c[i]--; + sum = T0; + } + } + } + + template + ProjinvOutput projinv(const SparseMatrix& L) + { + ProjinvOutput o; + projinv(o, L); + return o; + } +} diff --git a/src/external/PackedCSparse/qd/COPYING b/src/external/PackedCSparse/qd/COPYING new file mode 100644 index 00000000..a20ad70e --- /dev/null +++ b/src/external/PackedCSparse/qd/COPYING @@ -0,0 +1,16 @@ +This work was supported by the Director, Office of Science, Division +of Mathematical, Information, and Computational Sciences of the +U.S. Department of Energy under contract numbers DE-AC03-76SF00098 and +DE-AC02-05CH11231. + +Copyright (c) 2003-2009, The Regents of the University of California, +through Lawrence Berkeley National Laboratory (subject to receipt of +any required approvals from U.S. Dept. of Energy) All rights reserved. + +By downloading or using this software you are agreeing to the modified +BSD license that is in file "BSD-LBNL-License.doc" in the main ARPREC +directory. If you wish to use the software for commercial purposes +please contact the Technology Transfer Department at TTD@lbl.gov or +call 510-286-6457." + + diff --git a/src/external/PackedCSparse/qd/Makefile b/src/external/PackedCSparse/qd/Makefile new file mode 100644 index 00000000..791e92b9 --- /dev/null +++ b/src/external/PackedCSparse/qd/Makefile @@ -0,0 +1,15 @@ +QD_CPPFLAGS=$(CPPFLAGS) -I$(R_INCLUDE_DIR) -march=native + +QD_SOURCES= bits.cc c_dd.cc c_qd.cc dd_const.cc dd_real.cc fpu.cc \ + qd_const.cc qd_real.cc util.cc + +QD_OBJECTS=$(QD_SOURCES:.cc=.o) + +libqd.a: $(QD_OBJECTS) + $(AR) rc libqd.a $(QD_OBJECTS) + +.cc.o: + $(CC) $(CFLAGS) $(CPICFLAGS) $(QD_CPPFLAGS) -c $< -o $@ + +clean: + rm -rf $(QD_OBJECTS) libqd.a diff --git a/src/external/PackedCSparse/qd/NEWS b/src/external/PackedCSparse/qd/NEWS new file mode 100644 index 00000000..f32a7575 --- /dev/null +++ b/src/external/PackedCSparse/qd/NEWS @@ -0,0 +1,181 @@ +Changes for 2.3.22 + Made changes suggested by Vasiliy Sotnikov + +Changes for 2.3.21 + Changed renorm in include/qd/qd_inline.h + +Changes for 2.3.20 + added #include to quadt_test.cpp + changed references to 2.3.20 from 2.3.18 + +Changes for 2.3.19 + - Updated qd_real.cpp and dd_real.cpp to fix a buffer overflow problem. + +Changes for 2.3.18 + - Updated qd_real.cpp and dd_real.cpp to fix a problem in output. + +Changes for 2.3.17 + - updated qd_real.cpp, to fix a problem with improper treatment of + negative arguments in nroot. + +Changes for 2.3.16 + - Updated dd_real.cpp, to fix a problem with inaccurate values of + tanh for small arguments. + +Changes for 2.3.15 + - Updated qd_real.cpp, to fix a problem with static definitions. + +Changes for 2.3.14 + - Updated autoconfig (replaced config.sub and config.guess) + +Changes for 2.3.7 + - Fixed bug in to_digits where digits larger than 10 + where output occasionally. + +Changes for 2.3.6 + - Added fmod (C++) and mod (Fortran) functions. + +Changes for 2.3.5 + - Fixed bug in division of qd_real by dd_real. + - Fixed bug in ddoutc (Fortran ddmod.f). + - Now compiles with g++ 4.3. + - Distribute tests/coeff.dat. + +Changes for 2.3.4 + - Fixed bug in Makefile for cygwin / mingw systems. + +Changes for 2.3.3 + - Fixed bug in atan2. + +Changes for 2.3.2 + - Fixed bug in sin / cos / sincos where too much accuracy was + lost for (moderately) large angles. + - Use fused-multiply add intrinsics on IA-64 platforms if + compiled by Intel compiler. + - Fixed bug in c_dd_write and c_qd_write. + - Fixed bug were qdext.mod was not being installed. + +Changes for 2.3.1 + - Fixed bug in sincos and cos_taylor. This affected the result + of trigonometric functions in some cases. + +Changes for 2.3.0 + This is a fairly significant change, breaking API compatibility. + - Moved C++ main entry in libqdmod.a to libqd_f_main.a. + This allows to link Fortran code using QD with custom + C++ main function. Pure Fortran code will need to be linked + with qd_f_main library in addition to qdmod and qd library. + - Constructors accepting pointers made explicit. + - Fortran routines labeled as elemental or pure, where appropriate. + - Write() is now to_string(), and now takes a single fmtflag. + - dd_real addition and multiplication made commutative. + - dd_real now represented as array of two doubles, instead of + two discrete scalars. + - New Fortran generic routines to read / write, operations with + complex and integers. + - Improved exp, sin, and cos functions. + - Removed unused constants and obscure constants only used internally + from public interface. + +Changes for 2.2.6 + - Fixed bug in mixed precision multiplication: qd_real * dd_real. + +Changes for 2.2.5 + - Bug fix in qd_real addition when --enable-ieee-add is specified. + - Debugging routines dump and dump_bits updated; + dump_components removed (just use dump). + - Fortran support for Fortran strings. Use character arrays instead. + - Return NaN under error conditions. + - Added _inf constant; exp now returns Inf when argument is too large. + - Output formatting fixes for Inf and NaNs. + - Added more real-complex mixed arithmetic routines in Fortran + interface. + +Changes for 2.2.4 + - Added random_number interface for Fortran modules. + - Use slightly more conservative values for eps. + - Avoid unnecessary overflow near overflow threshold. + - Added radix, digits, min/maxexponent, range, and precision + intrinsics to Fortran interface. + - Added safe_max (C++) and safe_huge (Fortran). + +Changes for 2.2.3 + - Fix sign function bug in Fortran modules. + +Changes for 2.2.2 + - Do not bother setting uninitialized dd_real and qd_reals to zero. + - Use clock_gettime if available for timing. + - Fortran I/O should be more consistent with C++ version. + - fpu.h is now included with dd_real.h. + +Changes for 2.2.1 + - Minor fixes when printing in scientific format. + - Change search order of C++ compilers in Apple systems to avoid + case insensitive filesystems. + +Changes for 2.2.0 + - Added F95 interface for complex types. + - Renamed dd.h and qd.h to dd_real.h and qd_real.h, respectively. + This will break older C++ code using 2.1.x library, but it was + conflicting with QuickDraw libraries on Macs. (Hence the version + bump to 2.2). + - Removed overloaded typecast operators for int and double. These + permitted *automatic* conversion of dd_real/qd_real to double or + int, which is somewhat dangerous. Instead to_int and to_double + routines are added. + +Changes for 2.1.214 + - Updated pslq_test. + - Implmented numeric_limits<>. + - Better polyroot. + - Added isnan, isfinite, isinf functions. + - Fix / improve input output functions. + - Drop Microsoft Visual C++ 6.0 support. + - More efficient dd_real::sin. + +Changes for 2.1.213 + - Support for x86_64 platforms. + - Drop libtool support for now. + +Changes for 2.1.212 + - Support for pathCC compiler. + - Added accurate and sloppy versions of add / sub / mul / div avaialble. + - Added autodetection of fma functions. + +Changes for 2.1 (2003-12-30) + - added automake scripts. + - use libtool to compile / link and build libraries. + - supports standard installation targets (make install). + - support for Intel C++ compilers (icc / ecc). + - Fortran programs are now linked by C++ compiler. + - support for building shared library. + - minor bug fixes. + +Changes for 2.0 (2003-12-08) + - all header files are in "include/qd" directory. + - added autoconf scripts. + - added config.h and qd_config.h to store configuration information. + - renamed x86_* routines to fpu_* routines. + - added separate Fortran interface (f_* routines). + - options for sloppy multiply and sloppy divison separated. + - fixed C interface to be actually in C syntax. + - updated / added README, AUTHORS, NEWS, and LICENSE files. + - minor bug fixes. + +Changes for 1.2 (2003-12-04) + - added "dist-clean" target in Makefile + - initialize dd and qd variables to zero + - increases tolerance for qd / dd tests + - changed .cc extension to .cpp + - updated README, COPYING, and NEWS files + - added ChangeLog file + - fixed bug in '-all' flag in qd_test + - minor bug fixes + +Changes for 1.1 (2002-10-22) + - added "Changes" file (this file) + - fixed to + - fixed constant (3/4) * pi + - fixed exp(x) to return zero if x is a large negative number + - removed "docs" target in Makefile + diff --git a/src/external/PackedCSparse/qd/README b/src/external/PackedCSparse/qd/README new file mode 100644 index 00000000..8a085d72 --- /dev/null +++ b/src/external/PackedCSparse/qd/README @@ -0,0 +1,437 @@ +Quad Double computation package +Copyright (C) 2003-2019 +================================================ + +Revision date: 26 February 2019 + +Authors: +Yozo Hida U.C. Berkeley yozo@cs.berkeley.edu +Xiaoye S. Li Lawrence Berkeley Natl Lab xiaoye@nersc.gov +David H. Bailey Lawrence Berkeley Natl Lab dhbailey@lbl.gov + +C++ usage guide: +Alex Kaiser Lawrence Berkeley Natl Lab adkaiser@lbl.gov + +This work was supported by the Director, Office of Science, Division of Mathematical, +Information, and Computational Sciences of the U.S. Department of Energy under contract +number DE-AC02-05CH11231. + +This work was supported by the Director, Office of Science, Division of Mathematical, +Information, and Computational Sciences of the U.S. Department of Energy under contract +numbers DE-AC03-76SF00098 and DE-AC02-05CH11231. + +*** IMPORTANT NOTES: + +See the file COPYING for modified BSD license information. +See the file INSTALL for installation instructions. +See the file NEWS for recent revisions. +See the file docs/qd.pdf for additional information. + +Outline: + +I. Introduction +II. Installation of package, and linking and executing user files +III. C++ Usage +IV. Fortran Usage +V. Note on x86-Based Processors (MOST systems in use today) + + +I. Introduction + +This package provides numeric types of twice the precision of IEEE double (106 mantissa +bits, or approximately 32 decimal digits) and four times the precision of IEEE double (212 +mantissa bits, or approximately 64 decimal digits). Due to features such as operator and +function overloading, these facilities can be utilized with only minor modifications to +conventional C++ and Fortran-90 programs. + +In addition to the basic arithmetic operations (add, subtract, multiply, divide, square root), +common transcendental functions such as the exponential, logarithm, trigonometric and +hyperbolic functions are also included. A detailed description of the algorithms used is +available in the docs subdirectory (see docs/qd.pdf). An abridged version of this paper, +which was presented at the ARITH-15 conference, is also available at: + +Yozo Hida, Xiaoye S. Li and David H. Bailey, "Algorithms for quad-double precision + floating point arithmetic," 15th IEEE Symposium on Computer Arithmetic, IEEE Computer + Society, 2001, pg. 155-162, available at + https://www.davidhbailey.com/dhbpapers/arith15.pdf. + + +II. Installation of package, and linking and executing user files + +A. Directories + +There are six directories and several files in the main directory of this distribution, +described below + +src This contains the source code of the quad-double and double-double + library. This source code does not include inline functions, + which are found in the header files in the include directory. + +include This directory contains the header files. + +fortran This directory contains Fortran-90 files. + +tests This directory contains some simple (not comprehensive) tests. + +docs This directory contains a technical paper describing the algorithms. + +config This directory contains various scripts used by the configure + script and the Makefile. + +Please note that all commands refer to a Unix-type environment such as Mac OSX or Ubuntu +Linux using the bash shell. + +B. Installing and building + +To build the library, first run the included configure script by typing + + ./configure + +This script automatically generates makefiles for building the library and selects compilers +and necessary flags and libraries to include. If the user wishes to specify compilers or flags +they may use the following options. + + CXX C++ compiler to use + CXXFLAGS C++ compiler flags to use + CC C compiler to use (for C demo program) + CFLAGS C compiler flags to use (for C demo program) + FC Fortran 90 compiler + FCFLAGS Fortran 90 compiler flags to use + FCLIBS Fortran 90 libraries needed to link with C++ code. + +For example, if one is using GNU compilers, configure with: + + ./configure CXX=g++ FC=gfortran + +The Fortran and C++ compilers must produce compatible binaries. On some systems +additional flags must be included to ensure that portions of the +library are not built with 32 and 64 bit object files. For example, on +64-Bit Mac OSX 10.6 (Snow Leopard) and 10.7 (Lion) the correct +configure line using GNU compilers is: + + ./configure CXX=g++ FC=gfortran FCFLAGS=-m64 + +To build the library, simply type + + make + +and the automatically generated makefiles will build the library including archive files. + +To allow for easy linking to the library, the user may also wish to +install the archive files to a standard place. To do this type: + + make install + +This will also build the library if it has not already been built. Many systems, including Mac +and Ubuntu Linux systems, require administrator privileges to install the library at such +standard places. On such systems, one may type: + + sudo make install + +instead if one has sufficient access. + +The directory "tests" contains programs for high precision quadrature and integer-relation +detection. To build such programs, type: + + make demo + +in the "tests" directory. + +C. Linking and executing user programs + +C++ source files: + +The simplest way to link to the library is to install it to a standard place as described above, and use the -l option. For example + + g++ compileExample.cpp -o compileExample -l qd + +One can also use this method to build with make. A file called "compileExample.cpp" and the +associated makefile "makeCompileExample" illustrate the process. + +A third alternative is to use a link script. If one types "make demo" in the test +directory, the output produced gives guidance as to how to build the files. By +following the structure of the compiling commands one may copy the appropriate portions, +perhaps replacing the filename with an argument that the user can include at link time. +An example of such a script is as follows: + +g++ -DHAVE_CONFIG_H -I.. -I../include -I../include -O2 -MT $1.o -MD -MP -MF +.deps/qd_test.Tpo -c -o $1.o $1.cpp +mv -f .deps/$1.Tpo .deps/$1.Po +g++ -O2 -o $1 $1.o ../src/libqd.a -lm + +To use the link script, make it executable (by typing "chmod +x link.scr) and then type: + +./link.scr compileExample + +Note that the file extension is not included because the script handles all extensions, +expecting the source file to have the extension ".cpp". + +Fortran-90 source files: + +Similarly, a script for compiling fortran programs may be constructed as follows. +In the fortran directory, type "make quadtsq". This compiles the Fortran program +tquadts.f, links with all necessary library files, and produces the executable +"quadts". As this is being done, all flags and linked libraries are displayed. +For instance, on a 2019-era Apple Macintosh system, where the library was installed +as above with g++ for C++ and gfortran for Fortran-90, the following is output: + +gfortran -m64 -ffree-form -c -o tquadtsq.o tquadtsq.f +/bin/sh ../libtool --tag=CXX --mode=link g++ -O2 -o quadtsq tquadtsq.o second.o +libqdmod.la libqd_f_main.la ../src/libqd.la +-L/usr/local/gfortran/lib/gcc/x86_64-apple-darwin16/6.3.0 +-L/usr/local/gfortran/lib/gcc/x86_64-apple-darwin16/6.3.0/../../.. +-lgfortran -lquadmath -lm -lm + +Thus a general compile-link script is the following: + +gfortran -m64 -ffree-form -c -o $1.o $1.f90 +/bin/sh ../libtool --tag=CXX --mode=link g++ -O2 -o $1 $1.o second.o \ + libqdmod.la libqd_f_main.la ../src/libqd.la \ + -L/usr/local/gfortran/lib/gcc/x86_64-apple-darwin16/6.3.0 \ + -L/usr/local/gfortran/lib/gcc/x86_64-apple-darwin16/6.3.0/../../.. \ + -lgfortran -lquadmath -lm -lm + +Note that if the .f90 suffix is used for Fortran-90 source files, the +-ffree-form flag may be omitted, but the first line above should end with +"$1.f90" (as shown above). After forming the script, name file, "complink.scr", +and then type "chmod +x complink.scr". To use this script compile and link a +program named "prog.f90", type "./complink.scr prog". + + +III. C++ usage + +As much as possible, operator overloading is included to make basic programming as much +like using standard typed floating-point arithmetic. Changing many codes should be as +simple as changing type statements and a few other lines. + +i. Constructors + +To create dd_real and qd_real variables calculated to the proper precision, one must use +care to use the included constructors properly. Many computations in which variables are +not explicitly typed to multiple-precision may be evaluated with double-precision +arithmetic. The user must take care to ensure that this does not cause errors. In particular, +an expression such as 1.0/3.0 will be evaluated to double precision before assignment or +further arithmetic. Upon assignment to a multi-precision variable, the value will be zero +padded. This problem is serious and potentially difficult to debug. To avoid this, use the +included constructors to force arithmetic to be performed in the full precision requested. + +For a table with descriptions, please see the documentation file qd.pdf in the docs directory. + +ii. Included functions and Constants + +Supported functions include assignment operators, comparisons, arithmetic and +assignment operators, and increments for integer types. Standard C math functions such as +exponentiation, trigonometric, logarithmic, hyperbolic, exponential and rounding functions +are included. As in assignment statements, one must be careful with implied typing of +constants when using these functions. Many codes need particular conversion for the power +function, which is frequently used with constants that must be explicitly typed for multi- +precision codes. + +Many constants are included, which are global and calculated upon initialization. The +following list of constants is calculated for both the dd_real and qd_real classes separately. +Use care to select the correct value. + +For a table with descriptions, please see the included file README.pdf + +ii. Conversion of types + +Static casts may be used to convert constants between types. One may also use constructors +to return temporary multi-precision types within expressions, but should be careful, as this +will waste memory if done repeatedly. For example: + + qd_real y ; + y = sin( qd_real(4.0) / 3.0 ) ; + +C-style casts may be used, but are not recommended. Dynamic and reinterpret casts are +not supported and should be considered unreliable. Casting between multi-precision and +standard precision types can be dangerous, and care must be taken to ensure that programs +are working properly and accuracy has not degraded by use of a misplaced type-conversion. + +D. Available precision, Control of Precision Levels, + +The library provides greatly extended accuracy when compared to standard double +precision. The type dd_real provides for 106 mantissa bits, or about 32 decimal digits. The +type qd_real provides for 212 mantissa bits, or about 64 decimal digits. + +Both the dd_real and qd_real values use the exponent from the highest double-precision +word for arithmetic, and as such do not extend the total range of values available. That +means that the maximum absolute value for either data type is the same as that of double- +precision, or approximately 10^308. The precision near this range, however, is greatly +increased. + +E. I/O + +The standard I/O stream routines have been overloaded to be fully compatible with all +included data types. One may need to manually reset the precision of the stream to obtain +full output. For example, if 60 digits are desired, use: + +cout.precision(60) ; + +When reading values using cin, each input numerical value must start on a separate +line. Two formats are acceptable: + + 1. Write the full constant + 3. Mantissa e exponent + +Here are three valid examples: + + 1.1 + 3.14159 26535 89793 + 123.123123e50 + +When read using cin, these constants will be converted using full multi-precision accuracy. + + +IV. Fortran-90 Usage + +NEW (2007-01-10): The Fortran translation modules now support the complex datatypes +"dd_complex" and "qd_complex". + +Since the quad-double library is written in C++, it must be linked in with a C++ compiler (so +that C++ specific things such as static initializations are correctly handled). Thus the main +program must be written in C/C++ and call the Fortran 90 subroutine. The Fortran 90 +subroutine should be called f_main. + +Here is a sample Fortran-90 program, equivalent to the above C++ program: + + subroutine f_main + use qdmodule + implicit none + type (qd_real) a, b + a = 1.d0 + b = cos(a)**2 + sin(a)**2 - 1.d0 + call qdwrite(6, b) + stop + end subroutine + +This verifies that cos^2(1) + sin^2(1) = 1 to 64 digit accuracy. + +Most operators and generic function references, including many mixed-mode type +combinations with double-precision (ie real*8), have been overloaded (extended) to work +with double-double and quad-double data. It is important, however, that users keep in +mind the fact that expressions are evaluated strictly according to conventional Fortran +operator precedence rules. Thus some subexpressions may be evaluated only to 15-digit +accuracy. For example, with the code + + real*8 d1 + type (dd_real) t1, t2 + ... + t1 = cos (t2) + d1/3.d0 + +the expression d1/3.d0 is computed to real*8 accuracy only (about 15 digits), since both d1 +and 3.d0 have type real*8. This result is then converted to dd_real by zero extension before +being added to cos(t2). So, for example, if d1 held the value 1.d0, then the quotient d1/3.d0 +would only be accurate to 15 digits. If a fully accurate double-double quotient is required, +this should be written: + + real*8 d1 + type (dd_real) t1, t2 + ... + t1 = cos (t2) + ddreal (d1) / 3.d0 + +which forces all operations to be performed with double-double arithmetic. + +Along this line, a constant such as 1.1 appearing in an expression is evaluated only to real*4 +accuracy, and a constant such as 1.1d0 is evaluated only to real*8 accuracy (this is +according to standard Fortran conventions). If full quad-double accuracy is required, for +instance, one should write + + type (qd_real) t1 + ... + t1 = '1.1' + +The quotes enclosing 1.1 specify to the compiler that the constant is to be converted to +binary using quad-double arithmetic, before assignment to t1. Quoted constants may only +appear in assignment statements such as this. + +To link a Fortran-90 program with the C++ qd library, it is recommended to link with the +C++ compiler used to generate the library. The Fortran 90 interface (along with a C-style +main function calling f_main) is found in qdmod library. The qd-config script installed +during "make install" can be used to determine which flags to pass to compile and link your +programs: + + "qd-config --fcflags" displays compiler flags needed to compile your Fortran files. + "qd-config --fclibs" displays linker flags needed by the C++ linker to link in all the +necessary libraries. + +A sample Makefile that can be used as a template for compiling Fortran programs using +quad-double library is found in fortran/Makefile.sample. + +F90 functions defined with dd_real arguments: + Arithmetic: + - * / ** + Comparison tests: == < > <= >= /= + Others: abs, acos, aint, anint, asin, atan, atan2, cos, cosh, dble, erf, + erfc, exp, int, log, log10, max, min, mod, ddcsshf (cosh and sinh), + ddcssnf (cos and sin), ddranf (random number generator in (0,1)), + ddnrtf (n-th root), sign, sin, sinh, sqr, sqrt, tan, tanh + +Similar functions are provided for qd_real arguments with function names qdcsshf, +qdcssnf, qdranf and qdnrtf instead of the names in the list above. + +Input and output of double-double and quad-double data is done using the special +subroutines ddread, ddwrite, qdread and qdwrite. The first argument of these subroutines +is the Fortran I/O unit number, while additional arguments (as many as needed, up to 9 +arguments) are scalar variables or array elements of the appropriate type. Example: + + integer n + type (qd_real) qda, qdb, qdc(n) + ... + call qdwrite (6, qda, qdb) + do j = 1, n + call qdwrite (6, qdc(j)) + enddo + +Each input values must be on a separate line, and may include D or E exponents. Double- +double and quad-double constants may also be specified in assignment statements by +enclosing them in quotes, as in + + ... + type (qd_real) pi + ... + pi = +"3.14159265358979323846264338327950288419716939937510582097494459230" + ... + +Sample Fortran-90 programs illustrating some of these features are provided in the f90 +subdirectory. + + +V. Note on x86-Based Processors (MOST systems in use today) + +The algorithms in this library assume IEEE double precision floating point arithmetic. Since +Intel x86 processors have extended (80-bit) floating point registers, some compilers, +albeit a declining number, may generate commands for the 80-bit instructions. The QD +library does NOT work correctly with 80-bit instructions, so if one's code does not operate +correctly, this may be the reason. To avoid such problems, the round-to-double flag must be +enabled in the control word of the FPU for this library to function properly. The following +functions contains appropriate code to facilitate manipulation of this flag. For non-x86 +systems these functions do nothing (but still exist). + +fpu_fix_start This turns on the round-to-double bit in the control word. +fpu_fix_end This restores the control flag. + +These functions must be called by the main program, as follows: + + int main() { + unsigned int old_cw; + fpu_fix_start(&old_cw); + + ... user code using quad-double library ... + + fpu_fix_end(&old_cw); + } + +A Fortran-90 example is the following: + + subroutine f_main + use qdmodule + implicit none + integer*4 old_cw + + call f_fpu_fix_start(old_cw) + + ... user code using quad-double library ... + + call f_fpu_fix_end(old_cw) + end subroutine + diff --git a/src/external/PackedCSparse/qd/bits.cc b/src/external/PackedCSparse/qd/bits.cc new file mode 100644 index 00000000..4eaf9a26 --- /dev/null +++ b/src/external/PackedCSparse/qd/bits.cc @@ -0,0 +1,85 @@ +/* + * src/bits.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Defines various routines to get / set bits of a IEEE floating point + * number. This used by the library for debugging purposes. + */ + +#include +#include +#include +#include + +#include "qd_config.h" +#include "inline.h" +#include "bits.h" + +#ifdef HAVE_IEEEFP_H +#include +#endif + +using std::setw; + +int get_double_expn(double x) { + if (x == 0.0) + return INT_MIN; + if (QD_ISINF(x) || QD_ISNAN(x)) + return INT_MAX; + + double y = std::abs(x); + int i = 0; + if (y < 1.0) { + while (y < 1.0) { + y *= 2.0; + i++; + } + return -i; + } else if (y >= 2.0) { + while (y >= 2.0) { + y *= 0.5; + i++; + } + return i; + } + return 0; +} + +void print_double_info(std::ostream &os, double x) { + std::streamsize old_prec = os.precision(19); + std::ios_base::fmtflags old_flags = os.flags(); + os << std::scientific; + + os << setw(27) << x << ' '; + if (QD_ISNAN(x) || QD_ISINF(x) || (x == 0.0)) { + os << " "; + } else { + + x = std::abs(x); + int expn = get_double_expn(x); + double d = std::ldexp(1.0, expn); + os << setw(5) << expn << " "; + for (int i = 0; i < 53; i++) { + if (x >= d) { + x -= d; + os << '1'; + } else + os << '0'; + d *= 0.5; + } + + if (x != 0.0) { + // should not happen + os << " +trailing stuff"; + } + } + + os.precision(old_prec); + os.flags(old_flags); +} + diff --git a/src/external/PackedCSparse/qd/bits.h b/src/external/PackedCSparse/qd/bits.h new file mode 100644 index 00000000..58570aac --- /dev/null +++ b/src/external/PackedCSparse/qd/bits.h @@ -0,0 +1,32 @@ +/* + * include/bits.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * This file defines various routines to get / set bits of a IEEE floating + * point number. This is used by the library for debugging purposes. + */ + +#ifndef _QD_BITS_H +#define _QD_BITS_H + +#include +#include "qd_config.h" + +/* Returns the exponent of the double precision number. + Returns INT_MIN is x is zero, and INT_MAX if x is INF or NaN. */ +int get_double_expn(double x); + +/* Prints + SIGN EXPN MANTISSA + of the given double. If x is NaN, INF, or Zero, this + prints out the strings NaN, +/- INF, and 0. */ +void print_double_info(std::ostream &os, double x); + + +#endif /* _QD_BITS_H */ + diff --git a/src/external/PackedCSparse/qd/c_dd.cc b/src/external/PackedCSparse/qd/c_dd.cc new file mode 100644 index 00000000..0a7c12ac --- /dev/null +++ b/src/external/PackedCSparse/qd/c_dd.cc @@ -0,0 +1,314 @@ +/* + * src/c_dd.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains the C wrapper functions for double-double precision arithmetic. + * This can be used from Fortran code. + */ +#include + +#include "qd_config.h" +#include "dd_real.h" +#include "c_dd.h" + +#define TO_DOUBLE_PTR(a, ptr) ptr[0] = a.x[0]; ptr[1] = a.x[1]; + +extern "C" { + +/* add */ +void c_dd_add(const double *a, const double *b, double *c) { + dd_real cc; + cc = dd_real(a) + dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_dd_add_dd_d(const double *a, double b, double *c) { + dd_real cc; + cc = dd_real(a) + b; + TO_DOUBLE_PTR(cc, c); +} +void c_dd_add_d_dd(double a, const double *b, double *c) { + dd_real cc; + cc = a + dd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + +/* sub */ +void c_dd_sub(const double *a, const double *b, double *c) { + dd_real cc; + cc = dd_real(a) - dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_dd_sub_dd_d(const double *a, double b, double *c) { + dd_real cc; + cc = dd_real(a) - b; + TO_DOUBLE_PTR(cc, c); +} +void c_dd_sub_d_dd(double a, const double *b, double *c) { + dd_real cc; + cc = a - dd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + +/* mul */ +void c_dd_mul(const double *a, const double *b, double *c) { + dd_real cc; + cc = dd_real(a) * dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_dd_mul_dd_d(const double *a, double b, double *c) { + dd_real cc; + cc = dd_real(a) * b; + TO_DOUBLE_PTR(cc, c); +} +void c_dd_mul_d_dd(double a, const double *b, double *c) { + dd_real cc; + cc = a * dd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + +/* div */ +void c_dd_div(const double *a, const double *b, double *c) { + dd_real cc; + cc = dd_real(a) / dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_dd_div_dd_d(const double *a, double b, double *c) { + dd_real cc; + cc = dd_real(a) / b; + TO_DOUBLE_PTR(cc, c); +} +void c_dd_div_d_dd(double a, const double *b, double *c) { + dd_real cc; + cc = a / dd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + +/* copy */ +void c_dd_copy(const double *a, double *b) { + b[0] = a[0]; + b[1] = a[1]; +} +void c_dd_copy_d(double a, double *b) { + b[0] = a; + b[1] = 0.0; +} + + +void c_dd_sqrt(const double *a, double *b) { + dd_real bb; + bb = sqrt(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_sqr(const double *a, double *b) { + dd_real bb; + bb = sqr(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_abs(const double *a, double *b) { + dd_real bb; + bb = abs(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_npwr(const double *a, int n, double *b) { + dd_real bb; + bb = npwr(dd_real(a), n); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_nroot(const double *a, int n, double *b) { + dd_real bb; + bb = nroot(dd_real(a), n); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_nint(const double *a, double *b) { + dd_real bb; + bb = nint(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_aint(const double *a, double *b) { + dd_real bb; + bb = aint(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_floor(const double *a, double *b) { + dd_real bb; + bb = floor(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_ceil(const double *a, double *b) { + dd_real bb; + bb = ceil(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_log(const double *a, double *b) { + dd_real bb; + bb = log(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_log10(const double *a, double *b) { + dd_real bb; + bb = log10(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_exp(const double *a, double *b) { + dd_real bb; + bb = exp(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_sin(const double *a, double *b) { + dd_real bb; + bb = sin(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_cos(const double *a, double *b) { + dd_real bb; + bb = cos(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_tan(const double *a, double *b) { + dd_real bb; + bb = tan(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_asin(const double *a, double *b) { + dd_real bb; + bb = asin(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_acos(const double *a, double *b) { + dd_real bb; + bb = acos(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_atan(const double *a, double *b) { + dd_real bb; + bb = atan(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_atan2(const double *a, const double *b, double *c) { + dd_real cc; + cc = atan2(dd_real(a), dd_real(b)); + TO_DOUBLE_PTR(cc, c); +} + +void c_dd_sinh(const double *a, double *b) { + dd_real bb; + bb = sinh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_cosh(const double *a, double *b) { + dd_real bb; + bb = cosh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_tanh(const double *a, double *b) { + dd_real bb; + bb = tanh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_asinh(const double *a, double *b) { + dd_real bb; + bb = asinh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_acosh(const double *a, double *b) { + dd_real bb; + bb = acosh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_dd_atanh(const double *a, double *b) { + dd_real bb; + bb = atanh(dd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_dd_sincos(const double *a, double *s, double *c) { + dd_real ss, cc; + sincos(dd_real(a), ss, cc); + TO_DOUBLE_PTR(ss, s); + TO_DOUBLE_PTR(cc, c); +} + +void c_dd_sincosh(const double *a, double *s, double *c) { + dd_real ss, cc; + sincosh(dd_real(a), ss, cc); + TO_DOUBLE_PTR(ss, s); + TO_DOUBLE_PTR(cc, c); +} + +void c_dd_read(const char *s, double *a) { + dd_real aa(s); + TO_DOUBLE_PTR(aa, a); +} + +void c_dd_swrite(const double *a, int precision, char *s, int len) { + dd_real(a).write(s, len, precision); +} + +void c_dd_write(const double *a) { + std::cout << dd_real(a).to_string(dd_real::_ndigits) << std::endl; +} + +void c_dd_neg(const double *a, double *b) { + b[0] = -a[0]; + b[1] = -a[1]; +} + +void c_dd_rand(double *a) { + dd_real aa; + aa = ddrand(); + TO_DOUBLE_PTR(aa, a); +} + +void c_dd_comp(const double *a, const double *b, int *result) { + dd_real aa(a), bb(b); + if (aa < bb) + *result = -1; + else if (aa > bb) + *result = 1; + else + *result = 0; +} + +void c_dd_comp_dd_d(const double *a, double b, int *result) { + dd_real aa(a), bb(b); + if (aa < bb) + *result = -1; + else if (aa > bb) + *result = 1; + else + *result = 0; +} + +void c_dd_comp_d_dd(double a, const double *b, int *result) { + dd_real aa(a), bb(b); + if (aa < bb) + *result = -1; + else if (aa > bb) + *result = 1; + else + *result = 0; +} + +void c_dd_pi(double *a) { + TO_DOUBLE_PTR(dd_real::_pi, a); +} + +} diff --git a/src/external/PackedCSparse/qd/c_dd.h b/src/external/PackedCSparse/qd/c_dd.h new file mode 100644 index 00000000..310162ed --- /dev/null +++ b/src/external/PackedCSparse/qd/c_dd.h @@ -0,0 +1,98 @@ +/* + * include/c_dd.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains C wrapper function prototypes for double-double precision + * arithmetic. This can also be used from fortran code. + */ +#ifndef _QD_C_DD_H +#define _QD_C_DD_H + +#include "qd_config.h" +#include "fpu.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/* add */ +void c_dd_add(const double *a, const double *b, double *c); +void c_dd_add_d_dd(double a, const double *b, double *c); +void c_dd_add_dd_d(const double *a, double b, double *c); + +/* sub */ +void c_dd_sub(const double *a, const double *b, double *c); +void c_dd_sub_d_dd(double a, const double *b, double *c); +void c_dd_sub_dd_d(const double *a, double b, double *c); + +/* mul */ +void c_dd_mul(const double *a, const double *b, double *c); +void c_dd_mul_d_dd(double a, const double *b, double *c); +void c_dd_mul_dd_d(const double *a, double b, double *c); + +/* div */ +void c_dd_div(const double *a, const double *b, double *c); +void c_dd_div_d_dd(double a, const double *b, double *c); +void c_dd_div_dd_d(const double *a, double b, double *c); + +/* copy */ +void c_dd_copy(const double *a, double *b); +void c_dd_copy_d(double a, double *b); + +void c_dd_sqrt(const double *a, double *b); +void c_dd_sqr(const double *a, double *b); + +void c_dd_abs(const double *a, double *b); + +void c_dd_npwr(const double *a, int b, double *c); +void c_dd_nroot(const double *a, int b, double *c); + +void c_dd_nint(const double *a, double *b); +void c_dd_aint(const double *a, double *b); +void c_dd_floor(const double *a, double *b); +void c_dd_ceil(const double *a, double *b); + +void c_dd_exp(const double *a, double *b); +void c_dd_log(const double *a, double *b); +void c_dd_log10(const double *a, double *b); + +void c_dd_sin(const double *a, double *b); +void c_dd_cos(const double *a, double *b); +void c_dd_tan(const double *a, double *b); + +void c_dd_asin(const double *a, double *b); +void c_dd_acos(const double *a, double *b); +void c_dd_atan(const double *a, double *b); +void c_dd_atan2(const double *a, const double *b, double *c); + +void c_dd_sinh(const double *a, double *b); +void c_dd_cosh(const double *a, double *b); +void c_dd_tanh(const double *a, double *b); + +void c_dd_asinh(const double *a, double *b); +void c_dd_acosh(const double *a, double *b); +void c_dd_atanh(const double *a, double *b); + +void c_dd_sincos(const double *a, double *s, double *c); +void c_dd_sincosh(const double *a, double *s, double *c); + +void c_dd_read(const char *s, double *a); +void c_dd_swrite(const double *a, int precision, char *s, int len); +void c_dd_write(const double *a); +void c_dd_neg(const double *a, double *b); +void c_dd_rand(double *a); +void c_dd_comp(const double *a, const double *b, int *result); +void c_dd_comp_dd_d(const double *a, double b, int *result); +void c_dd_comp_d_dd(double a, const double *b, int *result); +void c_dd_pi(double *a); + +#ifdef __cplusplus +} +#endif + +#endif /* _QD_C_DD_H */ diff --git a/src/external/PackedCSparse/qd/c_qd.cc b/src/external/PackedCSparse/qd/c_qd.cc new file mode 100644 index 00000000..fd50e922 --- /dev/null +++ b/src/external/PackedCSparse/qd/c_qd.cc @@ -0,0 +1,450 @@ +/* + * src/c_qd.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains C wrapper function for quad-double precision arithmetic. + * This can be used from fortran code. + */ +#include + +#include "qd_config.h" +#include "qd_real.h" +#include "c_qd.h" + +#define TO_DOUBLE_PTR(a, ptr) ptr[0] = a.x[0]; ptr[1] = a.x[1]; \ + ptr[2] = a.x[2]; ptr[3] = a.x[3]; + +extern "C" { + + + +/* add */ +void c_qd_add(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) + qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_add_qd_dd(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) + dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_add_dd_qd(const double *a, const double *b, double *c) { + qd_real cc; + cc = dd_real(a) + qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_add_qd_d(const double *a, double b, double *c) { + qd_real cc; + cc = qd_real(a) + b; + TO_DOUBLE_PTR(cc, c); +} +void c_qd_add_d_qd(double a, const double *b, double *c) { + qd_real cc; + cc = a + qd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + + +/* sub */ +void c_qd_sub(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) - qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_sub_qd_dd(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) - dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_sub_dd_qd(const double *a, const double *b, double *c) { + qd_real cc; + cc = dd_real(a) - qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_sub_qd_d(const double *a, double b, double *c) { + qd_real cc; + cc = qd_real(a) - b; + TO_DOUBLE_PTR(cc, c); +} +void c_qd_sub_d_qd(double a, const double *b, double *c) { + qd_real cc; + cc = a - qd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + + +/* mul */ +void c_qd_mul(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) * qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_mul_qd_dd(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) * dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_mul_dd_qd(const double *a, const double *b, double *c) { + qd_real cc; + cc = dd_real(a) * qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_mul_qd_d(const double *a, double b, double *c) { + qd_real cc; + cc = qd_real(a) * b; + TO_DOUBLE_PTR(cc, c); +} +void c_qd_mul_d_qd(double a, const double *b, double *c) { + qd_real cc; + cc = a * qd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + + +/* div */ +void c_qd_div(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) / qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_div_qd_dd(const double *a, const double *b, double *c) { + qd_real cc; + cc = qd_real(a) / dd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_div_dd_qd(const double *a, const double *b, double *c) { + qd_real cc; + cc = dd_real(a) / qd_real(b); + TO_DOUBLE_PTR(cc, c); +} +void c_qd_div_qd_d(const double *a, double b, double *c) { + qd_real cc; + cc = qd_real(a) / b; + TO_DOUBLE_PTR(cc, c); +} +void c_qd_div_d_qd(double a, const double *b, double *c) { + qd_real cc; + cc = a / qd_real(b); + TO_DOUBLE_PTR(cc, c); +} + + + + +/* selfadd */ +void c_qd_selfadd(const double *a, double *b) { + qd_real bb(b); + bb += qd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfadd_dd(const double *a, double *b) { + qd_real bb(b); + bb += dd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfadd_d(double a, double *b) { + qd_real bb(b); + bb += a; + TO_DOUBLE_PTR(bb, b); +} + + + +/* selfsub */ +void c_qd_selfsub(const double *a, double *b) { + qd_real bb(b); + bb -= qd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfsub_dd(const double *a, double *b) { + qd_real bb(b); + bb -= dd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfsub_d(double a, double *b) { + qd_real bb(b); + bb -= a; + TO_DOUBLE_PTR(bb, b); +} + + + +/* selfmul */ +void c_qd_selfmul(const double *a, double *b) { + qd_real bb(b); + bb *= qd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfmul_dd(const double *a, double *b) { + qd_real bb(b); + bb *= dd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfmul_d(double a, double *b) { + qd_real bb(b); + bb *= a; + TO_DOUBLE_PTR(bb, b); +} + + + +/* selfdiv */ +void c_qd_selfdiv(const double *a, double *b) { + qd_real bb(b); + bb /= qd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfdiv_dd(const double *a, double *b) { + qd_real bb(b); + bb /= dd_real(a); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_selfdiv_d(double a, double *b) { + qd_real bb(b); + bb /= a; + TO_DOUBLE_PTR(bb, b); +} + + + +/* copy */ +void c_qd_copy(const double *a, double *b) { + b[0] = a[0]; + b[1] = a[1]; + b[2] = a[2]; + b[3] = a[3]; +} +void c_qd_copy_dd(const double *a, double *b) { + b[0] = a[0]; + b[1] = a[1]; + b[2] = 0.0; + b[3] = 0.0; +} +void c_qd_copy_d(double a, double *b) { + b[0] = a; + b[1] = 0.0; + b[2] = 0.0; + b[3] = 0.0; +} + + +void c_qd_sqrt(const double *a, double *b) { + qd_real bb; + bb = sqrt(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_sqr(const double *a, double *b) { + qd_real bb; + bb = sqr(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_abs(const double *a, double *b) { + qd_real bb; + bb = abs(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_npwr(const double *a, int n, double *b) { + qd_real bb; + bb = npwr(qd_real(a), n); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_nroot(const double *a, int n, double *b) { + qd_real bb; + bb = nroot(qd_real(a), n); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_nint(const double *a, double *b) { + qd_real bb; + bb = nint(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_aint(const double *a, double *b) { + qd_real bb; + bb = aint(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_floor(const double *a, double *b) { + qd_real bb; + bb = floor(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_ceil(const double *a, double *b) { + qd_real bb; + bb = ceil(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_log(const double *a, double *b) { + qd_real bb; + bb = log(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_log10(const double *a, double *b) { + qd_real bb; + bb = log10(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_exp(const double *a, double *b) { + qd_real bb; + bb = exp(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_sin(const double *a, double *b) { + qd_real bb; + bb = sin(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_cos(const double *a, double *b) { + qd_real bb; + bb = cos(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_tan(const double *a, double *b) { + qd_real bb; + bb = tan(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_asin(const double *a, double *b) { + qd_real bb; + bb = asin(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_acos(const double *a, double *b) { + qd_real bb; + bb = acos(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_atan(const double *a, double *b) { + qd_real bb; + bb = atan(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_atan2(const double *a, const double *b, double *c) { + qd_real cc; + cc = atan2(qd_real(a), qd_real(b)); + TO_DOUBLE_PTR(cc, c); +} + +void c_qd_sinh(const double *a, double *b) { + qd_real bb; + bb = sinh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_cosh(const double *a, double *b) { + qd_real bb; + bb = cosh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_tanh(const double *a, double *b) { + qd_real bb; + bb = tanh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_asinh(const double *a, double *b) { + qd_real bb; + bb = asinh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_acosh(const double *a, double *b) { + qd_real bb; + bb = acosh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} +void c_qd_atanh(const double *a, double *b) { + qd_real bb; + bb = atanh(qd_real(a)); + TO_DOUBLE_PTR(bb, b); +} + +void c_qd_sincos(const double *a, double *s, double *c) { + qd_real ss, cc; + sincos(qd_real(a), ss, cc); + TO_DOUBLE_PTR(cc, c); + TO_DOUBLE_PTR(ss, s); +} + +void c_qd_sincosh(const double *a, double *s, double *c) { + qd_real ss, cc; + sincosh(qd_real(a), ss, cc); + TO_DOUBLE_PTR(cc, c); + TO_DOUBLE_PTR(ss, s); +} + +void c_qd_read(const char *s, double *a) { + qd_real aa(s); + TO_DOUBLE_PTR(aa, a); +} + +void c_qd_swrite(const double *a, int precision, char *s, int len) { + qd_real(a).write(s, len, precision); +} + +void c_qd_write(const double *a) { + std::cout << qd_real(a).to_string(qd_real::_ndigits) << std::endl; +} + +void c_qd_neg(const double *a, double *b) { + b[0] = -a[0]; + b[1] = -a[1]; + b[2] = -a[2]; + b[3] = -a[3]; +} + +void c_qd_rand(double *a) { + qd_real aa; + aa = qdrand(); + TO_DOUBLE_PTR(aa, a); +} + +void c_qd_comp(const double *a, const double *b, int *result) { + qd_real aa(a), bb(b); + if (aa < bb) + *result = -1; + else if (aa > bb) + *result = 1; + else + *result = 0; +} + +void c_qd_comp_qd_d(const double *a, double b, int *result) { + qd_real aa(a); + if (aa < b) + *result = -1; + else if (aa > b) + *result = 1; + else + *result = 0; +} + +void c_qd_comp_d_qd(double a, const double *b, int *result) { + qd_real bb(b); + if (a < bb) + *result = -1; + else if (a > bb) + *result = 1; + else + *result = 0; +} + +void c_qd_pi(double *a) { + TO_DOUBLE_PTR(qd_real::_pi, a); +} + +} diff --git a/src/external/PackedCSparse/qd/c_qd.h b/src/external/PackedCSparse/qd/c_qd.h new file mode 100644 index 00000000..d11a7ff1 --- /dev/null +++ b/src/external/PackedCSparse/qd/c_qd.h @@ -0,0 +1,119 @@ +/* + * include/c_qd.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains C wrapper function prototypes for quad-double precision + * arithmetic. This can also be used from fortran code. + */ +#ifndef _QD_C_QD_H +#define _QD_C_QD_H + +#include "c_dd.h" +#include "qd_config.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/* add */ +void c_qd_add(const double *a, const double *b, double *c); +void c_qd_add_dd_qd(const double *a, const double *b, double *c); +void c_qd_add_qd_dd(const double *a, const double *b, double *c); +void c_qd_add_d_qd(double a, const double *b, double *c); +void c_qd_add_qd_d(const double *a, double b, double *c); +void c_qd_selfadd(const double *a, double *b); +void c_qd_selfadd_dd(const double *a, double *b); +void c_qd_selfadd_d(double a, double *b); + +/* sub */ +void c_qd_sub(const double *a, const double *b, double *c); +void c_qd_sub_dd_qd(const double *a, const double *b, double *c); +void c_qd_sub_qd_dd(const double *a, const double *b, double *c); +void c_qd_sub_d_qd(double a, const double *b, double *c); +void c_qd_sub_qd_d(const double *a, double b, double *c); +void c_qd_selfsub(const double *a, double *b); +void c_qd_selfsub_dd(const double *a, double *b); +void c_qd_selfsub_d(double a, double *b); + +/* mul */ +void c_qd_mul(const double *a, const double *b, double *c); +void c_qd_mul_dd_qd(const double *a, const double *b, double *c); +void c_qd_mul_qd_dd(const double *a, const double *b, double *c); +void c_qd_mul_d_qd(double a, const double *b, double *c); +void c_qd_mul_qd_d(const double *a, double b, double *c); +void c_qd_selfmul(const double *a, double *b); +void c_qd_selfmul_dd(const double *a, double *b); +void c_qd_selfmul_d(double a, double *b); + +/* div */ +void c_qd_div(const double *a, const double *b, double *c); +void c_qd_div_dd_qd(const double *a, const double *b, double *c); +void c_qd_div_qd_dd(const double *a, const double *b, double *c); +void c_qd_div_d_qd(double a, const double *b, double *c); +void c_qd_div_qd_d(const double *a, double b, double *c); +void c_qd_selfdiv(const double *a, double *b); +void c_qd_selfdiv_dd(const double *a, double *b); +void c_qd_selfdiv_d(double a, double *b); + +/* copy */ +void c_qd_copy(const double *a, double *b); +void c_qd_copy_dd(const double *a, double *b); +void c_qd_copy_d(double a, double *b); + +void c_qd_sqrt(const double *a, double *b); +void c_qd_sqr(const double *a, double *b); + +void c_qd_abs(const double *a, double *b); + +void c_qd_npwr(const double *a, int b, double *c); +void c_qd_nroot(const double *a, int b, double *c); + +void c_qd_nint(const double *a, double *b); +void c_qd_aint(const double *a, double *b); +void c_qd_floor(const double *a, double *b); +void c_qd_ceil(const double *a, double *b); + +void c_qd_exp(const double *a, double *b); +void c_qd_log(const double *a, double *b); +void c_qd_log10(const double *a, double *b); + +void c_qd_sin(const double *a, double *b); +void c_qd_cos(const double *a, double *b); +void c_qd_tan(const double *a, double *b); + +void c_qd_asin(const double *a, double *b); +void c_qd_acos(const double *a, double *b); +void c_qd_atan(const double *a, double *b); +void c_qd_atan2(const double *a, const double *b, double *c); + +void c_qd_sinh(const double *a, double *b); +void c_qd_cosh(const double *a, double *b); +void c_qd_tanh(const double *a, double *b); + +void c_qd_asinh(const double *a, double *b); +void c_qd_acosh(const double *a, double *b); +void c_qd_atanh(const double *a, double *b); + +void c_qd_sincos(const double *a, double *s, double *c); +void c_qd_sincosh(const double *a, double *s, double *c); + +void c_qd_read(const char *s, double *a); +void c_qd_swrite(const double *a, int precision, char *s, int len); +void c_qd_write(const double *a); +void c_qd_neg(const double *a, double *b); +void c_qd_rand(double *a); +void c_qd_comp(const double *a, const double *b, int *result); +void c_qd_comp_qd_d(const double *a, double b, int *result); +void c_qd_comp_d_qd(double a, const double *b, int *result); +void c_qd_pi(double *a); + +#ifdef __cplusplus +} +#endif + +#endif /* _QD_C_QD_H */ diff --git a/src/external/PackedCSparse/qd/dd_const.cc b/src/external/PackedCSparse/qd/dd_const.cc new file mode 100644 index 00000000..38b7b5ae --- /dev/null +++ b/src/external/PackedCSparse/qd/dd_const.cc @@ -0,0 +1,40 @@ +/* + * src/dd_const.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2007 + */ +#include "qd_config.h" +#include "dd_real.h" + +const dd_real dd_real::_2pi = dd_real(6.283185307179586232e+00, + 2.449293598294706414e-16); +const dd_real dd_real::_pi = dd_real(3.141592653589793116e+00, + 1.224646799147353207e-16); +const dd_real dd_real::_pi2 = dd_real(1.570796326794896558e+00, + 6.123233995736766036e-17); +const dd_real dd_real::_pi4 = dd_real(7.853981633974482790e-01, + 3.061616997868383018e-17); +const dd_real dd_real::_3pi4 = dd_real(2.356194490192344837e+00, + 9.1848509936051484375e-17); +const dd_real dd_real::_e = dd_real(2.718281828459045091e+00, + 1.445646891729250158e-16); +const dd_real dd_real::_log2 = dd_real(6.931471805599452862e-01, + 2.319046813846299558e-17); +const dd_real dd_real::_log10 = dd_real(2.302585092994045901e+00, + -2.170756223382249351e-16); +const dd_real dd_real::_nan = dd_real(qd::_d_nan, qd::_d_nan); +const dd_real dd_real::_inf = dd_real(qd::_d_inf, qd::_d_inf); + +const double dd_real::_eps = 4.93038065763132e-32; // 2^-104 +const double dd_real::_min_normalized = 2.0041683600089728e-292; // = 2^(-1022 + 53) +const dd_real dd_real::_max = + dd_real(1.79769313486231570815e+308, 9.97920154767359795037e+291); +const dd_real dd_real::_safe_max = + dd_real(1.7976931080746007281e+308, 9.97920154767359795037e+291); +const int dd_real::_ndigits = 31; + + diff --git a/src/external/PackedCSparse/qd/dd_inline.h b/src/external/PackedCSparse/qd/dd_inline.h new file mode 100644 index 00000000..89bc24f2 --- /dev/null +++ b/src/external/PackedCSparse/qd/dd_inline.h @@ -0,0 +1,621 @@ +/* + * include/dd_inline.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains small functions (suitable for inlining) in the double-double + * arithmetic package. + */ +#ifndef _QD_DD_INLINE_H +#define _QD_DD_INLINE_H + +#include +#include "inline.h" + +#ifndef QD_INLINE +#define inline +#endif + + +/*********** Additions ************/ +/* double-double = double + double */ +inline dd_real dd_real::add(double a, double b) { + double s, e; + s = qd::two_sum(a, b, e); + return dd_real(s, e); +} + +/* double-double + double */ +inline dd_real operator+(const dd_real &a, double b) { + double s1, s2; + s1 = qd::two_sum(a.x[0], b, s2); + s2 += a.x[1]; + s1 = qd::quick_two_sum(s1, s2, s2); + return dd_real(s1, s2); +} + +/* double-double + double-double */ +inline dd_real dd_real::ieee_add(const dd_real &a, const dd_real &b) { + /* This one satisfies IEEE style error bound, + due to K. Briggs and W. Kahan. */ + double s1, s2, t1, t2; + + s1 = qd::two_sum(a.x[0], b.x[0], s2); + t1 = qd::two_sum(a.x[1], b.x[1], t2); + s2 += t1; + s1 = qd::quick_two_sum(s1, s2, s2); + s2 += t2; + s1 = qd::quick_two_sum(s1, s2, s2); + return dd_real(s1, s2); +} + +inline dd_real dd_real::sloppy_add(const dd_real &a, const dd_real &b) { + /* This is the less accurate version ... obeys Cray-style + error bound. */ + double s, e; + + s = qd::two_sum(a.x[0], b.x[0], e); + e += (a.x[1] + b.x[1]); + s = qd::quick_two_sum(s, e, e); + return dd_real(s, e); +} + +inline dd_real operator+(const dd_real &a, const dd_real &b) { +#ifndef QD_IEEE_ADD + return dd_real::sloppy_add(a, b); +#else + return dd_real::ieee_add(a, b); +#endif +} + +/* double + double-double */ +inline dd_real operator+(double a, const dd_real &b) { + return (b + a); +} + + +/*********** Self-Additions ************/ +/* double-double += double */ +inline dd_real &dd_real::operator+=(double a) { + double s1, s2; + s1 = qd::two_sum(x[0], a, s2); + s2 += x[1]; + x[0] = qd::quick_two_sum(s1, s2, x[1]); + return *this; +} + +/* double-double += double-double */ +inline dd_real &dd_real::operator+=(const dd_real &a) { +#ifndef QD_IEEE_ADD + double s, e; + s = qd::two_sum(x[0], a.x[0], e); + e += x[1]; + e += a.x[1]; + x[0] = qd::quick_two_sum(s, e, x[1]); + return *this; +#else + double s1, s2, t1, t2; + s1 = qd::two_sum(x[0], a.x[0], s2); + t1 = qd::two_sum(x[1], a.x[1], t2); + s2 += t1; + s1 = qd::quick_two_sum(s1, s2, s2); + s2 += t2; + x[0] = qd::quick_two_sum(s1, s2, x[1]); + return *this; +#endif +} + +/*********** Subtractions ************/ +/* double-double = double - double */ +inline dd_real dd_real::sub(double a, double b) { + double s, e; + s = qd::two_diff(a, b, e); + return dd_real(s, e); +} + +/* double-double - double */ +inline dd_real operator-(const dd_real &a, double b) { + double s1, s2; + s1 = qd::two_diff(a.x[0], b, s2); + s2 += a.x[1]; + s1 = qd::quick_two_sum(s1, s2, s2); + return dd_real(s1, s2); +} + +/* double-double - double-double */ +inline dd_real operator-(const dd_real &a, const dd_real &b) { +#ifndef QD_IEEE_ADD + double s, e; + s = qd::two_diff(a.x[0], b.x[0], e); + e += a.x[1]; + e -= b.x[1]; + s = qd::quick_two_sum(s, e, e); + return dd_real(s, e); +#else + double s1, s2, t1, t2; + s1 = qd::two_diff(a.x[0], b.x[0], s2); + t1 = qd::two_diff(a.x[1], b.x[1], t2); + s2 += t1; + s1 = qd::quick_two_sum(s1, s2, s2); + s2 += t2; + s1 = qd::quick_two_sum(s1, s2, s2); + return dd_real(s1, s2); +#endif +} + +/* double - double-double */ +inline dd_real operator-(double a, const dd_real &b) { + double s1, s2; + s1 = qd::two_diff(a, b.x[0], s2); + s2 -= b.x[1]; + s1 = qd::quick_two_sum(s1, s2, s2); + return dd_real(s1, s2); +} + +/*********** Self-Subtractions ************/ +/* double-double -= double */ +inline dd_real &dd_real::operator-=(double a) { + double s1, s2; + s1 = qd::two_diff(x[0], a, s2); + s2 += x[1]; + x[0] = qd::quick_two_sum(s1, s2, x[1]); + return *this; +} + +/* double-double -= double-double */ +inline dd_real &dd_real::operator-=(const dd_real &a) { +#ifndef QD_IEEE_ADD + double s, e; + s = qd::two_diff(x[0], a.x[0], e); + e += x[1]; + e -= a.x[1]; + x[0] = qd::quick_two_sum(s, e, x[1]); + return *this; +#else + double s1, s2, t1, t2; + s1 = qd::two_diff(x[0], a.x[0], s2); + t1 = qd::two_diff(x[1], a.x[1], t2); + s2 += t1; + s1 = qd::quick_two_sum(s1, s2, s2); + s2 += t2; + x[0] = qd::quick_two_sum(s1, s2, x[1]); + return *this; +#endif +} + +/*********** Unary Minus ***********/ +inline dd_real dd_real::operator-() const { + return dd_real(-x[0], -x[1]); +} + +/*********** Multiplications ************/ +/* double-double = double * double */ +inline dd_real dd_real::mul(double a, double b) { + double p, e; + p = qd::two_prod(a, b, e); + return dd_real(p, e); +} + +/* double-double * (2.0 ^ exp) */ +inline dd_real ldexp(const dd_real &a, int exp) { + return dd_real(std::ldexp(a.x[0], exp), std::ldexp(a.x[1], exp)); +} + +/* double-double * double, where double is a power of 2. */ +inline dd_real mul_pwr2(const dd_real &a, double b) { + return dd_real(a.x[0] * b, a.x[1] * b); +} + +/* double-double * double */ +inline dd_real operator*(const dd_real &a, double b) { + double p1, p2; + + p1 = qd::two_prod(a.x[0], b, p2); + p2 += (a.x[1] * b); + p1 = qd::quick_two_sum(p1, p2, p2); + return dd_real(p1, p2); +} + +/* double-double * double-double */ +inline dd_real operator*(const dd_real &a, const dd_real &b) { + double p1, p2; + + p1 = qd::two_prod(a.x[0], b.x[0], p2); + p2 += (a.x[0] * b.x[1] + a.x[1] * b.x[0]); + p1 = qd::quick_two_sum(p1, p2, p2); + return dd_real(p1, p2); +} + +/* double * double-double */ +inline dd_real operator*(double a, const dd_real &b) { + return (b * a); +} + +/*********** Self-Multiplications ************/ +/* double-double *= double */ +inline dd_real &dd_real::operator*=(double a) { + double p1, p2; + p1 = qd::two_prod(x[0], a, p2); + p2 += x[1] * a; + x[0] = qd::quick_two_sum(p1, p2, x[1]); + return *this; +} + +/* double-double *= double-double */ +inline dd_real &dd_real::operator*=(const dd_real &a) { + double p1, p2; + p1 = qd::two_prod(x[0], a.x[0], p2); + p2 += a.x[1] * x[0]; + p2 += a.x[0] * x[1]; + x[0] = qd::quick_two_sum(p1, p2, x[1]); + return *this; +} + +/*********** Divisions ************/ +inline dd_real dd_real::div(double a, double b) { + double q1, q2; + double p1, p2; + double s, e; + + q1 = a / b; + + /* Compute a - q1 * b */ + p1 = qd::two_prod(q1, b, p2); + s = qd::two_diff(a, p1, e); + e -= p2; + + /* get next approximation */ + q2 = (s + e) / b; + + s = qd::quick_two_sum(q1, q2, e); + + return dd_real(s, e); +} + +/* double-double / double */ +inline dd_real operator/(const dd_real &a, double b) { + + double q1, q2; + double p1, p2; + double s, e; + dd_real r; + + q1 = a.x[0] / b; /* approximate quotient. */ + + /* Compute this - q1 * d */ + p1 = qd::two_prod(q1, b, p2); + s = qd::two_diff(a.x[0], p1, e); + e += a.x[1]; + e -= p2; + + /* get next approximation. */ + q2 = (s + e) / b; + + /* renormalize */ + r.x[0] = qd::quick_two_sum(q1, q2, r.x[1]); + + return r; +} + +inline dd_real dd_real::sloppy_div(const dd_real &a, const dd_real &b) { + double s1, s2; + double q1, q2; + dd_real r; + + q1 = a.x[0] / b.x[0]; /* approximate quotient */ + + /* compute this - q1 * dd */ + r = b * q1; + s1 = qd::two_diff(a.x[0], r.x[0], s2); + s2 -= r.x[1]; + s2 += a.x[1]; + + /* get next approximation */ + q2 = (s1 + s2) / b.x[0]; + + /* renormalize */ + r.x[0] = qd::quick_two_sum(q1, q2, r.x[1]); + return r; +} + +inline dd_real dd_real::accurate_div(const dd_real &a, const dd_real &b) { + double q1, q2, q3; + dd_real r; + + q1 = a.x[0] / b.x[0]; /* approximate quotient */ + + r = a - q1 * b; + + q2 = r.x[0] / b.x[0]; + r -= (q2 * b); + + q3 = r.x[0] / b.x[0]; + + q1 = qd::quick_two_sum(q1, q2, q2); + r = dd_real(q1, q2) + q3; + return r; +} + +/* double-double / double-double */ +inline dd_real operator/(const dd_real &a, const dd_real &b) { +#ifdef QD_SLOPPY_DIV + return dd_real::sloppy_div(a, b); +#else + return dd_real::accurate_div(a, b); +#endif +} + +/* double / double-double */ +inline dd_real operator/(double a, const dd_real &b) { + return dd_real(a) / b; +} + +inline dd_real inv(const dd_real &a) { + return 1.0 / a; +} + +/*********** Self-Divisions ************/ +/* double-double /= double */ +inline dd_real &dd_real::operator/=(double a) { + *this = *this / a; + return *this; +} + +/* double-double /= double-double */ +inline dd_real &dd_real::operator/=(const dd_real &a) { + *this = *this / a; + return *this; +} + +/********** Remainder **********/ +inline dd_real drem(const dd_real &a, const dd_real &b) { + dd_real n = nint(a / b); + return (a - n * b); +} + +inline dd_real divrem(const dd_real &a, const dd_real &b, dd_real &r) { + dd_real n = nint(a / b); + r = a - n * b; + return n; +} + +/*********** Squaring **********/ +inline dd_real sqr(const dd_real &a) { + double p1, p2; + double s1, s2; + p1 = qd::two_sqr(a.x[0], p2); + p2 += 2.0 * a.x[0] * a.x[1]; + p2 += a.x[1] * a.x[1]; + s1 = qd::quick_two_sum(p1, p2, s2); + return dd_real(s1, s2); +} + +inline dd_real dd_real::sqr(double a) { + double p1, p2; + p1 = qd::two_sqr(a, p2); + return dd_real(p1, p2); +} + + +/********** Exponentiation **********/ +inline dd_real dd_real::operator^(int n) { + return npwr(*this, n); +} + + +/*********** Assignments ************/ +/* double-double = double */ +inline dd_real &dd_real::operator=(double a) { + x[0] = a; + x[1] = 0.0; + return *this; +} + +/*********** Equality Comparisons ************/ +/* double-double == double */ +inline bool operator==(const dd_real &a, double b) { + return (a.x[0] == b && a.x[1] == 0.0); +} + +/* double-double == double-double */ +inline bool operator==(const dd_real &a, const dd_real &b) { + return (a.x[0] == b.x[0] && a.x[1] == b.x[1]); +} + +/* double == double-double */ +inline bool operator==(double a, const dd_real &b) { + return (a == b.x[0] && b.x[1] == 0.0); +} + +/*********** Greater-Than Comparisons ************/ +/* double-double > double */ +inline bool operator>(const dd_real &a, double b) { + return (a.x[0] > b || (a.x[0] == b && a.x[1] > 0.0)); +} + +/* double-double > double-double */ +inline bool operator>(const dd_real &a, const dd_real &b) { + return (a.x[0] > b.x[0] || (a.x[0] == b.x[0] && a.x[1] > b.x[1])); +} + +/* double > double-double */ +inline bool operator>(double a, const dd_real &b) { + return (a > b.x[0] || (a == b.x[0] && b.x[1] < 0.0)); +} + +/*********** Less-Than Comparisons ************/ +/* double-double < double */ +inline bool operator<(const dd_real &a, double b) { + return (a.x[0] < b || (a.x[0] == b && a.x[1] < 0.0)); +} + +/* double-double < double-double */ +inline bool operator<(const dd_real &a, const dd_real &b) { + return (a.x[0] < b.x[0] || (a.x[0] == b.x[0] && a.x[1] < b.x[1])); +} + +/* double < double-double */ +inline bool operator<(double a, const dd_real &b) { + return (a < b.x[0] || (a == b.x[0] && b.x[1] > 0.0)); +} + +/*********** Greater-Than-Or-Equal-To Comparisons ************/ +/* double-double >= double */ +inline bool operator>=(const dd_real &a, double b) { + return (a.x[0] > b || (a.x[0] == b && a.x[1] >= 0.0)); +} + +/* double-double >= double-double */ +inline bool operator>=(const dd_real &a, const dd_real &b) { + return (a.x[0] > b.x[0] || (a.x[0] == b.x[0] && a.x[1] >= b.x[1])); +} + +/* double >= double-double */ +inline bool operator>=(double a, const dd_real &b) { + return (b <= a); +} + +/*********** Less-Than-Or-Equal-To Comparisons ************/ +/* double-double <= double */ +inline bool operator<=(const dd_real &a, double b) { + return (a.x[0] < b || (a.x[0] == b && a.x[1] <= 0.0)); +} + +/* double-double <= double-double */ +inline bool operator<=(const dd_real &a, const dd_real &b) { + return (a.x[0] < b.x[0] || (a.x[0] == b.x[0] && a.x[1] <= b.x[1])); +} + +/* double <= double-double */ +inline bool operator<=(double a, const dd_real &b) { + return (b >= a); +} + +/*********** Not-Equal-To Comparisons ************/ +/* double-double != double */ +inline bool operator!=(const dd_real &a, double b) { + return (a.x[0] != b || a.x[1] != 0.0); +} + +/* double-double != double-double */ +inline bool operator!=(const dd_real &a, const dd_real &b) { + return (a.x[0] != b.x[0] || a.x[1] != b.x[1]); +} + +/* double != double-double */ +inline bool operator!=(double a, const dd_real &b) { + return (a != b.x[0] || b.x[1] != 0.0); +} + +/*********** Micellaneous ************/ +/* this == 0 */ +inline bool dd_real::is_zero() const { + return (x[0] == 0.0); +} + +/* this == 1 */ +inline bool dd_real::is_one() const { + return (x[0] == 1.0 && x[1] == 0.0); +} + +/* this > 0 */ +inline bool dd_real::is_positive() const { + return (x[0] > 0.0); +} + +/* this < 0 */ +inline bool dd_real::is_negative() const { + return (x[0] < 0.0); +} + +inline dd_real::operator bool() const { + return (x[0] != 0.0); +} + +inline dd_real::operator double() const { + return to_double(*this); +} + +/* Absolute value */ +inline dd_real abs(const dd_real &a) { + return (a.x[0] < 0.0) ? -a : a; +} + +inline dd_real fabs(const dd_real &a) { + return abs(a); +} + +/* Round to Nearest integer */ +inline dd_real nint(const dd_real &a) { + double hi = qd::nint(a.x[0]); + double lo; + + if (hi == a.x[0]) { + /* High word is an integer already. Round the low word.*/ + lo = qd::nint(a.x[1]); + + /* Renormalize. This is needed if x[0] = some integer, x[1] = 1/2.*/ + hi = qd::quick_two_sum(hi, lo, lo); + } else { + /* High word is not an integer. */ + lo = 0.0; + if (std::abs(hi-a.x[0]) == 0.5 && a.x[1] < 0.0) { + /* There is a tie in the high word, consult the low word + to break the tie. */ + hi -= 1.0; /* NOTE: This does not cause INEXACT. */ + } + } + + return dd_real(hi, lo); +} + +inline dd_real floor(const dd_real &a) { + double hi = std::floor(a.x[0]); + double lo = 0.0; + + if (hi == a.x[0]) { + /* High word is integer already. Round the low word. */ + lo = std::floor(a.x[1]); + hi = qd::quick_two_sum(hi, lo, lo); + } + + return dd_real(hi, lo); +} + +inline dd_real ceil(const dd_real &a) { + double hi = std::ceil(a.x[0]); + double lo = 0.0; + + if (hi == a.x[0]) { + /* High word is integer already. Round the low word. */ + lo = std::ceil(a.x[1]); + hi = qd::quick_two_sum(hi, lo, lo); + } + + return dd_real(hi, lo); +} + +inline dd_real aint(const dd_real &a) { + return (a.x[0] >= 0.0) ? floor(a) : ceil(a); +} + +/* Cast to double. */ +inline double to_double(const dd_real &a) { + return a.x[0]; +} + +/* Cast to int. */ +inline int to_int(const dd_real &a) { + return static_cast(a.x[0]); +} + +/* Random number generator */ +inline dd_real dd_real::rand() { + return ddrand(); +} + +#endif /* _QD_DD_INLINE_H */ diff --git a/src/external/PackedCSparse/qd/dd_real.cc b/src/external/PackedCSparse/qd/dd_real.cc new file mode 100644 index 00000000..ff4d5223 --- /dev/null +++ b/src/external/PackedCSparse/qd/dd_real.cc @@ -0,0 +1,1303 @@ +/* + * src/dd_real.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2007 + * + * Contains implementation of non-inlined functions of double-double + * package. Inlined functions are found in dd_inline.h (in include directory). + */ +#include +#include +#include +#include +#include +#include +#include + +#include "qd_config.h" +#include "dd_real.h" +#include "util.h" + +#include "bits.h" + +#ifndef QD_INLINE +#include "dd_inline.h" +#endif + +using std::cout; +using std::cerr; +using std::endl; +using std::ostream; +using std::istream; +using std::ios_base; +using std::string; +using std::setw; + +/* This routine is called whenever a fatal error occurs. */ +void dd_real::error(const char *msg) { + //if (msg) { cerr << "ERROR " << msg << endl; } +} + +/* Computes the square root of the double-double number dd. + NOTE: dd must be a non-negative number. */ +QD_API dd_real sqrt(const dd_real &a) { + /* Strategy: Use Karp's trick: if x is an approximation + to sqrt(a), then + + sqrt(a) = a*x + [a - (a*x)^2] * x / 2 (approx) + + The approximation is accurate to twice the accuracy of x. + Also, the multiplication (a*x) and [-]*x can be done with + only half the precision. + */ + + if (a.is_zero()) + return 0.0; + + if (a.is_negative()) { + dd_real::error("(dd_real::sqrt): Negative argument."); + return dd_real::_nan; + } + + double x = 1.0 / std::sqrt(a.x[0]); + double ax = a.x[0] * x; + return dd_real::add(ax, (a - dd_real::sqr(ax)).x[0] * (x * 0.5)); +} + +/* Computes the square root of a double in double-double precision. + NOTE: d must not be negative. */ +dd_real dd_real::sqrt(double d) { + return ::sqrt(dd_real(d)); +} + +/* Computes the n-th root of the double-double number a. + NOTE: n must be a positive integer. + NOTE: If n is even, then a must not be negative. */ +dd_real nroot(const dd_real &a, int n) { + /* Strategy: Use Newton iteration for the function + + f(x) = x^(-n) - a + + to find its root a^{-1/n}. The iteration is thus + + x' = x + x * (1 - a * x^n) / n + + which converges quadratically. We can then find + a^{1/n} by taking the reciprocal. + */ + + if (n <= 0) { + dd_real::error("(dd_real::nroot): N must be positive."); + return dd_real::_nan; + } + + if (n%2 == 0 && a.is_negative()) { + dd_real::error("(dd_real::nroot): Negative argument."); + return dd_real::_nan; + } + + if (n == 1) { + return a; + } + if (n == 2) { + return sqrt(a); + } + + if (a.is_zero()) + return 0.0; + + /* Note a^{-1/n} = exp(-log(a)/n) */ + dd_real r = abs(a); + dd_real x = std::exp(-std::log(r.x[0]) / n); + + /* Perform Newton's iteration. */ + x += x * (1.0 - r * npwr(x, n)) / static_cast(n); + if (a.x[0] < 0.0) + x = -x; + return 1.0/x; +} + +/* Computes the n-th power of a double-double number. + NOTE: 0^0 causes an error. */ +dd_real npwr(const dd_real &a, int n) { + + if (n == 0) { + if (a.is_zero()) { + dd_real::error("(dd_real::npwr): Invalid argument."); + return dd_real::_nan; + } + return 1.0; + } + + dd_real r = a; + dd_real s = 1.0; + int N = std::abs(n); + + if (N > 1) { + /* Use binary exponentiation */ + while (N > 0) { + if (N % 2 == 1) { + s *= r; + } + N /= 2; + if (N > 0) + r = sqr(r); + } + } else { + s = r; + } + + /* Compute the reciprocal if n is negative. */ + if (n < 0) + return (1.0 / s); + + return s; +} + +dd_real pow(const dd_real &a, int n) { + return npwr(a, n); +} + +dd_real pow(const dd_real &a, const dd_real &b) { + return exp(b * log(a)); +} + +static const int n_inv_fact = 15; +static const double inv_fact[n_inv_fact][2] = { + { 1.66666666666666657e-01, 9.25185853854297066e-18}, + { 4.16666666666666644e-02, 2.31296463463574266e-18}, + { 8.33333333333333322e-03, 1.15648231731787138e-19}, + { 1.38888888888888894e-03, -5.30054395437357706e-20}, + { 1.98412698412698413e-04, 1.72095582934207053e-22}, + { 2.48015873015873016e-05, 2.15119478667758816e-23}, + { 2.75573192239858925e-06, -1.85839327404647208e-22}, + { 2.75573192239858883e-07, 2.37677146222502973e-23}, + { 2.50521083854417202e-08, -1.44881407093591197e-24}, + { 2.08767569878681002e-09, -1.20734505911325997e-25}, + { 1.60590438368216133e-10, 1.25852945887520981e-26}, + { 1.14707455977297245e-11, 2.06555127528307454e-28}, + { 7.64716373181981641e-13, 7.03872877733453001e-30}, + { 4.77947733238738525e-14, 4.39920548583408126e-31}, + { 2.81145725434552060e-15, 1.65088427308614326e-31} +}; + +/* Exponential. Computes exp(x) in double-double precision. */ +dd_real exp(const dd_real &a) { + /* Strategy: We first reduce the size of x by noting that + + exp(kr + m * log(2)) = 2^m * exp(r)^k + + where m and k are integers. By choosing m appropriately + we can make |kr| <= log(2) / 2 = 0.347. Then exp(r) is + evaluated using the familiar Taylor series. Reducing the + argument substantially speeds up the convergence. */ + + const double k = 512.0; + const double inv_k = 1.0 / k; + + if (a.x[0] <= -709.0) + return 0.0; + + if (a.x[0] >= 709.0) + return dd_real::_inf; + + if (a.is_zero()) + return 1.0; + + if (a.is_one()) + return dd_real::_e; + + double m = std::floor(a.x[0] / dd_real::_log2.x[0] + 0.5); + dd_real r = mul_pwr2(a - dd_real::_log2 * m, inv_k); + dd_real s, t, p; + + p = sqr(r); + s = r + mul_pwr2(p, 0.5); + p *= r; + t = p * dd_real(inv_fact[0][0], inv_fact[0][1]); + int i = 0; + do { + s += t; + p *= r; + ++i; + t = p * dd_real(inv_fact[i][0], inv_fact[i][1]); + } while (std::abs(to_double(t)) > inv_k * dd_real::_eps && i < 5); + + s += t; + + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s += 1.0; + + return ldexp(s, static_cast(m)); +} + +/* Logarithm. Computes log(x) in double-double precision. + This is a natural logarithm (i.e., base e). */ +dd_real log(const dd_real &a) { + /* Strategy. The Taylor series for log converges much more + slowly than that of exp, due to the lack of the factorial + term in the denominator. Hence this routine instead tries + to determine the root of the function + + f(x) = exp(x) - a + + using Newton iteration. The iteration is given by + + x' = x - f(x)/f'(x) + = x - (1 - a * exp(-x)) + = x + a * exp(-x) - 1. + + Only one iteration is needed, since Newton's iteration + approximately doubles the number of digits per iteration. */ + + if (a.is_one()) { + return 0.0; + } + + if (a.x[0] <= 0.0) { + dd_real::error("(dd_real::log): Non-positive argument."); + return dd_real::_nan; + } + + dd_real x = std::log(a.x[0]); /* Initial approximation */ + + x = x + a * exp(-x) - 1.0; + return x; +} + +dd_real log10(const dd_real &a) { + return log(a) / dd_real::_log10; +} + +static const dd_real _pi16 = dd_real(1.963495408493620697e-01, + 7.654042494670957545e-18); + +/* Table of sin(k * pi/16) and cos(k * pi/16). */ +static const double sin_table [4][2] = { + {1.950903220161282758e-01, -7.991079068461731263e-18}, + {3.826834323650897818e-01, -1.005077269646158761e-17}, + {5.555702330196021776e-01, 4.709410940561676821e-17}, + {7.071067811865475727e-01, -4.833646656726456726e-17} +}; + +static const double cos_table [4][2] = { + {9.807852804032304306e-01, 1.854693999782500573e-17}, + {9.238795325112867385e-01, 1.764504708433667706e-17}, + {8.314696123025452357e-01, 1.407385698472802389e-18}, + {7.071067811865475727e-01, -4.833646656726456726e-17} +}; + +/* Computes sin(a) using Taylor series. + Assumes |a| <= pi/32. */ +static dd_real sin_taylor(const dd_real &a) { + const double thresh = 0.5 * std::abs(to_double(a)) * dd_real::_eps; + dd_real r, s, t, x; + + if (a.is_zero()) { + return 0.0; + } + + int i = 0; + x = -sqr(a); + s = a; + r = a; + do { + r *= x; + t = r * dd_real(inv_fact[i][0], inv_fact[i][1]); + s += t; + i += 2; + } while (i < n_inv_fact && std::abs(to_double(t)) > thresh); + + return s; +} + +static dd_real cos_taylor(const dd_real &a) { + const double thresh = 0.5 * dd_real::_eps; + dd_real r, s, t, x; + + if (a.is_zero()) { + return 1.0; + } + + x = -sqr(a); + r = x; + s = 1.0 + mul_pwr2(r, 0.5); + int i = 1; + do { + r *= x; + t = r * dd_real(inv_fact[i][0], inv_fact[i][1]); + s += t; + i += 2; + } while (i < n_inv_fact && std::abs(to_double(t)) > thresh); + + return s; +} + +static void sincos_taylor(const dd_real &a, + dd_real &sin_a, dd_real &cos_a) { + if (a.is_zero()) { + sin_a = 0.0; + cos_a = 1.0; + return; + } + + sin_a = sin_taylor(a); + cos_a = sqrt(1.0 - sqr(sin_a)); +} + + +dd_real sin(const dd_real &a) { + + /* Strategy. To compute sin(x), we choose integers a, b so that + + x = s + a * (pi/2) + b * (pi/16) + + and |s| <= pi/32. Using the fact that + + sin(pi/16) = 0.5 * sqrt(2 - sqrt(2 + sqrt(2))) + + we can compute sin(x) from sin(s), cos(s). This greatly + increases the convergence of the sine Taylor series. */ + + if (a.is_zero()) { + return 0.0; + } + + // approximately reduce modulo 2*pi + dd_real z = nint(a / dd_real::_2pi); + dd_real r = a - dd_real::_2pi * z; + + // approximately reduce modulo pi/2 and then modulo pi/16. + dd_real t; + double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5); + t = r - dd_real::_pi2 * q; + int j = static_cast(q); + q = std::floor(t.x[0] / _pi16.x[0] + 0.5); + t -= _pi16 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (j < -2 || j > 2) { + dd_real::error("(dd_real::sin): Cannot reduce modulo pi/2."); + return dd_real::_nan; + } + + if (abs_k > 4) { + dd_real::error("(dd_real::sin): Cannot reduce modulo pi/16."); + return dd_real::_nan; + } + + if (k == 0) { + switch (j) { + case 0: + return sin_taylor(t); + case 1: + return cos_taylor(t); + case -1: + return -cos_taylor(t); + default: + return -sin_taylor(t); + } + } + + dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]); + dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]); + dd_real sin_t, cos_t; + sincos_taylor(t, sin_t, cos_t); + if (j == 0) { + if (k > 0) { + r = u * sin_t + v * cos_t; + } else { + r = u * sin_t - v * cos_t; + } + } else if (j == 1) { + if (k > 0) { + r = u * cos_t - v * sin_t; + } else { + r = u * cos_t + v * sin_t; + } + } else if (j == -1) { + if (k > 0) { + r = v * sin_t - u * cos_t; + } else if (k < 0) { + r = -u * cos_t - v * sin_t; + } + } else { + if (k > 0) { + r = -u * sin_t - v * cos_t; + } else { + r = v * cos_t - u * sin_t; + } + } + + return r; +} + +dd_real cos(const dd_real &a) { + + if (a.is_zero()) { + return 1.0; + } + + // approximately reduce modulo 2*pi + dd_real z = nint(a / dd_real::_2pi); + dd_real r = a - z * dd_real::_2pi; + + // approximately reduce modulo pi/2 and then modulo pi/16 + dd_real t; + double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5); + t = r - dd_real::_pi2 * q; + int j = static_cast(q); + q = std::floor(t.x[0] / _pi16.x[0] + 0.5); + t -= _pi16 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (j < -2 || j > 2) { + dd_real::error("(dd_real::cos): Cannot reduce modulo pi/2."); + return dd_real::_nan; + } + + if (abs_k > 4) { + dd_real::error("(dd_real::cos): Cannot reduce modulo pi/16."); + return dd_real::_nan; + } + + if (k == 0) { + switch (j) { + case 0: + return cos_taylor(t); + case 1: + return -sin_taylor(t); + case -1: + return sin_taylor(t); + default: + return -cos_taylor(t); + } + } + + dd_real sin_t, cos_t; + sincos_taylor(t, sin_t, cos_t); + dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]); + dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]); + + if (j == 0) { + if (k > 0) { + r = u * cos_t - v * sin_t; + } else { + r = u * cos_t + v * sin_t; + } + } else if (j == 1) { + if (k > 0) { + r = - u * sin_t - v * cos_t; + } else { + r = v * cos_t - u * sin_t; + } + } else if (j == -1) { + if (k > 0) { + r = u * sin_t + v * cos_t; + } else { + r = u * sin_t - v * cos_t; + } + } else { + if (k > 0) { + r = v * sin_t - u * cos_t; + } else { + r = - u * cos_t - v * sin_t; + } + } + + return r; +} + +void sincos(const dd_real &a, dd_real &sin_a, dd_real &cos_a) { + + if (a.is_zero()) { + sin_a = 0.0; + cos_a = 1.0; + return; + } + + // approximately reduce modulo 2*pi + dd_real z = nint(a / dd_real::_2pi); + dd_real r = a - dd_real::_2pi * z; + + // approximately reduce module pi/2 and pi/16 + dd_real t; + double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5); + t = r - dd_real::_pi2 * q; + int j = static_cast(q); + int abs_j = std::abs(j); + q = std::floor(t.x[0] / _pi16.x[0] + 0.5); + t -= _pi16 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (abs_j > 2) { + dd_real::error("(dd_real::sincos): Cannot reduce modulo pi/2."); + cos_a = sin_a = dd_real::_nan; + return; + } + + if (abs_k > 4) { + dd_real::error("(dd_real::sincos): Cannot reduce modulo pi/16."); + cos_a = sin_a = dd_real::_nan; + return; + } + + dd_real sin_t, cos_t; + dd_real s, c; + + sincos_taylor(t, sin_t, cos_t); + + if (abs_k == 0) { + s = sin_t; + c = cos_t; + } else { + dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]); + dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]); + + if (k > 0) { + s = u * sin_t + v * cos_t; + c = u * cos_t - v * sin_t; + } else { + s = u * sin_t - v * cos_t; + c = u * cos_t + v * sin_t; + } + } + + if (abs_j == 0) { + sin_a = s; + cos_a = c; + } else if (j == 1) { + sin_a = c; + cos_a = -s; + } else if (j == -1) { + sin_a = -c; + cos_a = s; + } else { + sin_a = -s; + cos_a = -c; + } + +} + +dd_real atan(const dd_real &a) { + return atan2(a, dd_real(1.0)); +} + +dd_real atan2(const dd_real &y, const dd_real &x) { + /* Strategy: Instead of using Taylor series to compute + arctan, we instead use Newton's iteration to solve + the equation + + sin(z) = y/r or cos(z) = x/r + + where r = sqrt(x^2 + y^2). + The iteration is given by + + z' = z + (y - sin(z)) / cos(z) (for equation 1) + z' = z - (x - cos(z)) / sin(z) (for equation 2) + + Here, x and y are normalized so that x^2 + y^2 = 1. + If |x| > |y|, then first iteration is used since the + denominator is larger. Otherwise, the second is used. + */ + + if (x.is_zero()) { + + if (y.is_zero()) { + /* Both x and y is zero. */ + dd_real::error("(dd_real::atan2): Both arguments zero."); + return dd_real::_nan; + } + + return (y.is_positive()) ? dd_real::_pi2 : -dd_real::_pi2; + } else if (y.is_zero()) { + return (x.is_positive()) ? dd_real(0.0) : dd_real::_pi; + } + + if (x == y) { + return (y.is_positive()) ? dd_real::_pi4 : -dd_real::_3pi4; + } + + if (x == -y) { + return (y.is_positive()) ? dd_real::_3pi4 : -dd_real::_pi4; + } + + dd_real r = sqrt(sqr(x) + sqr(y)); + dd_real xx = x / r; + dd_real yy = y / r; + + /* Compute double precision approximation to atan. */ + dd_real z = std::atan2(to_double(y), to_double(x)); + dd_real sin_z, cos_z; + + if (std::abs(xx.x[0]) > std::abs(yy.x[0])) { + /* Use Newton iteration 1. z' = z + (y - sin(z)) / cos(z) */ + sincos(z, sin_z, cos_z); + z += (yy - sin_z) / cos_z; + } else { + /* Use Newton iteration 2. z' = z - (x - cos(z)) / sin(z) */ + sincos(z, sin_z, cos_z); + z -= (xx - cos_z) / sin_z; + } + + return z; +} + +dd_real tan(const dd_real &a) { + dd_real s, c; + sincos(a, s, c); + return s/c; +} + +dd_real asin(const dd_real &a) { + dd_real abs_a = abs(a); + + if (abs_a > 1.0) { + dd_real::error("(dd_real::asin): Argument out of domain."); + return dd_real::_nan; + } + + if (abs_a.is_one()) { + return (a.is_positive()) ? dd_real::_pi2 : -dd_real::_pi2; + } + + return atan2(a, sqrt(1.0 - sqr(a))); +} + +dd_real acos(const dd_real &a) { + dd_real abs_a = abs(a); + + if (abs_a > 1.0) { + dd_real::error("(dd_real::acos): Argument out of domain."); + return dd_real::_nan; + } + + if (abs_a.is_one()) { + return (a.is_positive()) ? dd_real(0.0) : dd_real::_pi; + } + + return atan2(sqrt(1.0 - sqr(a)), a); +} + +dd_real sinh(const dd_real &a) { + if (a.is_zero()) { + return 0.0; + } + + if (abs(a) > 0.05) { + dd_real ea = exp(a); + return mul_pwr2(ea - inv(ea), 0.5); + } + + /* since a is small, using the above formula gives + a lot of cancellation. So use Taylor series. */ + dd_real s = a; + dd_real t = a; + dd_real r = sqr(t); + double m = 1.0; + double thresh = std::abs((to_double(a)) * dd_real::_eps); + + do { + m += 2.0; + t *= r; + t /= (m-1) * m; + + s += t; + } while (abs(t) > thresh); + + return s; + +} + +dd_real cosh(const dd_real &a) { + if (a.is_zero()) { + return 1.0; + } + + dd_real ea = exp(a); + return mul_pwr2(ea + inv(ea), 0.5); +} + +dd_real tanh(const dd_real &a) { + if (a.is_zero()) { + return 0.0; + } + + if (std::abs(to_double(a)) > 0.05) { + dd_real ea = exp(a); + dd_real inv_ea = inv(ea); + return (ea - inv_ea) / (ea + inv_ea); + } else { + dd_real s, c; + s = sinh(a); + c = sqrt(1.0 + sqr(s)); + return s / c; + } +} + +void sincosh(const dd_real &a, dd_real &s, dd_real &c) { + if (std::abs(to_double(a)) <= 0.05) { + s = sinh(a); + c = sqrt(1.0 + sqr(s)); + } else { + dd_real ea = exp(a); + dd_real inv_ea = inv(ea); + s = mul_pwr2(ea - inv_ea, 0.5); + c = mul_pwr2(ea + inv_ea, 0.5); + } +} + +dd_real asinh(const dd_real &a) { + return log(a + sqrt(sqr(a) + 1.0)); +} + +dd_real acosh(const dd_real &a) { + if (a < 1.0) { + dd_real::error("(dd_real::acosh): Argument out of domain."); + return dd_real::_nan; + } + + return log(a + sqrt(sqr(a) - 1.0)); +} + +dd_real atanh(const dd_real &a) { + if (abs(a) >= 1.0) { + dd_real::error("(dd_real::atanh): Argument out of domain."); + return dd_real::_nan; + } + + return mul_pwr2(log((1.0 + a) / (1.0 - a)), 0.5); +} + +QD_API dd_real fmod(const dd_real &a, const dd_real &b) { + dd_real n = aint(a / b); + return (a - b * n); +} + +QD_API dd_real ddrand() { + static const double m_const = 4.6566128730773926e-10; /* = 2^{-31} */ + double m = m_const; + dd_real r = 0.0; + double d; + + /* Strategy: Generate 31 bits at a time, using lrand48 + random number generator. Shift the bits, and reapeat + 4 times. */ + + for (int i = 0; i < 4; i++, m *= m_const) { +// d = lrand48() * m; + d = std::rand() * m; + r += d; + } + + return r; +} + +/* polyeval(c, n, x) + Evaluates the given n-th degree polynomial at x. + The polynomial is given by the array of (n+1) coefficients. */ +dd_real polyeval(const dd_real *c, int n, const dd_real &x) { + /* Just use Horner's method of polynomial evaluation. */ + dd_real r = c[n]; + + for (int i = n-1; i >= 0; i--) { + r *= x; + r += c[i]; + } + + return r; +} + +/* polyroot(c, n, x0) + Given an n-th degree polynomial, finds a root close to + the given guess x0. Note that this uses simple Newton + iteration scheme, and does not work for multiple roots. */ +QD_API dd_real polyroot(const dd_real *c, int n, + const dd_real &x0, int max_iter, double thresh) { + dd_real x = x0; + dd_real f; + dd_real *d = new dd_real[n]; + bool conv = false; + int i; + double max_c = std::abs(to_double(c[0])); + double v; + + if (thresh == 0.0) thresh = dd_real::_eps; + + /* Compute the coefficients of the derivatives. */ + for (i = 1; i <= n; i++) { + v = std::abs(to_double(c[i])); + if (v > max_c) max_c = v; + d[i-1] = c[i] * static_cast(i); + } + thresh *= max_c; + + /* Newton iteration. */ + for (i = 0; i < max_iter; i++) { + f = polyeval(c, n, x); + + if (abs(f) < thresh) { + conv = true; + break; + } + x -= (f / polyeval(d, n-1, x)); + } + delete [] d; + + if (!conv) { + dd_real::error("(dd_real::polyroot): Failed to converge."); + return dd_real::_nan; + } + + return x; +} + + +/* Constructor. Reads a double-double number from the string s + and constructs a double-double number. */ +dd_real::dd_real(const char *s) { + if (dd_real::read(s, *this)) { + dd_real::error("(dd_real::dd_real): INPUT ERROR."); + *this = dd_real::_nan; + } +} + +dd_real &dd_real::operator=(const char *s) { + if (dd_real::read(s, *this)) { + dd_real::error("(dd_real::operator=): INPUT ERROR."); + *this = dd_real::_nan; + } + return *this; +} + +/* Outputs the double-double number dd. */ +ostream &operator<<(ostream &os, const dd_real &dd) { + bool showpos = (os.flags() & ios_base::showpos) != 0; + bool uppercase = (os.flags() & ios_base::uppercase) != 0; + return os << dd.to_string((int)os.precision(), (int)os.width(), os.flags(), + showpos, uppercase, os.fill()); +} + +/* Reads in the double-double number a. */ +istream &operator>>(istream &s, dd_real &a) { + char str[255]; + s >> str; + a = dd_real(str); + return s; +} + +void dd_real::to_digits(char *s, int &expn, int precision) const { + int D = precision + 1; /* number of digits to compute */ + + dd_real r = abs(*this); + int e; /* exponent */ + int i, d; + + if (x[0] == 0.0) { + /* this == 0.0 */ + expn = 0; + for (i = 0; i < precision; i++) s[i] = '0'; + return; + } + + /* First determine the (approximate) exponent. */ + e = to_int(std::floor(std::log10(std::abs(x[0])))); + + if (e < -300) { + r *= dd_real(10.0) ^ 300; + r /= dd_real(10.0) ^ (e + 300); + } else if (e > 300) { + r = ldexp(r, -53); + r /= dd_real(10.0) ^ e; + r = ldexp(r, 53); + } else { + r /= dd_real(10.0) ^ e; + } + + /* Fix exponent if we are off by one */ + if (r >= 10.0) { + r /= 10.0; + e++; + } else if (r < 1.0) { + r *= 10.0; + e--; + } + + if (r >= 10.0 || r < 1.0) { + dd_real::error("(dd_real::to_digits): can't compute exponent."); + return; + } + + /* Extract the digits */ + for (i = 0; i < D; i++) { + d = static_cast(r.x[0]); + r -= d; + r *= 10.0; + + s[i] = static_cast(d + '0'); + } + + /* Fix out of range digits. */ + for (i = D-1; i > 0; i--) { + if (s[i] < '0') { + s[i-1]--; + s[i] += 10; + } else if (s[i] > '9') { + s[i-1]++; + s[i] -= 10; + } + } + + if (s[0] <= '0') { + dd_real::error("(dd_real::to_digits): non-positive leading digit."); + return; + } + + /* Round, handle carry */ + if (s[D-1] >= '5') { + s[D-2]++; + + i = D-2; + while (i > 0 && s[i] > '9') { + s[i] -= 10; + s[--i]++; + } + } + + /* If first digit is 10, shift everything. */ + if (s[0] > '9') { + e++; + for (i = precision; i >= 2; i--) s[i] = s[i-1]; + s[0] = '1'; + s[1] = '0'; + } + + s[precision] = 0; + expn = e; +} + +/* Writes the double-double number into the character array s of length len. + The integer d specifies how many significant digits to write. + The string s must be able to hold at least (d+8) characters. + showpos indicates whether to use the + sign, and uppercase indicates + whether the E or e is to be used for the exponent. */ +void dd_real::write(char *s, int len, int precision, + bool showpos, bool uppercase) const { + string str = to_string(precision, 0, ios_base::scientific, showpos, uppercase); + std::strncpy(s, str.c_str(), len-1); + s[len-1] = 0; +} + + +void round_string(char *s, int precision, int *offset){ + /* + Input string must be all digits or errors will occur. + */ + + int i; + int D = precision ; + + /* Round, handle carry */ + if (D>0 && s[D] >= '5') { + s[D-1]++; + + i = D-1; + while (i > 0 && s[i] > '9') { + s[i] -= 10; + s[--i]++; + } + } + + /* If first digit is 10, shift everything. */ + if (s[0] > '9') { + // e++; // don't modify exponent here + for (i = precision; i >= 1; i--) s[i+1] = s[i]; + s[0] = '1'; + s[1] = '0'; + + (*offset)++ ; // now offset needs to be increased by one + precision++ ; + } + + s[precision] = 0; // add terminator for array +} + +string dd_real::to_string(int precision, int width, ios_base::fmtflags fmt, + bool showpos, bool uppercase, char fill) const { + string s; + bool fixed = (fmt & ios_base::fixed) != 0; + bool sgn = true; + int i, e = 0; + + if (isnan()) { + s = uppercase ? "NAN" : "nan"; + sgn = false; + } else { + if (*this < 0.0) + s += '-'; + else if (showpos) + s += '+'; + else + sgn = false; + + if (isinf()) { + s += uppercase ? "INF" : "inf"; + } else if (*this == 0.0) { + /* Zero case */ + s += '0'; + if (precision > 0) { + s += '.'; + s.append(precision, '0'); + } + } else { + /* Non-zero case */ + int off = (fixed ? (1 + to_int(floor(log10(abs(*this))))) : 1); + int d = precision + off; + + int d_with_extra = d; + if(fixed) + d_with_extra = std::max(60, d); // longer than the max accuracy for DD + + // highly special case - fixed mode, precision is zero, abs(*this) < 1.0 + // without this trap a number like 0.9 printed fixed with 0 precision prints as 0 + // should be rounded to 1. + if(fixed && (precision == 0) && (abs(*this) < 1.0)){ + if(abs(*this) >= 0.5) + s += '1'; + else + s += '0'; + + return s; + } + + // handle near zero to working precision (but not exactly zero) + if (fixed && d <= 0) { + s += '0'; + if (precision > 0) { + s += '.'; + s.append(precision, '0'); + } + } else { // default + + char *t; // = new char[d+1]; + int j; + + if(fixed){ + t = new char[d_with_extra+1]; + to_digits(t, e, d_with_extra); + } + else{ + t = new char[d+1]; + to_digits(t, e, d); + } + + off = e + 1; + + if (fixed) { + // fix the string if it's been computed incorrectly + // round here in the decimal string if required + round_string(t, d, &off); + + if (off > 0) { + for (i = 0; i < off; i++) s += t[i]; + if (precision > 0) { + s += '.'; + for (j = 0; j < precision; j++, i++) s += t[i]; + } + } else { + s += "0."; + if (off < 0) s.append(-off, '0'); + for (i = 0; i < d; i++) s += t[i]; + } + } else { + s += t[0]; + if (precision > 0) s += '.'; + + for (i = 1; i <= precision; i++) + s += t[i]; + + } + delete [] t; + } + } + + // trap for improper offset with large values + // without this trap, output of values of the for 10^j - 1 fail for j > 28 + // and are output with the point in the wrong place, leading to a dramatically off value + if(fixed && (precision > 0)){ + // make sure that the value isn't dramatically larger + double from_string = atof(s.c_str()); + + // if this ratio is large, then we've got problems + if( fabs( from_string / this->x[0] ) > 3.0 ){ + + // loop on the string, find the point, move it up one + // don't act on the first character + for(i=1; i < (int)s.length(); i++){ + if(s[i] == '.'){ + s[i] = s[i-1] ; + s[i-1] = '.' ; + break; + } + } + + from_string = atof(s.c_str()); + // if this ratio is large, then the string has not been fixed + if( fabs( from_string / this->x[0] ) > 3.0 ){ + dd_real::error("Re-rounding unsuccessful in large number fixed point trap.") ; + } + } + } + + + if (!fixed && !isinf()) { + /* Fill in exponent part */ + s += uppercase ? 'E' : 'e'; + append_expn(s, e); + } + } + + /* Fill in the blanks */ + int len = s.length(); + if (len < width) { + int delta = width - len; + if (fmt & ios_base::internal) { + if (sgn) + s.insert(static_cast(1), delta, fill); + else + s.insert(static_cast(0), delta, fill); + } else if (fmt & ios_base::left) { + s.append(delta, fill); + } else { + s.insert(static_cast(0), delta, fill); + } + } + + return s; +} + +/* Reads in a double-double number from the string s. */ +int dd_real::read(const char *s, dd_real &a) { + const char *p = s; + char ch; + int sign = 0; + int point = -1; + int nd = 0; + int e = 0; + bool done = false; + dd_real r = 0.0; + int nread; + + /* Skip any leading spaces */ + while (*p == ' ') + p++; + + while (!done && (ch = *p) != '\0') { + if (ch >= '0' && ch <= '9') { + int d = ch - '0'; + r *= 10.0; + r += static_cast(d); + nd++; + } else { + + switch (ch) { + + case '.': + if (point >= 0) + return -1; + point = nd; + break; + + case '-': + case '+': + if (sign != 0 || nd > 0) + return -1; + sign = (ch == '-') ? -1 : 1; + break; + + case 'E': + case 'e': + nread = std::sscanf(p+1, "%d", &e); + done = true; + if (nread != 1) + return -1; + break; + + default: + return -1; + } + } + + p++; + } + + if (point >= 0) { + e -= (nd - point); + } + + if (e != 0) { + r *= (dd_real(10.0) ^ e); + } + + a = (sign == -1) ? -r : r; + return 0; +} + +/* Debugging routines */ +void dd_real::dump(const string &name, std::ostream &os) const { + std::ios_base::fmtflags old_flags = os.flags(); + std::streamsize old_prec = os.precision(19); + os << std::scientific; + + if (name.length() > 0) os << name << " = "; + os << "[ " << setw(27) << x[0] << ", " << setw(27) << x[1] << " ]" << endl; + + os.precision(old_prec); + os.flags(old_flags); +} + +void dd_real::dump_bits(const string &name, std::ostream &os) const { + string::size_type len = name.length(); + if (len > 0) { + os << name << " = "; + len +=3; + } + os << "[ "; + len += 2; + print_double_info(os, x[0]); + os << endl; + for (string::size_type i = 0; i < len; i++) os << ' '; + print_double_info(os, x[1]); + os << " ]" << endl; +} + +dd_real dd_real::debug_rand() { + + if (std::rand() % 2 == 0) + return ddrand(); + + int expn = 0; + dd_real a = 0.0; + double d; + for (int i = 0; i < 2; i++) { + d = std::ldexp(static_cast(std::rand()) / RAND_MAX, -expn); + a += d; + expn = expn + 54 + std::rand() % 200; + } + return a; +} diff --git a/src/external/PackedCSparse/qd/dd_real.h b/src/external/PackedCSparse/qd/dd_real.h new file mode 100644 index 00000000..e16438aa --- /dev/null +++ b/src/external/PackedCSparse/qd/dd_real.h @@ -0,0 +1,293 @@ +/* + * include/dd_real.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2007 + * + * Double-double precision (>= 106-bit significand) floating point + * arithmetic package based on David Bailey's Fortran-90 double-double + * package, with some changes. See + * + * http://www.nersc.gov/~dhbailey/mpdist/mpdist.html + * + * for the original Fortran-90 version. + * + * Overall structure is similar to that of Keith Brigg's C++ double-double + * package. See + * + * http://www-epidem.plansci.cam.ac.uk/~kbriggs/doubledouble.html + * + * for more details. In particular, the fix for x86 computers is borrowed + * from his code. + * + * Yozo Hida + */ + +#ifndef _QD_DD_REAL_H +#define _QD_DD_REAL_H + +#include +#include +#include +#include +#include "qd_config.h" +#include "fpu.h" + +// Some compilers define isnan, isfinite, and isinf as macros, even for +// C++ codes, which cause havoc when overloading these functions. We undef +// them here. +#ifdef isnan +#undef isnan +#endif + +#ifdef isfinite +#undef isfinite +#endif + +#ifdef isinf +#undef isinf +#endif + +#ifdef max +#undef max +#endif + +#ifdef min +#undef min +#endif + +struct QD_API dd_real { + double x[2]; + + dd_real(double hi, double lo) { x[0] = hi; x[1] = lo; } + dd_real() {x[0] = 0.0; x[1] = 0.0; } + dd_real(double h) { x[0] = h; x[1] = 0.0; } + dd_real(int h) { + x[0] = (static_cast(h)); + x[1] = 0.0; + } + + dd_real (const char *s); + explicit dd_real (const double *d) { + x[0] = d[0]; x[1] = d[1]; + } + + static void error(const char *msg); + + double _hi() const { return x[0]; } + double _lo() const { return x[1]; } + + static const dd_real _2pi; + static const dd_real _pi; + static const dd_real _3pi4; + static const dd_real _pi2; + static const dd_real _pi4; + static const dd_real _e; + static const dd_real _log2; + static const dd_real _log10; + static const dd_real _nan; + static const dd_real _inf; + + static const double _eps; + static const double _min_normalized; + static const dd_real _max; + static const dd_real _safe_max; + static const int _ndigits; + + bool isnan() const { return QD_ISNAN(x[0]) || QD_ISNAN(x[1]); } + bool isfinite() const { return QD_ISFINITE(x[0]); } + bool isinf() const { return QD_ISINF(x[0]); } + + static dd_real add(double a, double b); + static dd_real ieee_add(const dd_real &a, const dd_real &b); + static dd_real sloppy_add(const dd_real &a, const dd_real &b); + + dd_real &operator+=(double a); + dd_real &operator+=(const dd_real &a); + + static dd_real sub(double a, double b); + + dd_real &operator-=(double a); + dd_real &operator-=(const dd_real &a); + + dd_real operator-() const; + + static dd_real mul(double a, double b); + + dd_real &operator*=(double a); + dd_real &operator*=(const dd_real &a); + + static dd_real div(double a, double b); + static dd_real sloppy_div(const dd_real &a, const dd_real &b); + static dd_real accurate_div(const dd_real &a, const dd_real &b); + + dd_real &operator/=(double a); + dd_real &operator/=(const dd_real &a); + + dd_real &operator=(double a); + dd_real &operator=(const char *s); + + dd_real operator^(int n); + static dd_real sqr(double d); + + static dd_real sqrt(double a); + + bool is_zero() const; + bool is_one() const; + bool is_positive() const; + bool is_negative() const; + + explicit operator bool() const; // new + explicit operator double() const; // new + + static dd_real rand(void); + + void to_digits(char *s, int &expn, int precision = _ndigits) const; + void write(char *s, int len, int precision = _ndigits, + bool showpos = false, bool uppercase = false) const; + std::string to_string(int precision = _ndigits, int width = 0, + std::ios_base::fmtflags fmt = static_cast(0), + bool showpos = false, bool uppercase = false, char fill = ' ') const; + int read(const char *s, dd_real &a); + + /* Debugging Methods */ + void dump(const std::string &name, std::ostream &os = std::cerr) const; + void dump_bits(const std::string &name, + std::ostream &os = std::cerr) const; + + static dd_real debug_rand(); +}; + + +namespace std { + template <> + class numeric_limits : public numeric_limits { + public: + inline static double epsilon() { return dd_real::_eps; } + inline static dd_real max() { return dd_real::_max; } + inline static dd_real safe_max() { return dd_real::_safe_max; } + inline static double min() { return dd_real::_min_normalized; } + static const int digits = 104; + static const int digits10 = 31; + }; +} + +QD_API dd_real ddrand(void); +QD_API dd_real sqrt(const dd_real &a); + +QD_API dd_real polyeval(const dd_real *c, int n, const dd_real &x); +QD_API dd_real polyroot(const dd_real *c, int n, + const dd_real &x0, int max_iter = 32, double thresh = 0.0); + +QD_API inline bool isnan(const dd_real &a) { return a.isnan(); } +QD_API inline bool isfinite(const dd_real &a) { return a.isfinite(); } +QD_API inline bool isinf(const dd_real &a) { return a.isinf(); } + +/* Computes dd * d where d is known to be a power of 2. */ +QD_API dd_real mul_pwr2(const dd_real &dd, double d); + +QD_API dd_real operator+(const dd_real &a, double b); +QD_API dd_real operator+(double a, const dd_real &b); +QD_API dd_real operator+(const dd_real &a, const dd_real &b); + +QD_API dd_real operator-(const dd_real &a, double b); +QD_API dd_real operator-(double a, const dd_real &b); +QD_API dd_real operator-(const dd_real &a, const dd_real &b); + +QD_API dd_real operator*(const dd_real &a, double b); +QD_API dd_real operator*(double a, const dd_real &b); +QD_API dd_real operator*(const dd_real &a, const dd_real &b); + +QD_API dd_real operator/(const dd_real &a, double b); +QD_API dd_real operator/(double a, const dd_real &b); +QD_API dd_real operator/(const dd_real &a, const dd_real &b); + +QD_API dd_real inv(const dd_real &a); + +QD_API dd_real rem(const dd_real &a, const dd_real &b); +QD_API dd_real drem(const dd_real &a, const dd_real &b); +QD_API dd_real divrem(const dd_real &a, const dd_real &b, dd_real &r); + +QD_API dd_real pow(const dd_real &a, int n); +QD_API dd_real pow(const dd_real &a, const dd_real &b); +QD_API dd_real npwr(const dd_real &a, int n); +QD_API dd_real sqr(const dd_real &a); + +QD_API dd_real sqrt(const dd_real &a); +QD_API dd_real nroot(const dd_real &a, int n); + +QD_API bool operator==(const dd_real &a, double b); +QD_API bool operator==(double a, const dd_real &b); +QD_API bool operator==(const dd_real &a, const dd_real &b); + +QD_API bool operator<=(const dd_real &a, double b); +QD_API bool operator<=(double a, const dd_real &b); +QD_API bool operator<=(const dd_real &a, const dd_real &b); + +QD_API bool operator>=(const dd_real &a, double b); +QD_API bool operator>=(double a, const dd_real &b); +QD_API bool operator>=(const dd_real &a, const dd_real &b); + +QD_API bool operator<(const dd_real &a, double b); +QD_API bool operator<(double a, const dd_real &b); +QD_API bool operator<(const dd_real &a, const dd_real &b); + +QD_API bool operator>(const dd_real &a, double b); +QD_API bool operator>(double a, const dd_real &b); +QD_API bool operator>(const dd_real &a, const dd_real &b); + +QD_API bool operator!=(const dd_real &a, double b); +QD_API bool operator!=(double a, const dd_real &b); +QD_API bool operator!=(const dd_real &a, const dd_real &b); + +QD_API dd_real nint(const dd_real &a); +QD_API dd_real floor(const dd_real &a); +QD_API dd_real ceil(const dd_real &a); +QD_API dd_real aint(const dd_real &a); + +QD_API dd_real ddrand(void); + +double to_double(const dd_real &a); +int to_int(const dd_real &a); + +QD_API dd_real exp(const dd_real &a); +QD_API dd_real ldexp(const dd_real &a, int exp); +QD_API dd_real log(const dd_real &a); +QD_API dd_real log10(const dd_real &a); + +QD_API dd_real sin(const dd_real &a); +QD_API dd_real cos(const dd_real &a); +QD_API dd_real tan(const dd_real &a); +QD_API void sincos(const dd_real &a, dd_real &sin_a, dd_real &cos_a); + +QD_API dd_real asin(const dd_real &a); +QD_API dd_real acos(const dd_real &a); +QD_API dd_real atan(const dd_real &a); +QD_API dd_real atan2(const dd_real &y, const dd_real &x); + +QD_API dd_real sinh(const dd_real &a); +QD_API dd_real cosh(const dd_real &a); +QD_API dd_real tanh(const dd_real &a); +QD_API void sincosh(const dd_real &a, + dd_real &sinh_a, dd_real &cosh_a); + +QD_API dd_real asinh(const dd_real &a); +QD_API dd_real acosh(const dd_real &a); +QD_API dd_real atanh(const dd_real &a); + +QD_API dd_real fabs(const dd_real &a); +QD_API dd_real abs(const dd_real &a); /* same as fabs */ + +QD_API dd_real fmod(const dd_real &a, const dd_real &b); + +QD_API std::ostream& operator<<(std::ostream &s, const dd_real &a); +QD_API std::istream& operator>>(std::istream &s, dd_real &a); +#ifdef QD_INLINE +#include "dd_inline.h" +#endif + +#endif /* _QD_DD_REAL_H */ + diff --git a/src/external/PackedCSparse/qd/fpu.cc b/src/external/PackedCSparse/qd/fpu.cc new file mode 100644 index 00000000..96ddc488 --- /dev/null +++ b/src/external/PackedCSparse/qd/fpu.cc @@ -0,0 +1,124 @@ +/* + * src/fpu.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains functions to set and restore the round-to-double flag in the + * control word of a x86 FPU. + */ + +#include "qd_config.h" +#include "fpu.h" + +#ifdef X86 +#ifdef _WIN32 +#include +#else + +#ifdef HAVE_FPU_CONTROL_H +#include +#endif + +#ifndef _FPU_GETCW +#define _FPU_GETCW(x) asm volatile ("fnstcw %0":"=m" (x)); +#endif + +#ifndef _FPU_SETCW +#define _FPU_SETCW(x) asm volatile ("fldcw %0": :"m" (x)); +#endif + +#ifndef _FPU_EXTENDED +#define _FPU_EXTENDED 0x0300 +#endif + +#ifndef _FPU_DOUBLE +#define _FPU_DOUBLE 0x0200 +#endif + +#endif +#endif /* X86 */ + +extern "C" { + +void fpu_fix_start(unsigned int *old_cw) { +#ifdef X86 +#ifdef _WIN32 +#ifdef __BORLANDC__ + /* Win 32 Borland C */ + unsigned short cw = _control87(0, 0); + _control87(0x0200, 0x0300); + if (old_cw) { + *old_cw = cw; + } +#else + /* Win 32 MSVC */ + unsigned int cw = _control87(0, 0); + _control87(0x00010000, 0x00030000); + if (old_cw) { + *old_cw = cw; + } +#endif +#else + /* Linux */ + volatile unsigned short cw, new_cw; + _FPU_GETCW(cw); + + new_cw = (cw & ~_FPU_EXTENDED) | _FPU_DOUBLE; + _FPU_SETCW(new_cw); + + if (old_cw) { + *old_cw = cw; + } +#endif +#endif +} + +void fpu_fix_end(unsigned int *old_cw) { +#ifdef X86 +#ifdef _WIN32 + +#ifdef __BORLANDC__ + /* Win 32 Borland C */ + if (old_cw) { + unsigned short cw = (unsigned short) *old_cw; + _control87(cw, 0xFFFF); + } +#else + /* Win 32 MSVC */ + if (old_cw) { + _control87(*old_cw, 0xFFFFFFFF); + } +#endif + +#else + /* Linux */ + if (old_cw) { + int cw; + cw = *old_cw; + _FPU_SETCW(cw); + } +#endif +#endif +} + +#ifdef HAVE_FORTRAN + +#define f_fpu_fix_start FC_FUNC_(f_fpu_fix_start, F_FPU_FIX_START) +#define f_fpu_fix_end FC_FUNC_(f_fpu_fix_end, F_FPU_FIX_END) + +void f_fpu_fix_start(unsigned int *old_cw) { + fpu_fix_start(old_cw); +} + +void f_fpu_fix_end(unsigned int *old_cw) { + fpu_fix_end(old_cw); +} + +#endif + +} + diff --git a/src/external/PackedCSparse/qd/fpu.h b/src/external/PackedCSparse/qd/fpu.h new file mode 100644 index 00000000..35eab18c --- /dev/null +++ b/src/external/PackedCSparse/qd/fpu.h @@ -0,0 +1,39 @@ +/* + * include/fpu.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2001 + * + * Contains functions to set and restore the round-to-double flag in the + * control word of a x86 FPU. The algorithms in the double-double and + * quad-double package does not function with the extended mode found in + * these FPU. + */ +#ifndef _QD_FPU_H +#define _QD_FPU_H + +#include "qd_config.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/* + * Set the round-to-double flag, and save the old control word in old_cw. + * If old_cw is NULL, the old control word is not saved. + */ +QD_API void fpu_fix_start(unsigned int *old_cw); + +/* + * Restore the control word. + */ +QD_API void fpu_fix_end(unsigned int *old_cw); + +#ifdef __cplusplus +} +#endif + +#endif /* _QD_FPU_H */ diff --git a/src/external/PackedCSparse/qd/inline.h b/src/external/PackedCSparse/qd/inline.h new file mode 100644 index 00000000..52425545 --- /dev/null +++ b/src/external/PackedCSparse/qd/inline.h @@ -0,0 +1,143 @@ +/* + * include/inline.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * This file contains the basic functions used both by double-double + * and quad-double package. These are declared as inline functions as + * they are the smallest building blocks of the double-double and + * quad-double arithmetic. + */ +#ifndef _QD_INLINE_H +#define _QD_INLINE_H + +#define _QD_SPLITTER 134217729.0 // = 2^27 + 1 +#define _QD_SPLIT_THRESH 6.69692879491417e+299 // = 2^996 + +#ifdef QD_VACPP_BUILTINS_H +/* For VisualAge C++ __fmadd */ +#include +#endif + +#include +#include + +namespace qd { + +static const double _d_nan = std::numeric_limits::quiet_NaN(); +static const double _d_inf = std::numeric_limits::infinity(); + +/*********** Basic Functions ************/ +/* Computes fl(a+b) and err(a+b). Assumes |a| >= |b|. */ +inline double quick_two_sum(double a, double b, double &err) { + double s = a + b; + err = b - (s - a); + return s; +} + +/* Computes fl(a-b) and err(a-b). Assumes |a| >= |b| */ +inline double quick_two_diff(double a, double b, double &err) { + double s = a - b; + err = (a - s) - b; + return s; +} + +/* Computes fl(a+b) and err(a+b). */ +inline double two_sum(double a, double b, double &err) { + double s = a + b; + double bb = s - a; + err = (a - (s - bb)) + (b - bb); + return s; +} + +/* Computes fl(a-b) and err(a-b). */ +inline double two_diff(double a, double b, double &err) { + double s = a - b; + double bb = s - a; + err = (a - (s - bb)) - (b + bb); + return s; +} + +#ifndef QD_FMS +/* Computes high word and lo word of a */ +inline void split(double a, double &hi, double &lo) { + double temp; + if (a > _QD_SPLIT_THRESH || a < -_QD_SPLIT_THRESH) { + a *= 3.7252902984619140625e-09; // 2^-28 + temp = _QD_SPLITTER * a; + hi = temp - (temp - a); + lo = a - hi; + hi *= 268435456.0; // 2^28 + lo *= 268435456.0; // 2^28 + } else { + temp = _QD_SPLITTER * a; + hi = temp - (temp - a); + lo = a - hi; + } +} +#endif + +/* Computes fl(a*b) and err(a*b). */ +inline double two_prod(double a, double b, double &err) { +#ifdef QD_FMS + double p = a * b; + err = QD_FMS(a, b, p); + return p; +#else + double a_hi, a_lo, b_hi, b_lo; + double p = a * b; + split(a, a_hi, a_lo); + split(b, b_hi, b_lo); + err = ((a_hi * b_hi - p) + a_hi * b_lo + a_lo * b_hi) + a_lo * b_lo; + return p; +#endif +} + +/* Computes fl(a*a) and err(a*a). Faster than the above method. */ +inline double two_sqr(double a, double &err) { +#ifdef QD_FMS + double p = a * a; + err = QD_FMS(a, a, p); + return p; +#else + double hi, lo; + double q = a * a; + split(a, hi, lo); + err = ((hi * hi - q) + 2.0 * hi * lo) + lo * lo; + return q; +#endif +} + +/* Computes the nearest integer to d. */ +inline double nint(double d) { + if (d == std::floor(d)) + return d; + return std::floor(d + 0.5); +} + +/* Computes the truncated integer. */ +inline double aint(double d) { + return (d >= 0.0) ? std::floor(d) : std::ceil(d); +} + +/* These are provided to give consistent + interface for double with double-double and quad-double. */ +inline void sincosh(double t, double &sinh_t, double &cosh_t) { + sinh_t = std::sinh(t); + cosh_t = std::cosh(t); +} + +inline double sqr(double t) { + return t * t; +} + +inline double to_double(double a) { return a; } +inline int to_int(double a) { return static_cast(a); } + +} + +#endif /* _QD_INLINE_H */ diff --git a/src/external/PackedCSparse/qd/qd.pdf b/src/external/PackedCSparse/qd/qd.pdf new file mode 100644 index 0000000000000000000000000000000000000000..9525b8cac490f4c3229bc7074e37075cebe7eae0 GIT binary patch literal 194502 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 SmE72tYykD$gkxvSUpTr@w^W=dn^OhAde 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 +#include + +/* include/qd/qd_config.h. Generated from qd_config.h.in by configure. */ +#ifndef _QD_QD_CONFIG_H +#define _QD_QD_CONFIG_H 1 + +#ifndef QD_API +#define QD_API /**/ +#endif + +/* Set to 1 if using VisualAge C++ compiler for __fmadd builtin. */ +#ifndef QD_VACPP_BUILTINS_H +/* #undef QD_VACPP_BUILTINS_H */ +#endif + +/* If fused multiply-add is available, define to correct macro for + using it. It is invoked as QD_FMA(a, b, c) to compute fl(a * b + c). + If correctly rounded multiply-add is not available (or if unsure), + keep it undefined.*/ +#ifndef QD_FMA +/* #undef QD_FMA */ +#endif + +/* If fused multiply-subtract is available, define to correct macro for + using it. It is invoked as QD_FMS(a, b, c) to compute fl(a * b - c). + If correctly rounded multiply-add is not available (or if unsure), + keep it undefined.*/ +#ifndef QD_FMS +#define QD_FMS(a, b, c) std::fma(a,b,-c) +/* #undef QD_FMS */ +#endif + +/* Set the following to 1 to define commonly used function + to be inlined. This should be set to 1 unless the compiler + does not support the "inline" keyword, or if building for + debugging purposes. */ +#ifndef QD_INLINE +#define QD_INLINE 1 +#endif + +/* Set the following to 1 to use ANSI C++ standard header files + such as cmath, iostream, etc. If set to zero, it will try to + include math.h, iostream.h, etc, instead. */ +#ifndef QD_HAVE_STD +#define QD_HAVE_STD 1 +#endif + +/* Set the following to 1 to make the addition and subtraction + to satisfy the IEEE-style error bound + + fl(a + b) = (1 + d) * (a + b) + + where |d| <= eps. If set to 0, the addition and subtraction + will satisfy the weaker Cray-style error bound + + fl(a + b) = (1 + d1) * a + (1 + d2) * b + + where |d1| <= eps and |d2| eps. */ +#ifndef QD_IEEE_ADD +/* #undef QD_IEEE_ADD */ +#endif + +/* Set the following to 1 to use slightly inaccurate but faster + version of multiplication. */ +#ifndef QD_SLOPPY_MUL +#define QD_SLOPPY_MUL 1 +#endif + +/* Set the following to 1 to use slightly inaccurate but faster + version of division. */ +#ifndef QD_SLOPPY_DIV +#define QD_SLOPPY_DIV 1 +#endif + +/* Define this macro to be the isfinite(x) function. */ +#ifndef QD_ISFINITE +#define QD_ISFINITE(x) std::isfinite(x) +#endif + +/* Define this macro to be the isinf(x) function. */ +#ifndef QD_ISINF +#define QD_ISINF(x) std::isinf(x) +#endif + +/* Define this macro to be the isnan(x) function. */ +#ifndef QD_ISNAN +#define QD_ISNAN(x) std::isnan(x) +#endif + + +#endif /* _QD_QD_CONFIG_H */ diff --git a/src/external/PackedCSparse/qd/qd_const.cc b/src/external/PackedCSparse/qd/qd_const.cc new file mode 100644 index 00000000..6f4e01d2 --- /dev/null +++ b/src/external/PackedCSparse/qd/qd_const.cc @@ -0,0 +1,62 @@ +/* + * src/qd_const.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Defines constants used in quad-double package. + */ +#include "qd_config.h" +#include "qd_real.h" + +/* Some useful constants. */ +const qd_real qd_real::_2pi = qd_real(6.283185307179586232e+00, + 2.449293598294706414e-16, + -5.989539619436679332e-33, + 2.224908441726730563e-49); +const qd_real qd_real::_pi = qd_real(3.141592653589793116e+00, + 1.224646799147353207e-16, + -2.994769809718339666e-33, + 1.112454220863365282e-49); +const qd_real qd_real::_pi2 = qd_real(1.570796326794896558e+00, + 6.123233995736766036e-17, + -1.497384904859169833e-33, + 5.562271104316826408e-50); +const qd_real qd_real::_pi4 = qd_real(7.853981633974482790e-01, + 3.061616997868383018e-17, + -7.486924524295849165e-34, + 2.781135552158413204e-50); +const qd_real qd_real::_3pi4 = qd_real(2.356194490192344837e+00, + 9.1848509936051484375e-17, + 3.9168984647504003225e-33, + -2.5867981632704860386e-49); +const qd_real qd_real::_e = qd_real(2.718281828459045091e+00, + 1.445646891729250158e-16, + -2.127717108038176765e-33, + 1.515630159841218954e-49); +const qd_real qd_real::_log2 = qd_real(6.931471805599452862e-01, + 2.319046813846299558e-17, + 5.707708438416212066e-34, + -3.582432210601811423e-50); +const qd_real qd_real::_log10 = qd_real(2.302585092994045901e+00, + -2.170756223382249351e-16, + -9.984262454465776570e-33, + -4.023357454450206379e-49); +const qd_real qd_real::_nan = qd_real(qd::_d_nan, qd::_d_nan, + qd::_d_nan, qd::_d_nan); +const qd_real qd_real::_inf = qd_real(qd::_d_inf, qd::_d_inf, + qd::_d_inf, qd::_d_inf); + +const double qd_real::_eps = 1.21543267145725e-63; // = 2^-209 +const double qd_real::_min_normalized = 1.6259745436952323e-260; // = 2^(-1022 + 3*53) +const qd_real qd_real::_max = qd_real( + 1.79769313486231570815e+308, 9.97920154767359795037e+291, + 5.53956966280111259858e+275, 3.07507889307840487279e+259); +const qd_real qd_real::_safe_max = qd_real( + 1.7976931080746007281e+308, 9.97920154767359795037e+291, + 5.53956966280111259858e+275, 3.07507889307840487279e+259); +const int qd_real::_ndigits = 62; + diff --git a/src/external/PackedCSparse/qd/qd_inline.h b/src/external/PackedCSparse/qd/qd_inline.h new file mode 100644 index 00000000..89ba275b --- /dev/null +++ b/src/external/PackedCSparse/qd/qd_inline.h @@ -0,0 +1,1047 @@ +/* + * include/qd_inline.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2001 + * + * Contains small functions (suitable for inlining) in the quad-double + * arithmetic package. + */ +#ifndef _QD_QD_INLINE_H +#define _QD_QD_INLINE_H + +#include +#include "inline.h" + +#ifndef QD_INLINE +#define inline +#endif + +/********** Constructors **********/ +inline qd_real::qd_real(double x0, double x1, double x2, double x3) { + x[0] = x0; + x[1] = x1; + x[2] = x2; + x[3] = x3; +} + +inline qd_real::qd_real(const double *xx) { + x[0] = xx[0]; + x[1] = xx[1]; + x[2] = xx[2]; + x[3] = xx[3]; +} + +inline qd_real::qd_real(double x0) { + x[0] = x0; + x[1] = x[2] = x[3] = 0.0; +} + +inline qd_real::qd_real() { + x[0] = 0.0; + x[1] = 0.0; + x[2] = 0.0; + x[3] = 0.0; +} + +inline qd_real::qd_real(const dd_real &a) { + x[0] = a._hi(); + x[1] = a._lo(); + x[2] = x[3] = 0.0; +} + +inline qd_real::qd_real(int i) { + x[0] = static_cast(i); + x[1] = x[2] = x[3] = 0.0; +} + +/********** Accessors **********/ +inline double qd_real::operator[](int i) const { + return x[i]; +} + +inline double &qd_real::operator[](int i) { + return x[i]; +} + +inline bool qd_real::isnan() const { + return QD_ISNAN(x[0]) || QD_ISNAN(x[1]) || QD_ISNAN(x[2]) || QD_ISNAN(x[3]); +} + +/********** Renormalization **********/ +namespace qd { +inline void quick_renorm(double &c0, double &c1, + double &c2, double &c3, double &c4) { + double t0, t1, t2, t3; + double s; + s = qd::quick_two_sum(c3, c4, t3); + s = qd::quick_two_sum(c2, s , t2); + s = qd::quick_two_sum(c1, s , t1); + c0 = qd::quick_two_sum(c0, s , t0); + + s = qd::quick_two_sum(t2, t3, t2); + s = qd::quick_two_sum(t1, s , t1); + c1 = qd::quick_two_sum(t0, s , t0); + + s = qd::quick_two_sum(t1, t2, t1); + c2 = qd::quick_two_sum(t0, s , t0); + + c3 = t0 + t1; +} + +inline void renorm(double &c0, double &c1, + double &c2, double &c3) { + double s0, s1, s2 = 0.0, s3 = 0.0; + + if (QD_ISINF(c0)) return; + + s0 = qd::quick_two_sum(c2, c3, c3); + s0 = qd::quick_two_sum(c1, s0, c2); + c0 = qd::quick_two_sum(c0, s0, c1); + + s0 = c0; + s1 = c1; + if (s1 != 0.0) { + s1 = qd::quick_two_sum(s1, c2, s2); + if (s2 != 0.0) + s2 = qd::quick_two_sum(s2, c3, s3); + else + s1 = qd::quick_two_sum(s1, c3, s2); + } else { + s0 = qd::quick_two_sum(s0, c2, s1); + if (s1 != 0.0) + s1 = qd::quick_two_sum(s1, c3, s2); + else + s0 = qd::quick_two_sum(s0, c3, s1); + } + + c0 = s0; + c1 = s1; + c2 = s2; + c3 = s3; +} + +inline void renorm(double &c0, double &c1, + double &c2, double &c3, double &c4) { + double s0, s1, s2 = 0.0, s3 = 0.0; + + if (QD_ISINF(c0)) return; + + s0 = qd::quick_two_sum(c3, c4, c4); + s0 = qd::quick_two_sum(c2, s0, c3); + s0 = qd::quick_two_sum(c1, s0, c2); + c0 = qd::quick_two_sum(c0, s0, c1); + + s0 = c0; + s1 = c1; + + if (s1 != 0.0) { + s1 = qd::quick_two_sum(s1, c2, s2); + if (s2 != 0.0) { + s2 = qd::quick_two_sum(s2, c3, s3); + if (s3 != 0.0) + s3 += c4; + else + s2 = qd::quick_two_sum(s2, c4, s3); + } else { + s1 = qd::quick_two_sum(s1, c3, s2); + if (s2 != 0.0) + s2 = qd::quick_two_sum(s2, c4, s3); + else + s1 = qd::quick_two_sum(s1, c4, s2); + } + } else { + s0 = qd::quick_two_sum(s0, c2, s1); + if (s1 != 0.0) { + s1 = qd::quick_two_sum(s1, c3, s2); + if (s2 != 0.0) + s2 = qd::quick_two_sum(s2, c4, s3); + else + s1 = qd::quick_two_sum(s1, c4, s2); + } else { + s0 = qd::quick_two_sum(s0, c3, s1); + if (s1 != 0.0) + s1 = qd::quick_two_sum(s1, c4, s2); + else + s0 = qd::quick_two_sum(s0, c4, s1); + } + } + + c0 = s0; + c1 = s1; + c2 = s2; + c3 = s3; +} +} + +inline void qd_real::renorm() { + qd::renorm(x[0], x[1], x[2], x[3]); +} + +inline void qd_real::renorm(double &e) { + qd::renorm(x[0], x[1], x[2], x[3], e); +} + + +/********** Additions ************/ +namespace qd { + +inline void three_sum(double &a, double &b, double &c) { + double t1, t2, t3; + t1 = qd::two_sum(a, b, t2); + a = qd::two_sum(c, t1, t3); + b = qd::two_sum(t2, t3, c); +} + +inline void three_sum2(double &a, double &b, double &c) { + double t1, t2, t3; + t1 = qd::two_sum(a, b, t2); + a = qd::two_sum(c, t1, t3); + b = t2 + t3; +} + +} + +/* quad-double + double */ +inline qd_real operator+(const qd_real &a, double b) { + double c0, c1, c2, c3; + double e; + + c0 = qd::two_sum(a[0], b, e); + c1 = qd::two_sum(a[1], e, e); + c2 = qd::two_sum(a[2], e, e); + c3 = qd::two_sum(a[3], e, e); + + qd::renorm(c0, c1, c2, c3, e); + + return qd_real(c0, c1, c2, c3); +} + +/* quad-double + double-double */ +inline qd_real operator+(const qd_real &a, const dd_real &b) { + + double s0, s1, s2, s3; + double t0, t1; + + s0 = qd::two_sum(a[0], b._hi(), t0); + s1 = qd::two_sum(a[1], b._lo(), t1); + + s1 = qd::two_sum(s1, t0, t0); + + s2 = a[2]; + qd::three_sum(s2, t0, t1); + + s3 = qd::two_sum(t0, a[3], t0); + t0 += t1; + + qd::renorm(s0, s1, s2, s3, t0); + return qd_real(s0, s1, s2, s3); +} + + +/* double + quad-double */ +inline qd_real operator+(double a, const qd_real &b) { + return (b + a); +} + +/* double-double + quad-double */ +inline qd_real operator+(const dd_real &a, const qd_real &b) { + return (b + a); +} + +namespace qd { + +/* s = quick_three_accum(a, b, c) adds c to the dd-pair (a, b). + * If the result does not fit in two doubles, then the sum is + * output into s and (a,b) contains the remainder. Otherwise + * s is zero and (a,b) contains the sum. */ +inline double quick_three_accum(double &a, double &b, double c) { + double s; + bool za, zb; + + s = qd::two_sum(b, c, b); + s = qd::two_sum(a, s, a); + + za = (a != 0.0); + zb = (b != 0.0); + + if (za && zb) + return s; + + if (!zb) { + b = a; + a = s; + } else { + a = s; + } + + return 0.0; +} + +} + +inline qd_real qd_real::ieee_add(const qd_real &a, const qd_real &b) { + int i, j, k; + double s, t; + double u, v; /* double-length accumulator */ + double x[4] = {0.0, 0.0, 0.0, 0.0}; + + i = j = k = 0; + if (std::abs(a[i]) > std::abs(b[j])) + u = a[i++]; + else + u = b[j++]; + if (std::abs(a[i]) > std::abs(b[j])) + v = a[i++]; + else + v = b[j++]; + + u = qd::quick_two_sum(u, v, v); + + while (k < 4) { + if (i >= 4 && j >= 4) { + x[k] = u; + if (k < 3) + x[++k] = v; + break; + } + + if (i >= 4) + t = b[j++]; + else if (j >= 4) + t = a[i++]; + else if (std::abs(a[i]) > std::abs(b[j])) { + t = a[i++]; + } else + t = b[j++]; + + s = qd::quick_three_accum(u, v, t); + + if (s != 0.0) { + x[k++] = s; + } + } + + /* add the rest. */ + for (k = i; k < 4; k++) + x[3] += a[k]; + for (k = j; k < 4; k++) + x[3] += b[k]; + + qd::renorm(x[0], x[1], x[2], x[3]); + return qd_real(x[0], x[1], x[2], x[3]); +} + +inline qd_real qd_real::sloppy_add(const qd_real &a, const qd_real &b) { + /* + double s0, s1, s2, s3; + double t0, t1, t2, t3; + + s0 = qd::two_sum(a[0], b[0], t0); + s1 = qd::two_sum(a[1], b[1], t1); + s2 = qd::two_sum(a[2], b[2], t2); + s3 = qd::two_sum(a[3], b[3], t3); + + s1 = qd::two_sum(s1, t0, t0); + qd::three_sum(s2, t0, t1); + qd::three_sum2(s3, t0, t2); + t0 = t0 + t1 + t3; + + qd::renorm(s0, s1, s2, s3, t0); + return qd_real(s0, s1, s2, s3, t0); + */ + + /* Same as above, but addition re-organized to minimize + data dependency ... unfortunately some compilers are + not very smart to do this automatically */ + double s0, s1, s2, s3; + double t0, t1, t2, t3; + + double v0, v1, v2, v3; + double u0, u1, u2, u3; + double w0, w1, w2, w3; + + s0 = a[0] + b[0]; + s1 = a[1] + b[1]; + s2 = a[2] + b[2]; + s3 = a[3] + b[3]; + + v0 = s0 - a[0]; + v1 = s1 - a[1]; + v2 = s2 - a[2]; + v3 = s3 - a[3]; + + u0 = s0 - v0; + u1 = s1 - v1; + u2 = s2 - v2; + u3 = s3 - v3; + + w0 = a[0] - u0; + w1 = a[1] - u1; + w2 = a[2] - u2; + w3 = a[3] - u3; + + u0 = b[0] - v0; + u1 = b[1] - v1; + u2 = b[2] - v2; + u3 = b[3] - v3; + + t0 = w0 + u0; + t1 = w1 + u1; + t2 = w2 + u2; + t3 = w3 + u3; + + s1 = qd::two_sum(s1, t0, t0); + qd::three_sum(s2, t0, t1); + qd::three_sum2(s3, t0, t2); + t0 = t0 + t1 + t3; + + /* renormalize */ + qd::renorm(s0, s1, s2, s3, t0); + return qd_real(s0, s1, s2, s3); +} + +/* quad-double + quad-double */ +inline qd_real operator+(const qd_real &a, const qd_real &b) { +#ifndef QD_IEEE_ADD + return qd_real::sloppy_add(a, b); +#else + return qd_real::ieee_add(a, b); +#endif +} + + + +/********** Self-Additions ************/ +/* quad-double += double */ +inline qd_real &qd_real::operator+=(double a) { + *this = *this + a; + return *this; +} + +/* quad-double += double-double */ +inline qd_real &qd_real::operator+=(const dd_real &a) { + *this = *this + a; + return *this; +} + +/* quad-double += quad-double */ +inline qd_real &qd_real::operator+=(const qd_real &a) { + *this = *this + a; + return *this; +} + +/********** Unary Minus **********/ +inline qd_real qd_real::operator-() const { + return qd_real(-x[0], -x[1], -x[2], -x[3]); +} + +/********** Subtractions **********/ +inline qd_real operator-(const qd_real &a, double b) { + return (a + (-b)); +} + +inline qd_real operator-(double a, const qd_real &b) { + return (a + (-b)); +} + +inline qd_real operator-(const qd_real &a, const dd_real &b) { + return (a + (-b)); +} + +inline qd_real operator-(const dd_real &a, const qd_real &b) { + return (a + (-b)); +} + +inline qd_real operator-(const qd_real &a, const qd_real &b) { + return (a + (-b)); +} + +/********** Self-Subtractions **********/ +inline qd_real &qd_real::operator-=(double a) { + return ((*this) += (-a)); +} + +inline qd_real &qd_real::operator-=(const dd_real &a) { + return ((*this) += (-a)); +} + +inline qd_real &qd_real::operator-=(const qd_real &a) { + return ((*this) += (-a)); +} + + +inline qd_real operator*(double a, const qd_real &b) { + return (b * a); +} + +inline qd_real operator*(const dd_real &a, const qd_real &b) { + return (b * a); +} + +inline qd_real mul_pwr2(const qd_real &a, double b) { + return qd_real(a[0] * b, a[1] * b, a[2] * b, a[3] * b); +} + +/********** Multiplications **********/ +inline qd_real operator*(const qd_real &a, double b) { + double p0, p1, p2, p3; + double q0, q1, q2; + double s0, s1, s2, s3, s4; + + p0 = qd::two_prod(a[0], b, q0); + p1 = qd::two_prod(a[1], b, q1); + p2 = qd::two_prod(a[2], b, q2); + p3 = a[3] * b; + + s0 = p0; + + s1 = qd::two_sum(q0, p1, s2); + + qd::three_sum(s2, q1, p2); + + qd::three_sum2(q1, q2, p3); + s3 = q1; + + s4 = q2 + p2; + + qd::renorm(s0, s1, s2, s3, s4); + return qd_real(s0, s1, s2, s3); + +} + +/* quad-double * double-double */ +/* a0 * b0 0 + a0 * b1 1 + a1 * b0 2 + a1 * b1 3 + a2 * b0 4 + a2 * b1 5 + a3 * b0 6 + a3 * b1 7 */ +inline qd_real operator*(const qd_real &a, const dd_real &b) { + double p0, p1, p2, p3, p4; + double q0, q1, q2, q3, q4; + double s0, s1, s2; + double t0, t1; + + p0 = qd::two_prod(a[0], b._hi(), q0); + p1 = qd::two_prod(a[0], b._lo(), q1); + p2 = qd::two_prod(a[1], b._hi(), q2); + p3 = qd::two_prod(a[1], b._lo(), q3); + p4 = qd::two_prod(a[2], b._hi(), q4); + + qd::three_sum(p1, p2, q0); + + /* Five-Three-Sum */ + qd::three_sum(p2, p3, p4); + q1 = qd::two_sum(q1, q2, q2); + s0 = qd::two_sum(p2, q1, t0); + s1 = qd::two_sum(p3, q2, t1); + s1 = qd::two_sum(s1, t0, t0); + s2 = t0 + t1 + p4; + p2 = s0; + + p3 = a[2] * b._hi() + a[3] * b._lo() + q3 + q4; + qd::three_sum2(p3, q0, s1); + p4 = q0 + s2; + + qd::renorm(p0, p1, p2, p3, p4); + return qd_real(p0, p1, p2, p3); +} + +/* quad-double * quad-double */ +/* a0 * b0 0 + a0 * b1 1 + a1 * b0 2 + a0 * b2 3 + a1 * b1 4 + a2 * b0 5 + a0 * b3 6 + a1 * b2 7 + a2 * b1 8 + a3 * b0 9 */ +inline qd_real qd_real::sloppy_mul(const qd_real &a, const qd_real &b) { + double p0, p1, p2, p3, p4, p5; + double q0, q1, q2, q3, q4, q5; + double t0, t1; + double s0, s1, s2; + + p0 = qd::two_prod(a[0], b[0], q0); + + p1 = qd::two_prod(a[0], b[1], q1); + p2 = qd::two_prod(a[1], b[0], q2); + + p3 = qd::two_prod(a[0], b[2], q3); + p4 = qd::two_prod(a[1], b[1], q4); + p5 = qd::two_prod(a[2], b[0], q5); + + /* Start Accumulation */ + qd::three_sum(p1, p2, q0); + + /* Six-Three Sum of p2, q1, q2, p3, p4, p5. */ + qd::three_sum(p2, q1, q2); + qd::three_sum(p3, p4, p5); + /* compute (s0, s1, s2) = (p2, q1, q2) + (p3, p4, p5). */ + s0 = qd::two_sum(p2, p3, t0); + s1 = qd::two_sum(q1, p4, t1); + s2 = q2 + p5; + s1 = qd::two_sum(s1, t0, t0); + s2 += (t0 + t1); + + /* O(eps^3) order terms */ + s1 += a[0]*b[3] + a[1]*b[2] + a[2]*b[1] + a[3]*b[0] + q0 + q3 + q4 + q5; + qd::renorm(p0, p1, s0, s1, s2); + return qd_real(p0, p1, s0, s1); +} + +inline qd_real qd_real::accurate_mul(const qd_real &a, const qd_real &b) { + double p0, p1, p2, p3, p4, p5; + double q0, q1, q2, q3, q4, q5; + double p6, p7, p8, p9; + double q6, q7, q8, q9; + double r0, r1; + double t0, t1; + double s0, s1, s2; + + p0 = qd::two_prod(a[0], b[0], q0); + + p1 = qd::two_prod(a[0], b[1], q1); + p2 = qd::two_prod(a[1], b[0], q2); + + p3 = qd::two_prod(a[0], b[2], q3); + p4 = qd::two_prod(a[1], b[1], q4); + p5 = qd::two_prod(a[2], b[0], q5); + + /* Start Accumulation */ + qd::three_sum(p1, p2, q0); + + /* Six-Three Sum of p2, q1, q2, p3, p4, p5. */ + qd::three_sum(p2, q1, q2); + qd::three_sum(p3, p4, p5); + /* compute (s0, s1, s2) = (p2, q1, q2) + (p3, p4, p5). */ + s0 = qd::two_sum(p2, p3, t0); + s1 = qd::two_sum(q1, p4, t1); + s2 = q2 + p5; + s1 = qd::two_sum(s1, t0, t0); + s2 += (t0 + t1); + + /* O(eps^3) order terms */ + p6 = qd::two_prod(a[0], b[3], q6); + p7 = qd::two_prod(a[1], b[2], q7); + p8 = qd::two_prod(a[2], b[1], q8); + p9 = qd::two_prod(a[3], b[0], q9); + + /* Nine-Two-Sum of q0, s1, q3, q4, q5, p6, p7, p8, p9. */ + q0 = qd::two_sum(q0, q3, q3); + q4 = qd::two_sum(q4, q5, q5); + p6 = qd::two_sum(p6, p7, p7); + p8 = qd::two_sum(p8, p9, p9); + /* Compute (t0, t1) = (q0, q3) + (q4, q5). */ + t0 = qd::two_sum(q0, q4, t1); + t1 += (q3 + q5); + /* Compute (r0, r1) = (p6, p7) + (p8, p9). */ + r0 = qd::two_sum(p6, p8, r1); + r1 += (p7 + p9); + /* Compute (q3, q4) = (t0, t1) + (r0, r1). */ + q3 = qd::two_sum(t0, r0, q4); + q4 += (t1 + r1); + /* Compute (t0, t1) = (q3, q4) + s1. */ + t0 = qd::two_sum(q3, s1, t1); + t1 += q4; + + /* O(eps^4) terms -- Nine-One-Sum */ + t1 += a[1] * b[3] + a[2] * b[2] + a[3] * b[1] + q6 + q7 + q8 + q9 + s2; + + qd::renorm(p0, p1, s0, t0, t1); + return qd_real(p0, p1, s0, t0); +} + +inline qd_real operator*(const qd_real &a, const qd_real &b) { +#ifdef QD_SLOPPY_MUL + return qd_real::sloppy_mul(a, b); +#else + return qd_real::accurate_mul(a, b); +#endif +} + +/* quad-double ^ 2 = (x0 + x1 + x2 + x3) ^ 2 + = x0 ^ 2 + 2 x0 * x1 + (2 x0 * x2 + x1 ^ 2) + + (2 x0 * x3 + 2 x1 * x2) */ +inline qd_real sqr(const qd_real &a) { + double p0, p1, p2, p3, p4, p5; + double q0, q1, q2, q3; + double s0, s1; + double t0, t1; + + p0 = qd::two_sqr(a[0], q0); + p1 = qd::two_prod(2.0 * a[0], a[1], q1); + p2 = qd::two_prod(2.0 * a[0], a[2], q2); + p3 = qd::two_sqr(a[1], q3); + + p1 = qd::two_sum(q0, p1, q0); + + q0 = qd::two_sum(q0, q1, q1); + p2 = qd::two_sum(p2, p3, p3); + + s0 = qd::two_sum(q0, p2, t0); + s1 = qd::two_sum(q1, p3, t1); + + s1 = qd::two_sum(s1, t0, t0); + t0 += t1; + + s1 = qd::quick_two_sum(s1, t0, t0); + p2 = qd::quick_two_sum(s0, s1, t1); + p3 = qd::quick_two_sum(t1, t0, q0); + + p4 = 2.0 * a[0] * a[3]; + p5 = 2.0 * a[1] * a[2]; + + p4 = qd::two_sum(p4, p5, p5); + q2 = qd::two_sum(q2, q3, q3); + + t0 = qd::two_sum(p4, q2, t1); + t1 = t1 + p5 + q3; + + p3 = qd::two_sum(p3, t0, p4); + p4 = p4 + q0 + t1; + + qd::renorm(p0, p1, p2, p3, p4); + return qd_real(p0, p1, p2, p3); + +} + +/********** Self-Multiplication **********/ +/* quad-double *= double */ +inline qd_real &qd_real::operator*=(double a) { + *this = (*this * a); + return *this; +} + +/* quad-double *= double-double */ +inline qd_real &qd_real::operator*=(const dd_real &a) { + *this = (*this * a); + return *this; +} + +/* quad-double *= quad-double */ +inline qd_real &qd_real::operator*=(const qd_real &a) { + *this = *this * a; + return *this; +} + +inline qd_real operator/ (const qd_real &a, const dd_real &b) { +#ifdef QD_SLOPPY_DIV + return qd_real::sloppy_div(a, b); +#else + return qd_real::accurate_div(a, b); +#endif +} + +inline qd_real operator/(const qd_real &a, const qd_real &b) { +#ifdef QD_SLOPPY_DIV + return qd_real::sloppy_div(a, b); +#else + return qd_real::accurate_div(a, b); +#endif +} + +/* double / quad-double */ +inline qd_real operator/(double a, const qd_real &b) { + return qd_real(a) / b; +} + +/* double-double / quad-double */ +inline qd_real operator/(const dd_real &a, const qd_real &b) { + return qd_real(a) / b; +} + +/********** Self-Divisions **********/ +/* quad-double /= double */ +inline qd_real &qd_real::operator/=(double a) { + *this = (*this / a); + return *this; +} + +/* quad-double /= double-double */ +inline qd_real &qd_real::operator/=(const dd_real &a) { + *this = (*this / a); + return *this; +} + +/* quad-double /= quad-double */ +inline qd_real &qd_real::operator/=(const qd_real &a) { + *this = (*this / a); + return *this; +} + + +/********** Exponentiation **********/ +inline qd_real qd_real::operator^(int n) const { + return pow(*this, n); +} + +/********** Miscellaneous **********/ +inline qd_real abs(const qd_real &a) { + return (a[0] < 0.0) ? -a : a; +} + +inline qd_real fabs(const qd_real &a) { + return abs(a); +} + +/* Quick version. May be off by one when qd is very close + to the middle of two integers. */ +inline qd_real quick_nint(const qd_real &a) { + qd_real r = qd_real(qd::nint(a[0]), qd::nint(a[1]), + qd::nint(a[2]), qd::nint(a[3])); + r.renorm(); + return r; +} + +/*********** Assignments ************/ +/* quad-double = double */ +inline qd_real &qd_real::operator=(double a) { + x[0] = a; + x[1] = x[2] = x[3] = 0.0; + return *this; +} + +/* quad-double = double-double */ +inline qd_real &qd_real::operator=(const dd_real &a) { + x[0] = a._hi(); + x[1] = a._lo(); + x[2] = x[3] = 0.0; + return *this; +} + +/********** Equality Comparison **********/ +inline bool operator==(const qd_real &a, double b) { + return (a[0] == b && a[1] == 0.0 && a[2] == 0.0 && a[3] == 0.0); +} + +inline bool operator==(double a, const qd_real &b) { + return (b == a); +} + +inline bool operator==(const qd_real &a, const dd_real &b) { + return (a[0] == b._hi() && a[1] == b._lo() && + a[2] == 0.0 && a[3] == 0.0); +} + +inline bool operator==(const dd_real &a, const qd_real &b) { + return (b == a); +} + +inline bool operator==(const qd_real &a, const qd_real &b) { + return (a[0] == b[0] && a[1] == b[1] && + a[2] == b[2] && a[3] == b[3]); +} + + +/********** Less-Than Comparison ***********/ +inline bool operator<(const qd_real &a, double b) { + return (a[0] < b || (a[0] == b && a[1] < 0.0)); +} + +inline bool operator<(double a, const qd_real &b) { + return (b > a); +} + +inline bool operator<(const qd_real &a, const dd_real &b) { + return (a[0] < b._hi() || + (a[0] == b._hi() && (a[1] < b._lo() || + (a[1] == b._lo() && a[2] < 0.0)))); +} + +inline bool operator<(const dd_real &a, const qd_real &b) { + return (b > a); +} + +inline bool operator<(const qd_real &a, const qd_real &b) { + return (a[0] < b[0] || + (a[0] == b[0] && (a[1] < b[1] || + (a[1] == b[1] && (a[2] < b[2] || + (a[2] == b[2] && a[3] < b[3])))))); +} + +/********** Greater-Than Comparison ***********/ +inline bool operator>(const qd_real &a, double b) { + return (a[0] > b || (a[0] == b && a[1] > 0.0)); +} + +inline bool operator>(double a, const qd_real &b) { + return (b < a); +} + +inline bool operator>(const qd_real &a, const dd_real &b) { + return (a[0] > b._hi() || + (a[0] == b._hi() && (a[1] > b._lo() || + (a[1] == b._lo() && a[2] > 0.0)))); +} + +inline bool operator>(const dd_real &a, const qd_real &b) { + return (b < a); +} + +inline bool operator>(const qd_real &a, const qd_real &b) { + return (a[0] > b[0] || + (a[0] == b[0] && (a[1] > b[1] || + (a[1] == b[1] && (a[2] > b[2] || + (a[2] == b[2] && a[3] > b[3])))))); +} + + +/********** Less-Than-Or-Equal-To Comparison **********/ +inline bool operator<=(const qd_real &a, double b) { + return (a[0] < b || (a[0] == b && a[1] <= 0.0)); +} + +inline bool operator<=(double a, const qd_real &b) { + return (b >= a); +} + +inline bool operator<=(const qd_real &a, const dd_real &b) { + return (a[0] < b._hi() || + (a[0] == b._hi() && (a[1] < b._lo() || + (a[1] == b._lo() && a[2] <= 0.0)))); +} + +inline bool operator<=(const dd_real &a, const qd_real &b) { + return (b >= a); +} + +inline bool operator<=(const qd_real &a, const qd_real &b) { + return (a[0] < b[0] || + (a[0] == b[0] && (a[1] < b[1] || + (a[1] == b[1] && (a[2] < b[2] || + (a[2] == b[2] && a[3] <= b[3])))))); +} + +/********** Greater-Than-Or-Equal-To Comparison **********/ +inline bool operator>=(const qd_real &a, double b) { + return (a[0] > b || (a[0] == b && a[1] >= 0.0)); +} + +inline bool operator>=(double a, const qd_real &b) { + return (b <= a); +} + +inline bool operator>=(const qd_real &a, const dd_real &b) { + return (a[0] > b._hi() || + (a[0] == b._hi() && (a[1] > b._lo() || + (a[1] == b._lo() && a[2] >= 0.0)))); +} + +inline bool operator>=(const dd_real &a, const qd_real &b) { + return (b <= a); +} + +inline bool operator>=(const qd_real &a, const qd_real &b) { + return (a[0] > b[0] || + (a[0] == b[0] && (a[1] > b[1] || + (a[1] == b[1] && (a[2] > b[2] || + (a[2] == b[2] && a[3] >= b[3])))))); +} + + + +/********** Not-Equal-To Comparison **********/ +inline bool operator!=(const qd_real &a, double b) { + return !(a == b); +} + +inline bool operator!=(double a, const qd_real &b) { + return !(a == b); +} + +inline bool operator!=(const qd_real &a, const dd_real &b) { + return !(a == b); +} + +inline bool operator!=(const dd_real &a, const qd_real &b) { + return !(a == b); +} + +inline bool operator!=(const qd_real &a, const qd_real &b) { + return !(a == b); +} + + + +inline qd_real aint(const qd_real &a) { + return (a[0] >= 0) ? floor(a) : ceil(a); +} + +inline bool qd_real::is_zero() const { + return (x[0] == 0.0); +} + +inline bool qd_real::is_one() const { + return (x[0] == 1.0 && x[1] == 0.0 && x[2] == 0.0 && x[3] == 0.0); +} + +inline bool qd_real::is_positive() const { + return (x[0] > 0.0); +} + +inline bool qd_real::is_negative() const { + return (x[0] < 0.0); +} + +inline qd_real::operator bool() const { + return (x[0] != 0.0); +} + +inline qd_real::operator double() const { + return to_double(*this); +} + +inline dd_real to_dd_real(const qd_real &a) { + return dd_real(a[0], a[1]); +} + +inline double to_double(const qd_real &a) { + return a[0]; +} + +inline int to_int(const qd_real &a) { + return static_cast(a[0]); +} + +inline qd_real inv(const qd_real &qd) { + return 1.0 / qd; +} + +inline qd_real max(const qd_real &a, const qd_real &b) { + return (a > b) ? a : b; +} + +inline qd_real max(const qd_real &a, const qd_real &b, + const qd_real &c) { + return (a > b) ? ((a > c) ? a : c) : ((b > c) ? b : c); +} + +inline qd_real min(const qd_real &a, const qd_real &b) { + return (a < b) ? a : b; +} + +inline qd_real min(const qd_real &a, const qd_real &b, + const qd_real &c) { + return (a < b) ? ((a < c) ? a : c) : ((b < c) ? b : c); +} + +/* Random number generator */ +inline qd_real qd_real::rand() { + return qdrand(); +} + +inline qd_real ldexp(const qd_real &a, int n) { + return qd_real(std::ldexp(a[0], n), std::ldexp(a[1], n), + std::ldexp(a[2], n), std::ldexp(a[3], n)); +} + +#endif /* _QD_QD_INLINE_H */ diff --git a/src/external/PackedCSparse/qd/qd_real.cc b/src/external/PackedCSparse/qd/qd_real.cc new file mode 100644 index 00000000..02cb7aa3 --- /dev/null +++ b/src/external/PackedCSparse/qd/qd_real.cc @@ -0,0 +1,2624 @@ +/* + * src/qd_real.cc + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2007 + * + * Contains implementation of non-inlined functions of quad-double + * package. Inlined functions are found in qd_inline.h (in include directory). + */ +#include +#include +#include +#include +#include +#include +#include + +#include "qd_config.h" +#include "qd_real.h" +#include "util.h" + +#include "bits.h" + +#ifndef QD_INLINE +#include "qd_inline.h" +#endif + +using std::cout; +using std::cerr; +using std::endl; +using std::istream; +using std::ostream; +using std::ios_base; +using std::string; +using std::setw; + +using namespace qd; + +void qd_real::error(const char *msg) { + //if (msg) { cerr << "ERROR " << msg << endl; } +} + +/********** Multiplications **********/ + +qd_real nint(const qd_real &a) { + double x0, x1, x2, x3; + + x0 = nint(a[0]); + x1 = x2 = x3 = 0.0; + + if (x0 == a[0]) { + /* First double is already an integer. */ + x1 = nint(a[1]); + + if (x1 == a[1]) { + /* Second double is already an integer. */ + x2 = nint(a[2]); + + if (x2 == a[2]) { + /* Third double is already an integer. */ + x3 = nint(a[3]); + } else { + if (std::abs(x2 - a[2]) == 0.5 && a[3] < 0.0) { + x2 -= 1.0; + } + } + + } else { + if (std::abs(x1 - a[1]) == 0.5 && a[2] < 0.0) { + x1 -= 1.0; + } + } + + } else { + /* First double is not an integer. */ + if (std::abs(x0 - a[0]) == 0.5 && a[1] < 0.0) { + x0 -= 1.0; + } + } + + renorm(x0, x1, x2, x3); + return qd_real(x0, x1, x2, x3); +} + +qd_real floor(const qd_real &a) { + double x0, x1, x2, x3; + x1 = x2 = x3 = 0.0; + x0 = std::floor(a[0]); + + if (x0 == a[0]) { + x1 = std::floor(a[1]); + + if (x1 == a[1]) { + x2 = std::floor(a[2]); + + if (x2 == a[2]) { + x3 = std::floor(a[3]); + } + } + + renorm(x0, x1, x2, x3); + return qd_real(x0, x1, x2, x3); + } + + return qd_real(x0, x1, x2, x3); +} + +qd_real ceil(const qd_real &a) { + double x0, x1, x2, x3; + x1 = x2 = x3 = 0.0; + x0 = std::ceil(a[0]); + + if (x0 == a[0]) { + x1 = std::ceil(a[1]); + + if (x1 == a[1]) { + x2 = std::ceil(a[2]); + + if (x2 == a[2]) { + x3 = std::ceil(a[3]); + } + } + + renorm(x0, x1, x2, x3); + return qd_real(x0, x1, x2, x3); + } + + return qd_real(x0, x1, x2, x3); +} + + + +/********** Divisions **********/ +/* quad-double / double */ +qd_real operator/(const qd_real &a, double b) { + /* Strategy: compute approximate quotient using high order + doubles, and then correct it 3 times using the remainder. + (Analogous to long division.) */ + double t0, t1; + double q0, q1, q2, q3; + qd_real r; + + q0 = a[0] / b; /* approximate quotient */ + + /* Compute the remainder a - q0 * b */ + t0 = two_prod(q0, b, t1); + r = a - dd_real(t0, t1); + + /* Compute the first correction */ + q1 = r[0] / b; + t0 = two_prod(q1, b, t1); + r -= dd_real(t0, t1); + + /* Second correction to the quotient. */ + q2 = r[0] / b; + t0 = two_prod(q2, b, t1); + r -= dd_real(t0, t1); + + /* Final correction to the quotient. */ + q3 = r[0] / b; + + renorm(q0, q1, q2, q3); + return qd_real(q0, q1, q2, q3); +} + +qd_real::qd_real(const char *s) { + if (qd_real::read(s, *this)) { + qd_real::error("(qd_real::qd_real): INPUT ERROR."); + *this = qd_real::_nan; + } +} + +qd_real &qd_real::operator=(const char *s) { + if (qd_real::read(s, *this)) { + qd_real::error("(qd_real::operator=): INPUT ERROR."); + *this = qd_real::_nan; + } + return *this; +} + +istream &operator>>(istream &s, qd_real &qd) { + char str[255]; + s >> str; + qd = qd_real(str); + return s; +} + +ostream &operator<<(ostream &os, const qd_real &qd) { + bool showpos = (os.flags() & ios_base::showpos) != 0; + bool uppercase = (os.flags() & ios_base::uppercase) != 0; + return os << qd.to_string((int)os.precision(), (int)os.width(), os.flags(), + showpos, uppercase, os.fill()); +} + +/* Read a quad-double from s. */ +int qd_real::read(const char *s, qd_real &qd) { + const char *p = s; + char ch; + int sign = 0; + int point = -1; /* location of decimal point */ + int nd = 0; /* number of digits read */ + int e = 0; /* exponent. */ + bool done = false; + qd_real r = 0.0; /* number being read */ + + /* Skip any leading spaces */ + while (*p == ' ') p++; + + while (!done && (ch = *p) != '\0') { + if (ch >= '0' && ch <= '9') { + /* It's a digit */ + int d = ch - '0'; + r *= 10.0; + r += static_cast(d); + nd++; + } else { + /* Non-digit */ + switch (ch) { + case '.': + if (point >= 0) + return -1; /* we've already encountered a decimal point. */ + point = nd; + break; + case '-': + case '+': + if (sign != 0 || nd > 0) + return -1; /* we've already encountered a sign, or if its + not at first position. */ + sign = (ch == '-') ? -1 : 1; + break; + case 'E': + case 'e': + int nread; + nread = std::sscanf(p+1, "%d", &e); + done = true; + if (nread != 1) + return -1; /* read of exponent failed. */ + break; + case ' ': + done = true; + break; + default: + return -1; + + } + } + + p++; + } + + + + /* Adjust exponent to account for decimal point */ + if (point >= 0) { + e -= (nd - point); + } + + /* Multiply the the exponent */ + if (e != 0) { + r *= (qd_real(10.0) ^ e); + } + + qd = (sign < 0) ? -r : r; + return 0; +} + +void qd_real::to_digits(char *s, int &expn, int precision) const { + int D = precision + 1; /* number of digits to compute */ + + qd_real r = abs(*this); + int e; /* exponent */ + int i, d; + + if (x[0] == 0.0) { + /* this == 0.0 */ + expn = 0; + for (i = 0; i < precision; i++) s[i] = '0'; + return; + } + + /* First determine the (approximate) exponent. */ + e = static_cast(std::floor(std::log10(std::abs(x[0])))); + + if (e < -300) { + r *= qd_real(10.0) ^ 300; + r /= qd_real(10.0) ^ (e + 300); + } else if (e > 300) { + r = ldexp(r, -53); + r /= qd_real(10.0) ^ e; + r = ldexp(r, 53); + } else { + r /= qd_real(10.0) ^ e; + } + + /* Fix exponent if we are off by one */ + if (r >= 10.0) { + r /= 10.0; + e++; + } else if (r < 1.0) { + r *= 10.0; + e--; + } + + if (r >= 10.0 || r < 1.0) { + qd_real::error("(qd_real::to_digits): can't compute exponent."); + return; + } + + /* Extract the digits */ + for (i = 0; i < D; i++) { + d = static_cast(r[0]); + r -= d; + r *= 10.0; + + s[i] = static_cast(d + '0'); + } + + /* Fix out of range digits. */ + for (i = D-1; i > 0; i--) { + if (s[i] < '0') { + s[i-1]--; + s[i] += 10; + } else if (s[i] > '9') { + s[i-1]++; + s[i] -= 10; + } + } + + if (s[0] <= '0') { + qd_real::error("(qd_real::to_digits): non-positive leading digit."); + return; + } + + /* Round, handle carry */ + if (s[D-1] >= '5') { + s[D-2]++; + + i = D-2; + while (i > 0 && s[i] > '9') { + s[i] -= 10; + s[--i]++; + } + } + + /* If first digit is 10, shift everything. */ + if (s[0] > '9') { + e++; + for (i = precision; i >= 2; i--) s[i] = s[i-1]; + s[0] = '1'; + s[1] = '0'; + } + + s[precision] = 0; + expn = e; +} + +/* Writes the quad-double number into the character array s of length len. + The integer d specifies how many significant digits to write. + The string s must be able to hold at least (d+8) characters. + showpos indicates whether to use the + sign, and uppercase indicates + whether the E or e is to be used for the exponent. */ +void qd_real::write(char *s, int len, int precision, + bool showpos, bool uppercase) const { + string str = to_string(precision, 0, ios_base::scientific, showpos, uppercase); + strncpy(s, str.c_str(), len-1); + s[len-1] = 0; +} + +void round_string_qd(char *s, int precision, int *offset){ + /* + Input string must be all digits or errors will occur. + */ + + int i; + int D = precision ; + + /* Round, handle carry */ + if (D>0 && s[D] >= '5') { + s[D-1]++; + + i = D-1; + while (i > 0 && s[i] > '9') { + s[i] -= 10; + s[--i]++; + } + } + + /* If first digit is 10, shift everything. */ + if (s[0] > '9') { + // e++; // don't modify exponent here + for (i = precision; i >= 1; i--) s[i+1] = s[i]; + s[0] = '1'; + s[1] = '0'; + + (*offset)++ ; // now offset needs to be increased by one + precision++ ; + } + + s[precision] = 0; // add terminator for array +} + + +string qd_real::to_string(int precision, int width, ios_base::fmtflags fmt, + bool showpos, bool uppercase, char fill) const { + string s; + bool fixed = (fmt & ios_base::fixed) != 0; + bool sgn = true; + int i, e = 0; + + if (isinf()) { + if (*this < 0.0) + s += '-'; + else if (showpos) + s += '+'; + else + sgn = false; + s += uppercase ? "INF" : "inf"; + } else if (isnan()) { + s = uppercase ? "NAN" : "nan"; + sgn = false; + } else { + if (*this < 0.0) + s += '-'; + else if (showpos) + s += '+'; + else + sgn = false; + + if (*this == 0.0) { + /* Zero case */ + s += '0'; + if (precision > 0) { + s += '.'; + s.append(precision, '0'); + } + } else { + /* Non-zero case */ + int off = (fixed ? (1 + to_int(floor(log10(abs(*this))))) : 1); + int d = precision + off; + + int d_with_extra = d; + if(fixed) + d_with_extra = std::max(120, d); // longer than the max accuracy for DD + + // highly special case - fixed mode, precision is zero, abs(*this) < 1.0 + // without this trap a number like 0.9 printed fixed with 0 precision prints as 0 + // should be rounded to 1. + if(fixed && (precision == 0) && (abs(*this) < 1.0)){ + if(abs(*this) >= 0.5) + s += '1'; + else + s += '0'; + + return s; + } + + // handle near zero to working precision (but not exactly zero) + if (fixed && d <= 0) { + s += '0'; + if (precision > 0) { + s += '.'; + s.append(precision, '0'); + } + } else { // default + + char *t ; // = new char[d+1]; + int j; + + if(fixed){ + t = new char[d_with_extra+1]; + to_digits(t, e, d_with_extra); + } + else{ + t = new char[d+1]; + to_digits(t, e, d); + } + + off = e + 1; + + if (fixed) { + // fix the string if it's been computed incorrectly + // round here in the decimal string if required + round_string_qd(t, d, &off); + + if (off > 0) { + for (i = 0; i < off; i++) s += t[i]; + if (precision > 0) { + s += '.'; + for (j = 0; j < precision; j++, i++) s += t[i]; + } + } else { + s += "0."; + if (off < 0) s.append(-off, '0'); + for (i = 0; i < d; i++) s += t[i]; + } + } else { + s += t[0]; + if (precision > 0) s += '.'; + + for (i = 1; i <= precision; i++) + s += t[i]; + + } + delete [] t; + } + } + + // trap for improper offset with large values + // without this trap, output of values of the for 10^j - 1 fail for j > 28 + // and are output with the point in the wrong place, leading to a dramatically off value + if(fixed && (precision > 0)){ + // make sure that the value isn't dramatically larger + double from_string = atof(s.c_str()); + + // if this ratio is large, then we've got problems + if( fabs( from_string / this->x[0] ) > 3.0 ){ + + // loop on the string, find the point, move it up one + // don't act on the first character + for(i=1; i < (int)s.length(); i++){ + if(s[i] == '.'){ + s[i] = s[i-1] ; + s[i-1] = '.' ; + break; + } + } + + from_string = atof(s.c_str()); + // if this ratio is large, then the string has not been fixed + if( fabs( from_string / this->x[0] ) > 3.0 ){ + dd_real::error("Re-rounding unsuccessful in large number fixed point trap.") ; + } + } + } + + if (!fixed) { + /* Fill in exponent part */ + s += uppercase ? 'E' : 'e'; + append_expn(s, e); + } + } + + /* Fill in the blanks */ + size_t len = s.length(); + if (len < width) { + int delta = width - len; + if (fmt & ios_base::internal) { + if (sgn) + s.insert(static_cast(1), delta, fill); + else + s.insert(static_cast(0), delta, fill); + } else if (fmt & ios_base::left) { + s.append(delta, fill); + } else { + s.insert(static_cast(0), delta, fill); + } + } + + return s; +} + +/* Computes qd^n, where n is an integer. */ +qd_real pow(const qd_real &a, int n) { + if (n == 0) + return 1.0; + + qd_real r = a; /* odd-case multiplier */ + qd_real s = 1.0; /* current answer */ + int N = std::abs(n); + + if (N > 1) { + + /* Use binary exponentiation. */ + while (N > 0) { + if (N % 2 == 1) { + /* If odd, multiply by r. Note eventually N = 1, so this + eventually executes. */ + s *= r; + } + N /= 2; + if (N > 0) + r = sqr(r); + } + + } else { + s = r; + } + + if (n < 0) + return (1.0 / s); + + return s; +} + +qd_real pow(const qd_real &a, const qd_real &b) { + return exp(b * log(a)); +} + +qd_real npwr(const qd_real &a, int n) { + return pow(a, n); +} + +/* Debugging routines */ +void qd_real::dump_bits(const string &name, std::ostream &os) const { + string::size_type len = name.length(); + if (len > 0) { + os << name << " = "; + len += 3; + } + os << "[ "; + len += 2; + for (int j = 0; j < 4; j++) { + if (j > 0) for (string::size_type i = 0; i < len; i++) os << ' '; + print_double_info(os, x[j]); + if (j < 3) + os << endl; + else + os << " ]" << endl; + } +} + +void qd_real::dump(const string &name, std::ostream &os) const { + std::ios_base::fmtflags old_flags = os.flags(); + std::streamsize old_prec = os.precision(19); + os << std::scientific; + + string::size_type len = name.length(); + if (len > 0) { + os << name << " = "; + len += 3; + } + os << "[ "; + len += 2; + os << setw(27) << x[0] << ", " << setw(26) << x[1] << "," << endl; + for (string::size_type i = 0; i < len; i++) os << ' '; + os << setw(27) << x[2] << ", " << setw(26) << x[3] << " ]" << endl; + + os.precision(old_prec); + os.flags(old_flags); +} + +/* Divisions */ +/* quad-double / double-double */ +qd_real qd_real::sloppy_div(const qd_real &a, const dd_real &b) { + double q0, q1, q2, q3; + qd_real r; + qd_real qd_b(b); + + q0 = a[0] / b._hi(); + r = a - q0 * qd_b; + + q1 = r[0] / b._hi(); + r -= (q1 * qd_b); + + q2 = r[0] / b._hi(); + r -= (q2 * qd_b); + + q3 = r[0] / b._hi(); + + ::renorm(q0, q1, q2, q3); + return qd_real(q0, q1, q2, q3); +} + +qd_real qd_real::accurate_div(const qd_real &a, const dd_real &b) { + double q0, q1, q2, q3, q4; + qd_real r; + qd_real qd_b(b); + + q0 = a[0] / b._hi(); + r = a - q0 * qd_b; + + q1 = r[0] / b._hi(); + r -= (q1 * qd_b); + + q2 = r[0] / b._hi(); + r -= (q2 * qd_b); + + q3 = r[0] / b._hi(); + r -= (q3 * qd_b); + + q4 = r[0] / b._hi(); + + ::renorm(q0, q1, q2, q3, q4); + return qd_real(q0, q1, q2, q3); +} + +/* quad-double / quad-double */ +qd_real qd_real::sloppy_div(const qd_real &a, const qd_real &b) { + double q0, q1, q2, q3; + + qd_real r; + + q0 = a[0] / b[0]; + r = a - (b * q0); + + q1 = r[0] / b[0]; + r -= (b * q1); + + q2 = r[0] / b[0]; + r -= (b * q2); + + q3 = r[0] / b[0]; + + ::renorm(q0, q1, q2, q3); + + return qd_real(q0, q1, q2, q3); +} + +qd_real qd_real::accurate_div(const qd_real &a, const qd_real &b) { + double q0, q1, q2, q3; + + qd_real r; + + q0 = a[0] / b[0]; + r = a - (b * q0); + + q1 = r[0] / b[0]; + r -= (b * q1); + + q2 = r[0] / b[0]; + r -= (b * q2); + + q3 = r[0] / b[0]; + + r -= (b * q3); + double q4 = r[0] / b[0]; + + ::renorm(q0, q1, q2, q3, q4); + + return qd_real(q0, q1, q2, q3); +} + +QD_API qd_real sqrt(const qd_real &a) { + /* Strategy: + + Perform the following Newton iteration: + + x' = x + (1 - a * x^2) * x / 2; + + which converges to 1/sqrt(a), starting with the + double precision approximation to 1/sqrt(a). + Since Newton's iteration more or less doubles the + number of correct digits, we only need to perform it + twice. + */ + + if (a.is_zero()) + return 0.0; + + if (a.is_negative()) { + qd_real::error("(qd_real::sqrt): Negative argument."); + return qd_real::_nan; + } + + qd_real r = (1.0 / std::sqrt(a[0])); + qd_real h = mul_pwr2(a, 0.5); + + r += ((0.5 - h * sqr(r)) * r); + r += ((0.5 - h * sqr(r)) * r); + r += ((0.5 - h * sqr(r)) * r); + + r *= a; + return r; +} + + +/* Computes the n-th root of a */ +qd_real nroot(const qd_real &a, int n) { + /* Strategy: Use Newton's iteration to solve + + 1/(x^n) - a = 0 + + Newton iteration becomes + + x' = x + x * (1 - a * x^n) / n + + Since Newton's iteration converges quadratically, + we only need to perform it twice. + + */ + if (n <= 0) { + qd_real::error("(qd_real::nroot): N must be positive."); + return qd_real::_nan; + } + + if (n % 2 == 0 && a.is_negative()) { + qd_real::error("(qd_real::nroot): Negative argument."); + return qd_real::_nan; + } + + if (n == 1) { + return a; + } + if (n == 2) { + return sqrt(a); + } + if (a.is_zero()) { + return qd_real(0.0); + } + + + /* Note a^{-1/n} = exp(-log(a)/n) */ + qd_real r = abs(a); + qd_real x = std::exp(-std::log(r.x[0]) / n); + + /* Perform Newton's iteration. */ + double dbl_n = static_cast(n); + x += x * (1.0 - r * npwr(x, n)) / dbl_n; + x += x * (1.0 - r * npwr(x, n)) / dbl_n; + x += x * (1.0 - r * npwr(x, n)) / dbl_n; + if (a[0] < 0.0){ + x = -x; + } + return 1.0 / x; +} + +static const int n_inv_fact = 15; +static const qd_real inv_fact[n_inv_fact] = { + qd_real( 1.66666666666666657e-01, 9.25185853854297066e-18, + 5.13581318503262866e-34, 2.85094902409834186e-50), + qd_real( 4.16666666666666644e-02, 2.31296463463574266e-18, + 1.28395329625815716e-34, 7.12737256024585466e-51), + qd_real( 8.33333333333333322e-03, 1.15648231731787138e-19, + 1.60494162032269652e-36, 2.22730392507682967e-53), + qd_real( 1.38888888888888894e-03, -5.30054395437357706e-20, + -1.73868675534958776e-36, -1.63335621172300840e-52), + qd_real( 1.98412698412698413e-04, 1.72095582934207053e-22, + 1.49269123913941271e-40, 1.29470326746002471e-58), + qd_real( 2.48015873015873016e-05, 2.15119478667758816e-23, + 1.86586404892426588e-41, 1.61837908432503088e-59), + qd_real( 2.75573192239858925e-06, -1.85839327404647208e-22, + 8.49175460488199287e-39, -5.72661640789429621e-55), + qd_real( 2.75573192239858883e-07, 2.37677146222502973e-23, + -3.26318890334088294e-40, 1.61435111860404415e-56), + qd_real( 2.50521083854417202e-08, -1.44881407093591197e-24, + 2.04267351467144546e-41, -8.49632672007163175e-58), + qd_real( 2.08767569878681002e-09, -1.20734505911325997e-25, + 1.70222792889287100e-42, 1.41609532150396700e-58), + qd_real( 1.60590438368216133e-10, 1.25852945887520981e-26, + -5.31334602762985031e-43, 3.54021472597605528e-59), + qd_real( 1.14707455977297245e-11, 2.06555127528307454e-28, + 6.88907923246664603e-45, 5.72920002655109095e-61), + qd_real( 7.64716373181981641e-13, 7.03872877733453001e-30, + -7.82753927716258345e-48, 1.92138649443790242e-64), + qd_real( 4.77947733238738525e-14, 4.39920548583408126e-31, + -4.89221204822661465e-49, 1.20086655902368901e-65), + qd_real( 2.81145725434552060e-15, 1.65088427308614326e-31, + -2.87777179307447918e-50, 4.27110689256293549e-67) +}; + +qd_real exp(const qd_real &a) { + /* Strategy: We first reduce the size of x by noting that + + exp(kr + m * log(2)) = 2^m * exp(r)^k + + where m and k are integers. By choosing m appropriately + we can make |kr| <= log(2) / 2 = 0.347. Then exp(r) is + evaluated using the familiar Taylor series. Reducing the + argument substantially speeds up the convergence. */ + + const double k = ldexp(1.0, 16); + const double inv_k = 1.0 / k; + + if (a[0] <= -709.0) + return 0.0; + + if (a[0] >= 709.0) + return qd_real::_inf; + + if (a.is_zero()) + return 1.0; + + if (a.is_one()) + return qd_real::_e; + + double m = std::floor(a.x[0] / qd_real::_log2.x[0] + 0.5); + qd_real r = mul_pwr2(a - qd_real::_log2 * m, inv_k); + qd_real s, p, t; + double thresh = inv_k * qd_real::_eps; + + p = sqr(r); + s = r + mul_pwr2(p, 0.5); + int i = 0; + do { + p *= r; + t = p * inv_fact[i++]; + s += t; + } while (std::abs(to_double(t)) > thresh && i < 9); + + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s = mul_pwr2(s, 2.0) + sqr(s); + s += 1.0; + return ldexp(s, static_cast(m)); +} + +/* Logarithm. Computes log(x) in quad-double precision. + This is a natural logarithm (i.e., base e). */ +qd_real log(const qd_real &a) { + /* Strategy. The Taylor series for log converges much more + slowly than that of exp, due to the lack of the factorial + term in the denominator. Hence this routine instead tries + to determine the root of the function + + f(x) = exp(x) - a + + using Newton iteration. The iteration is given by + + x' = x - f(x)/f'(x) + = x - (1 - a * exp(-x)) + = x + a * exp(-x) - 1. + + Two iteration is needed, since Newton's iteration + approximately doubles the number of digits per iteration. */ + + if (a.is_one()) { + return 0.0; + } + + if (a[0] <= 0.0) { + qd_real::error("(qd_real::log): Non-positive argument."); + return qd_real::_nan; + } + + if (a[0] == 0.0) { + return -qd_real::_inf; + } + + qd_real x = std::log(a[0]); /* Initial approximation */ + + x = x + a * exp(-x) - 1.0; + x = x + a * exp(-x) - 1.0; + x = x + a * exp(-x) - 1.0; + + return x; +} + +qd_real log10(const qd_real &a) { + return log(a) / qd_real::_log10; +} + +static const qd_real _pi1024 = qd_real( + 3.067961575771282340e-03, 1.195944139792337116e-19, + -2.924579892303066080e-36, 1.086381075061880158e-52); + +/* Table of sin(k * pi/1024) and cos(k * pi/1024). */ +static const qd_real sin_table [] = { + qd_real( 3.0679567629659761e-03, 1.2690279085455925e-19, + 5.2879464245328389e-36, -1.7820334081955298e-52), + qd_real( 6.1358846491544753e-03, 9.0545257482474933e-20, + 1.6260113133745320e-37, -9.7492001208767410e-55), + qd_real( 9.2037547820598194e-03, -1.2136591693535934e-19, + 5.5696903949425567e-36, 1.2505635791936951e-52), + qd_real( 1.2271538285719925e-02, 6.9197907640283170e-19, + -4.0203726713435555e-36, -2.0688703606952816e-52), + qd_real( 1.5339206284988102e-02, -8.4462578865401696e-19, + 4.6535897505058629e-35, -1.3923682978570467e-51), + qd_real( 1.8406729905804820e-02, 7.4195533812833160e-19, + 3.9068476486787607e-35, 3.6393321292898614e-52), + qd_real( 2.1474080275469508e-02, -4.5407960207688566e-19, + -2.2031770119723005e-35, 1.2709814654833741e-51), + qd_real( 2.4541228522912288e-02, -9.1868490125778782e-20, + 4.8706148704467061e-36, -2.8153947855469224e-52), + qd_real( 2.7608145778965743e-02, -1.5932358831389269e-18, + -7.0475416242776030e-35, -2.7518494176602744e-51), + qd_real( 3.0674803176636626e-02, -1.6936054844107918e-20, + -2.0039543064442544e-36, -1.6267505108658196e-52), + qd_real( 3.3741171851377587e-02, -2.0096074292368340e-18, + -1.3548237016537134e-34, 6.5554881875899973e-51), + qd_real( 3.6807222941358832e-02, 6.1060088803529842e-19, + -4.0448721259852727e-35, -2.1111056765671495e-51), + qd_real( 3.9872927587739811e-02, 4.6657453481183289e-19, + 3.4119333562288684e-35, 2.4007534726187511e-51), + qd_real( 4.2938256934940820e-02, 2.8351940588660907e-18, + 1.6991309601186475e-34, 6.8026536098672629e-51), + qd_real( 4.6003182130914630e-02, -1.1182813940157788e-18, + 7.5235020270378946e-35, 4.1187304955493722e-52), + qd_real( 4.9067674327418015e-02, -6.7961037205182801e-19, + -4.4318868124718325e-35, -9.9376628132525316e-52), + qd_real( 5.2131704680283324e-02, -2.4243695291953779e-18, + -1.3675405320092298e-34, -8.3938137621145070e-51), + qd_real( 5.5195244349689941e-02, -1.3340299860891103e-18, + -3.4359574125665608e-35, 1.1911462755409369e-51), + qd_real( 5.8258264500435759e-02, 2.3299905496077492e-19, + 1.9376108990628660e-36, -5.1273775710095301e-53), + qd_real( 6.1320736302208578e-02, -5.1181134064638108e-19, + -4.2726335866706313e-35, 2.6368495557440691e-51), + qd_real( 6.4382630929857465e-02, -4.2325997000052705e-18, + 3.3260117711855937e-35, 1.4736267706718352e-51), + qd_real( 6.7443919563664065e-02, -6.9221796556983636e-18, + 1.5909286358911040e-34, -7.8828946891835218e-51), + qd_real( 7.0504573389613870e-02, -6.8552791107342883e-18, + -1.9961177630841580e-34, 2.0127129580485300e-50), + qd_real( 7.3564563599667426e-02, -2.7784941506273593e-18, + -9.1240375489852821e-35, -1.9589752023546795e-51), + qd_real( 7.6623861392031492e-02, 2.3253700287958801e-19, + -1.3186083921213440e-36, -4.9927872608099673e-53), + qd_real( 7.9682437971430126e-02, -4.4867664311373041e-18, + 2.8540789143650264e-34, 2.8491348583262741e-51), + qd_real( 8.2740264549375692e-02, 1.4735983530877760e-18, + 3.7284093452233713e-35, 2.9024430036724088e-52), + qd_real( 8.5797312344439894e-02, -3.3881893830684029e-18, + -1.6135529531508258e-34, 7.7294651620588049e-51), + qd_real( 8.8853552582524600e-02, -3.7501775830290691e-18, + 3.7543606373911573e-34, 2.2233701854451859e-50), + qd_real( 9.1908956497132724e-02, 4.7631594854274564e-18, + 1.5722874642939344e-34, -4.8464145447831456e-51), + qd_real( 9.4963495329639006e-02, -6.5885886400417564e-18, + -2.1371116991641965e-34, 1.3819370559249300e-50), + qd_real( 9.8017140329560604e-02, -1.6345823622442560e-18, + -1.3209238810006454e-35, -3.5691060049117942e-52), + qd_real( 1.0106986275482782e-01, 3.3164325719308656e-18, + -1.2004224885132282e-34, 7.2028828495418631e-51), + qd_real( 1.0412163387205457e-01, 6.5760254085385100e-18, + 1.7066246171219214e-34, -4.9499340996893514e-51), + qd_real( 1.0717242495680884e-01, 6.4424044279026198e-18, + -8.3956976499698139e-35, -4.0667730213318321e-51), + qd_real( 1.1022220729388306e-01, -5.6789503537823233e-19, + 1.0380274792383233e-35, 1.5213997918456695e-52), + qd_real( 1.1327095217756435e-01, 2.7100481012132900e-18, + 1.5323292999491619e-35, 4.9564432810360879e-52), + qd_real( 1.1631863091190477e-01, 1.0294914877509705e-18, + -9.3975734948993038e-35, 1.3534827323719708e-52), + qd_real( 1.1936521481099137e-01, -3.9500089391898506e-18, + 3.5317349978227311e-34, 1.8856046807012275e-51), + qd_real( 1.2241067519921620e-01, 2.8354501489965335e-18, + 1.8151655751493305e-34, -2.8716592177915192e-51), + qd_real( 1.2545498341154623e-01, 4.8686751763148235e-18, + 5.9878105258097936e-35, -3.3534629098722107e-51), + qd_real( 1.2849811079379317e-01, 3.8198603954988802e-18, + -1.8627501455947798e-34, -2.4308161133527791e-51), + qd_real( 1.3154002870288312e-01, -5.0039708262213813e-18, + -1.2983004159245552e-34, -4.6872034915794122e-51), + qd_real( 1.3458070850712620e-01, -9.1670359171480699e-18, + 1.5916493007073973e-34, 4.0237002484366833e-51), + qd_real( 1.3762012158648604e-01, 6.6253255866774482e-18, + -2.3746583031401459e-34, -9.3703876173093250e-52), + qd_real( 1.4065823933284924e-01, -7.9193932965524741e-18, + 6.0972464202108397e-34, 2.4566623241035797e-50), + qd_real( 1.4369503315029444e-01, 1.1472723016618666e-17, + -5.1884954557576435e-35, -4.2220684832186607e-51), + qd_real( 1.4673047445536175e-01, 3.7269471470465677e-18, + 3.7352398151250827e-34, -4.0881822289508634e-51), + qd_real( 1.4976453467732151e-01, 8.0812114131285151e-18, + 1.2979142554917325e-34, 9.9380667487736254e-51), + qd_real( 1.5279718525844344e-01, -7.6313573938416838e-18, + 5.7714690450284125e-34, -3.7731132582986687e-50), + qd_real( 1.5582839765426523e-01, 3.0351307187678221e-18, + -1.0976942315176184e-34, 7.8734647685257867e-51), + qd_real( 1.5885814333386145e-01, -4.0163200573859079e-18, + -9.2840580257628812e-35, -2.8567420029274875e-51), + qd_real( 1.6188639378011183e-01, 1.1850519643573528e-17, + -5.0440990519162957e-34, 3.0510028707928009e-50), + qd_real( 1.6491312048996992e-01, -7.0405288319166738e-19, + 3.3211107491245527e-35, 8.6663299254686031e-52), + qd_real( 1.6793829497473117e-01, 5.4284533721558139e-18, + -3.3263339336181369e-34, -1.8536367335123848e-50), + qd_real( 1.7096188876030122e-01, 9.1919980181759094e-18, + -6.7688743940982606e-34, -1.0377711384318389e-50), + qd_real( 1.7398387338746382e-01, 5.8151994618107928e-18, + -1.6751014298301606e-34, -6.6982259797164963e-51), + qd_real( 1.7700422041214875e-01, 6.7329300635408167e-18, + 2.8042736644246623e-34, 3.6786888232793599e-51), + qd_real( 1.8002290140569951e-01, 7.9701826047392143e-18, + -7.0765920110524977e-34, 1.9622512608461784e-50), + qd_real( 1.8303988795514095e-01, 7.7349918688637383e-18, + -4.4803769968145083e-34, 1.1201148793328890e-50), + qd_real( 1.8605515166344666e-01, -1.2564893007679552e-17, + 7.5953844248530810e-34, -3.8471695132415039e-51), + qd_real( 1.8906866414980622e-01, -7.6208955803527778e-18, + -4.4792298656662981e-34, -4.4136824096645007e-50), + qd_real( 1.9208039704989244e-01, 4.3348343941174903e-18, + -2.3404121848139937e-34, 1.5789970962611856e-50), + qd_real( 1.9509032201612828e-01, -7.9910790684617313e-18, + 6.1846270024220713e-34, -3.5840270918032937e-50), + qd_real( 1.9809841071795359e-01, -1.8434411800689445e-18, + 1.4139031318237285e-34, 1.0542811125343809e-50), + qd_real( 2.0110463484209190e-01, 1.1010032669300739e-17, + -3.9123576757413791e-34, 2.4084852500063531e-51), + qd_real( 2.0410896609281687e-01, 6.0941297773957752e-18, + -2.8275409970449641e-34, 4.6101008563532989e-51), + qd_real( 2.0711137619221856e-01, -1.0613362528971356e-17, + 2.2456805112690884e-34, 1.3483736125280904e-50), + qd_real( 2.1011183688046961e-01, 1.1561548476512844e-17, + 6.0355905610401254e-34, 3.3329909618405675e-50), + qd_real( 2.1311031991609136e-01, 1.2031873821063860e-17, + -3.4142699719695635e-34, -1.2436262780241778e-50), + qd_real( 2.1610679707621952e-01, -1.0111196082609117e-17, + 7.2789545335189643e-34, -2.9347540365258610e-50), + qd_real( 2.1910124015686980e-01, -3.6513812299150776e-19, + -2.3359499418606442e-35, 3.1785298198458653e-52), + qd_real( 2.2209362097320354e-01, -3.0337210995812162e-18, + 6.6654668033632998e-35, 2.0110862322656942e-51), + qd_real( 2.2508391135979283e-01, 3.9507040822556510e-18, + 2.4287993958305375e-35, 5.6662797513020322e-52), + qd_real( 2.2807208317088573e-01, 8.2361837339258012e-18, + 6.9786781316397937e-34, -6.4122962482639504e-51), + qd_real( 2.3105810828067111e-01, 1.0129787149761869e-17, + -6.9359234615816044e-34, -2.8877355604883782e-50), + qd_real( 2.3404195858354343e-01, -6.9922402696101173e-18, + -5.7323031922750280e-34, 5.3092579966872727e-51), + qd_real( 2.3702360599436720e-01, 8.8544852285039918e-18, + 1.3588480826354134e-34, 1.0381022520213867e-50), + qd_real( 2.4000302244874150e-01, -1.2137758975632164e-17, + -2.6448807731703891e-34, -1.9929733800670473e-51), + qd_real( 2.4298017990326390e-01, -8.7514315297196632e-18, + -6.5723260373079431e-34, -1.0333158083172177e-50), + qd_real( 2.4595505033579462e-01, -1.1129044052741832e-17, + 4.3805998202883397e-34, 1.2219399554686291e-50), + qd_real( 2.4892760574572018e-01, -8.1783436100020990e-18, + 5.5666875261111840e-34, 3.8080473058748167e-50), + qd_real( 2.5189781815421697e-01, -1.7591436032517039e-17, + -1.0959681232525285e-33, 5.6209426020232456e-50), + qd_real( 2.5486565960451457e-01, -1.3602299806901461e-19, + -6.0073844642762535e-36, -3.0072751311893878e-52), + qd_real( 2.5783110216215899e-01, 1.8480038630879957e-17, + 3.3201664714047599e-34, -5.5547819290576764e-51), + qd_real( 2.6079411791527551e-01, 4.2721420983550075e-18, + 5.6782126934777920e-35, 3.1428338084365397e-51), + qd_real( 2.6375467897483140e-01, -1.8837947680038700e-17, + 1.3720129045754794e-33, -8.2763406665966033e-50), + qd_real( 2.6671275747489837e-01, 2.0941222578826688e-17, + -1.1303466524727989e-33, 1.9954224050508963e-50), + qd_real( 2.6966832557291509e-01, 1.5765657618133259e-17, + -6.9696142173370086e-34, -4.0455346879146776e-50), + qd_real( 2.7262135544994898e-01, 7.8697166076387850e-18, + 6.6179388602933372e-35, -2.7642903696386267e-51), + qd_real( 2.7557181931095814e-01, 1.9320328962556582e-17, + 1.3932094180100280e-33, 1.3617253920018116e-50), + qd_real( 2.7851968938505312e-01, -1.0030273719543544e-17, + 7.2592115325689254e-34, -1.0068516296655851e-50), + qd_real( 2.8146493792575800e-01, -1.2322299641274009e-17, + -1.0564788706386435e-34, 7.5137424251265885e-51), + qd_real( 2.8440753721127182e-01, 2.2209268510661475e-17, + -9.1823095629523708e-34, -5.2192875308892218e-50), + qd_real( 2.8734745954472951e-01, 1.5461117367645717e-17, + -6.3263973663444076e-34, -2.2982538416476214e-50), + qd_real( 2.9028467725446239e-01, -1.8927978707774251e-17, + 1.1522953157142315e-33, 7.4738655654716596e-50), + qd_real( 2.9321916269425863e-01, 2.2385430811901833e-17, + 1.3662484646539680e-33, -4.2451325253996938e-50), + qd_real( 2.9615088824362384e-01, -2.0220736360876938e-17, + -7.9252212533920413e-35, -2.8990577729572470e-51), + qd_real( 2.9907982630804048e-01, 1.6701181609219447e-18, + 8.6091151117316292e-35, 3.9931286230012102e-52), + qd_real( 3.0200594931922808e-01, -1.7167666235262474e-17, + 2.3336182149008069e-34, 8.3025334555220004e-51), + qd_real( 3.0492922973540243e-01, -2.2989033898191262e-17, + -1.4598901099661133e-34, 3.7760487693121827e-51), + qd_real( 3.0784964004153487e-01, 2.7074088527245185e-17, + 1.2568858206899284e-33, 7.2931815105901645e-50), + qd_real( 3.1076715274961147e-01, 2.0887076364048513e-17, + -3.0130590791065942e-34, 1.3876739009935179e-51), + qd_real( 3.1368174039889146e-01, 1.4560447299968912e-17, + 3.6564186898011595e-34, 1.1654264734999375e-50), + qd_real( 3.1659337555616585e-01, 2.1435292512726283e-17, + 1.2338169231377316e-33, 3.3963542100989293e-50), + qd_real( 3.1950203081601569e-01, -1.3981562491096626e-17, + 8.1730000697411350e-34, -7.7671096270210952e-50), + qd_real( 3.2240767880106985e-01, -4.0519039937959398e-18, + 3.7438302780296796e-34, 8.7936731046639195e-51), + qd_real( 3.2531029216226293e-01, 7.9171249463765892e-18, + -6.7576622068146391e-35, 2.3021655066929538e-51), + qd_real( 3.2820984357909255e-01, -2.6693140719641896e-17, + 7.8928851447534788e-34, 2.5525163821987809e-51), + qd_real( 3.3110630575987643e-01, -2.7469465474778694e-17, + -1.3401245916610206e-33, 6.5531762489976163e-50), + qd_real( 3.3399965144200938e-01, 2.2598986806288142e-17, + 7.8063057192586115e-34, 2.0427600895486683e-50), + qd_real( 3.3688985339222005e-01, -4.2000940033475092e-19, + -2.9178652969985438e-36, -1.1597376437036749e-52), + qd_real( 3.3977688440682685e-01, 6.6028679499418282e-18, + 1.2575009988669683e-34, 2.5569067699008304e-51), + qd_real( 3.4266071731199438e-01, 1.9261518449306319e-17, + -9.2754189135990867e-34, 8.5439996687390166e-50), + qd_real( 3.4554132496398904e-01, 2.7251143672916123e-17, + 7.0138163601941737e-34, -1.4176292197454015e-50), + qd_real( 3.4841868024943456e-01, 3.6974420514204918e-18, + 3.5532146878499996e-34, 1.9565462544501322e-50), + qd_real( 3.5129275608556715e-01, -2.2670712098795844e-17, + -1.6994216673139631e-34, -1.2271556077284517e-50), + qd_real( 3.5416352542049040e-01, -1.6951763305764860e-17, + 1.2772331777814617e-33, -3.3703785435843310e-50), + qd_real( 3.5703096123343003e-01, -4.8218191137919166e-19, + -4.1672436994492361e-35, -7.1531167149364352e-52), + qd_real( 3.5989503653498817e-01, -1.7601687123839282e-17, + 1.3375125473046791e-33, 7.9467815593584340e-50), + qd_real( 3.6275572436739723e-01, -9.1668352663749849e-18, + -7.4317843956936735e-34, -2.0199582511804564e-50), + qd_real( 3.6561299780477385e-01, 1.6217898770457546e-17, + 1.1286970151961055e-33, -7.1825287318139010e-50), + qd_real( 3.6846682995337232e-01, 1.0463640796159268e-17, + 2.0554984738517304e-35, 1.0441861305618769e-51), + qd_real( 3.7131719395183754e-01, 3.4749239648238266e-19, + -7.5151053042866671e-37, -2.8153468438650851e-53), + qd_real( 3.7416406297145799e-01, 8.0114103761962118e-18, + 5.3429599813406052e-34, 1.0351378796539210e-50), + qd_real( 3.7700741021641826e-01, -2.7255302041956930e-18, + 6.3646586445018137e-35, 8.3048657176503559e-52), + qd_real( 3.7984720892405116e-01, 9.9151305855172370e-18, + 4.8761409697224886e-34, 1.4025084000776705e-50), + qd_real( 3.8268343236508978e-01, -1.0050772696461588e-17, + -2.0605316302806695e-34, -1.2717724698085205e-50), + qd_real( 3.8551605384391885e-01, 1.5177665396472313e-17, + 1.4198230518016535e-33, 5.8955167159904235e-50), + qd_real( 3.8834504669882630e-01, -1.0053770598398717e-17, + 7.5942999255057131e-34, -3.1967974046654219e-50), + qd_real( 3.9117038430225387e-01, 1.7997787858243995e-17, + -1.0613482402609856e-33, -5.4582148817791032e-50), + qd_real( 3.9399204006104810e-01, 9.7649241641239336e-18, + -2.1233599441284617e-34, -5.5529836795340819e-51), + qd_real( 3.9680998741671031e-01, 2.0545063670840126e-17, + 6.1347058801922842e-34, 1.0733788150636430e-50), + qd_real( 3.9962419984564684e-01, -1.5065497476189372e-17, + -9.9653258881867298e-34, -5.7524323712725355e-50), + qd_real( 4.0243465085941843e-01, 1.0902619339328270e-17, + 7.3998528125989765e-34, 2.2745784806823499e-50), + qd_real( 4.0524131400498986e-01, 9.9111401942899884e-18, + -2.5169070895434648e-34, 9.2772984818436573e-53), + qd_real( 4.0804416286497869e-01, -7.0006015137351311e-18, + -1.4108207334268228e-34, 1.5175546997577136e-52), + qd_real( 4.1084317105790397e-01, -2.4219835190355499e-17, + -1.1418902925313314e-33, -2.0996843165093468e-50), + qd_real( 4.1363831223843456e-01, -1.0393984940597871e-17, + -1.1481681174503880e-34, -2.0281052851028680e-51), + qd_real( 4.1642956009763721e-01, -2.5475580413131732e-17, + -3.4482678506112824e-34, 7.1788619351865480e-51), + qd_real( 4.1921688836322396e-01, -4.2232463750110590e-18, + -3.6053023045255790e-34, -2.2209673210025631e-50), + qd_real( 4.2200027079979968e-01, 4.3543266994128527e-18, + 3.1734310272251190e-34, -1.3573247980738668e-50), + qd_real( 4.2477968120910881e-01, 2.7462312204277281e-17, + -4.6552847802111948e-34, 6.5961781099193122e-51), + qd_real( 4.2755509343028208e-01, 9.4111898162954726e-18, + -1.7446682426598801e-34, -2.2054492626480169e-51), + qd_real( 4.3032648134008261e-01, 2.2259686974092690e-17, + 8.5972591314085075e-34, -2.9420897889003020e-50), + qd_real( 4.3309381885315196e-01, 1.1224283329847517e-17, + 5.3223748041075651e-35, 5.3926192627014212e-51), + qd_real( 4.3585707992225547e-01, 1.6230515450644527e-17, + -6.4371449063579431e-35, -6.9102436481386757e-51), + qd_real( 4.3861623853852766e-01, -2.0883315831075090e-17, + -1.4259583540891877e-34, 6.3864763590657077e-52), + qd_real( 4.4137126873171667e-01, 2.2360783886964969e-17, + 1.1864769603515770e-34, -3.8087003266189232e-51), + qd_real( 4.4412214457042926e-01, -2.4218874422178315e-17, + 2.2205230838703907e-34, 9.2133035911356258e-51), + qd_real( 4.4686884016237421e-01, -1.9222136142309382e-17, + -4.4425678589732049e-35, -1.3673609292149535e-51), + qd_real( 4.4961132965460660e-01, 4.8831924232035243e-18, + 2.7151084498191381e-34, -1.5653993171613154e-50), + qd_real( 4.5234958723377089e-01, -1.4827977472196122e-17, + -7.6947501088972324e-34, 1.7656856882031319e-50), + qd_real( 4.5508358712634384e-01, -1.2379906758116472e-17, + 5.5289688955542643e-34, -8.5382312840209386e-51), + qd_real( 4.5781330359887723e-01, -8.4554254922295949e-18, + -6.3770394246764263e-34, 3.1778253575564249e-50), + qd_real( 4.6053871095824001e-01, 1.8488777492177872e-17, + -1.0527732154209725e-33, 3.3235593490947102e-50), + qd_real( 4.6325978355186020e-01, -7.3514924533231707e-18, + 6.7175396881707035e-34, 3.9594127612123379e-50), + qd_real( 4.6597649576796618e-01, -3.3023547778235135e-18, + 3.4904677050476886e-35, 3.4483855263874246e-51), + qd_real( 4.6868882203582796e-01, -2.2949251681845054e-17, + -1.1364757641823658e-33, 6.8840522501918612e-50), + qd_real( 4.7139673682599764e-01, 6.5166781360690130e-18, + 2.9457546966235984e-34, -6.2159717738836630e-51), + qd_real( 4.7410021465055002e-01, -8.1451601548978075e-18, + -3.4789448555614422e-34, -1.1681943974658508e-50), + qd_real( 4.7679923006332214e-01, -1.0293515338305794e-17, + -3.6582045008369952e-34, 1.7424131479176475e-50), + qd_real( 4.7949375766015301e-01, 1.8419999662684771e-17, + -1.3040838621273312e-33, 1.0977131822246471e-50), + qd_real( 4.8218377207912277e-01, -2.5861500925520442e-17, + -6.2913197606500007e-36, 4.0802359808684726e-52), + qd_real( 4.8486924800079112e-01, -1.8034004203262245e-17, + -3.5244276906958044e-34, -1.7138318654749246e-50), + qd_real( 4.8755016014843594e-01, 1.4231090931273653e-17, + -1.8277733073262697e-34, -1.5208291790429557e-51), + qd_real( 4.9022648328829116e-01, -5.1496145643440404e-18, + -3.6903027405284104e-34, 1.5172940095151304e-50), + qd_real( 4.9289819222978404e-01, -1.0257831676562186e-18, + 6.9520817760885069e-35, -2.4260961214090389e-51), + qd_real( 4.9556526182577254e-01, -9.4323241942365362e-18, + 3.1212918657699143e-35, 4.2009072375242736e-52), + qd_real( 4.9822766697278187e-01, -1.6126383830540798e-17, + -1.5092897319298871e-33, 1.1049298890895917e-50), + qd_real( 5.0088538261124083e-01, -3.9604015147074639e-17, + -2.2208395201898007e-33, 1.3648202735839417e-49), + qd_real( 5.0353838372571758e-01, -1.6731308204967497e-17, + -1.0140233644074786e-33, 4.0953071937671477e-50), + qd_real( 5.0618664534515534e-01, -4.8321592986493711e-17, + 9.2858107226642252e-34, 4.2699802401037005e-50), + qd_real( 5.0883014254310699e-01, 4.7836968268014130e-17, + -1.0727022928806035e-33, 2.7309374513672757e-50), + qd_real( 5.1146885043797041e-01, -1.3088001221007579e-17, + 4.0929033363366899e-34, -3.7952190153477926e-50), + qd_real( 5.1410274419322177e-01, -4.5712707523615624e-17, + 1.5488279442238283e-33, -2.5853959305521130e-50), + qd_real( 5.1673179901764987e-01, 8.3018617233836515e-18, + 5.8251027467695202e-34, -2.2812397190535076e-50), + qd_real( 5.1935599016558964e-01, -5.5331248144171145e-17, + -3.1628375609769026e-35, -2.4091972051188571e-51), + qd_real( 5.2197529293715439e-01, -4.6555795692088883e-17, + 4.6378980936850430e-34, -3.3470542934689532e-51), + qd_real( 5.2458968267846895e-01, -4.3068869040082345e-17, + -4.2013155291932055e-34, -1.5096069926700274e-50), + qd_real( 5.2719913478190139e-01, -4.2202983480560619e-17, + 8.5585916184867295e-34, 7.9974339336732307e-50), + qd_real( 5.2980362468629472e-01, -4.8067841706482342e-17, + 5.8309721046630296e-34, -8.9740761521756660e-51), + qd_real( 5.3240312787719801e-01, -4.1020306135800895e-17, + -1.9239996374230821e-33, -1.5326987913812184e-49), + qd_real( 5.3499761988709726e-01, -5.3683132708358134e-17, + -1.3900569918838112e-33, 2.7154084726474092e-50), + qd_real( 5.3758707629564551e-01, -2.2617365388403054e-17, + -5.9787279033447075e-34, 3.1204419729043625e-51), + qd_real( 5.4017147272989285e-01, 2.7072447965935839e-17, + 1.1698799709213829e-33, -5.9094668515881500e-50), + qd_real( 5.4275078486451589e-01, 1.7148261004757101e-17, + -1.3525905925200870e-33, 4.9724411290727323e-50), + qd_real( 5.4532498842204646e-01, -4.1517817538384258e-17, + -1.5318930219385941e-33, 6.3629921101413974e-50), + qd_real( 5.4789405917310019e-01, -2.4065878297113363e-17, + -3.5639213669362606e-36, -2.6013270854271645e-52), + qd_real( 5.5045797293660481e-01, -8.3319903015807663e-18, + -2.3058454035767633e-34, -2.1611290432369010e-50), + qd_real( 5.5301670558002758e-01, -4.7061536623798204e-17, + -1.0617111545918056e-33, -1.6196316144407379e-50), + qd_real( 5.5557023301960218e-01, 4.7094109405616768e-17, + -2.0640520383682921e-33, 1.2290163188567138e-49), + qd_real( 5.5811853122055610e-01, 1.3481176324765226e-17, + -5.5016743873011438e-34, -2.3484822739335416e-50), + qd_real( 5.6066157619733603e-01, -7.3956418153476152e-18, + 3.9680620611731193e-34, 3.1995952200836223e-50), + qd_real( 5.6319934401383409e-01, 2.3835775146854829e-17, + 1.3511793173769814e-34, 9.3201311581248143e-51), + qd_real( 5.6573181078361323e-01, -3.4096079596590466e-17, + -1.7073289744303546e-33, 8.9147089975404507e-50), + qd_real( 5.6825895267013160e-01, -5.0935673642769248e-17, + -1.6274356351028249e-33, 9.8183151561702966e-51), + qd_real( 5.7078074588696726e-01, 2.4568151455566208e-17, + -1.2844481247560350e-33, -1.8037634376936261e-50), + qd_real( 5.7329716669804220e-01, 8.5176611669306400e-18, + -6.4443208788026766e-34, 2.2546105543273003e-50), + qd_real( 5.7580819141784534e-01, -3.7909495458942734e-17, + -2.7433738046854309e-33, 1.1130841524216795e-49), + qd_real( 5.7831379641165559e-01, -2.6237691512372831e-17, + 1.3679051680738167e-33, -3.1409808935335900e-50), + qd_real( 5.8081395809576453e-01, 1.8585338586613408e-17, + 2.7673843114549181e-34, 1.9605349619836937e-50), + qd_real( 5.8330865293769829e-01, 3.4516601079044858e-18, + 1.8065977478946306e-34, -6.3953958038544646e-51), + qd_real( 5.8579785745643886e-01, -3.7485501964311294e-18, + 2.7965403775536614e-34, -7.1816936024157202e-51), + qd_real( 5.8828154822264533e-01, -2.9292166725006846e-17, + -2.3744954603693934e-33, -1.1571631191512480e-50), + qd_real( 5.9075970185887428e-01, -4.7013584170659542e-17, + 2.4808417611768356e-33, 1.2598907673643198e-50), + qd_real( 5.9323229503979980e-01, 1.2892320944189053e-17, + 5.3058364776359583e-34, 4.1141674699390052e-50), + qd_real( 5.9569930449243336e-01, -1.3438641936579467e-17, + -6.7877687907721049e-35, -5.6046937531684890e-51), + qd_real( 5.9816070699634227e-01, 3.8801885783000657e-17, + -1.2084165858094663e-33, -4.0456610843430061e-50), + qd_real( 6.0061647938386897e-01, -4.6398198229461932e-17, + -1.6673493003710801e-33, 5.1982824378491445e-50), + qd_real( 6.0306659854034816e-01, 3.7323357680559650e-17, + 2.7771920866974305e-33, -1.6194229649742458e-49), + qd_real( 6.0551104140432555e-01, -3.1202672493305677e-17, + 1.2761267338680916e-33, -4.0859368598379647e-50), + qd_real( 6.0794978496777363e-01, 3.5160832362096660e-17, + -2.5546242776778394e-34, -1.4085313551220694e-50), + qd_real( 6.1038280627630948e-01, -2.2563265648229169e-17, + 1.3185575011226730e-33, 8.2316691420063460e-50), + qd_real( 6.1281008242940971e-01, -4.2693476568409685e-18, + 2.5839965886650320e-34, 1.6884412005622537e-50), + qd_real( 6.1523159058062682e-01, 2.6231417767266950e-17, + -1.4095366621106716e-33, 7.2058690491304558e-50), + qd_real( 6.1764730793780398e-01, -4.7478594510902452e-17, + -7.2986558263123996e-34, -3.0152327517439154e-50), + qd_real( 6.2005721176328921e-01, -2.7983410837681118e-17, + 1.1649951056138923e-33, -5.4539089117135207e-50), + qd_real( 6.2246127937414997e-01, 5.2940728606573002e-18, + -4.8486411215945827e-35, 1.2696527641980109e-52), + qd_real( 6.2485948814238634e-01, 3.3671846037243900e-17, + -2.7846053391012096e-33, 5.6102718120012104e-50), + qd_real( 6.2725181549514408e-01, 3.0763585181253225e-17, + 2.7068930273498138e-34, -1.1172240309286484e-50), + qd_real( 6.2963823891492698e-01, 4.1115334049626806e-17, + -1.9167473580230747e-33, 1.1118424028161730e-49), + qd_real( 6.3201873593980906e-01, -4.0164942296463612e-17, + -7.2208643641736723e-34, 3.7828920470544344e-50), + qd_real( 6.3439328416364549e-01, 1.0420901929280035e-17, + 4.1174558929280492e-34, -1.4464152986630705e-51), + qd_real( 6.3676186123628420e-01, 3.1419048711901611e-17, + -2.2693738415126449e-33, -1.6023584204297388e-49), + qd_real( 6.3912444486377573e-01, 1.2416796312271043e-17, + -6.2095419626356605e-34, 2.7762065999506603e-50), + qd_real( 6.4148101280858316e-01, -9.9883430115943310e-18, + 4.1969230376730128e-34, 5.6980543799257597e-51), + qd_real( 6.4383154288979150e-01, -3.2084798795046886e-17, + -1.2595311907053305e-33, -4.0205885230841536e-50), + qd_real( 6.4617601298331639e-01, -2.9756137382280815e-17, + -1.0275370077518259e-33, 8.0852478665893014e-51), + qd_real( 6.4851440102211244e-01, 3.9870270313386831e-18, + 1.9408388509540788e-34, -5.1798420636193190e-51), + qd_real( 6.5084668499638088e-01, 3.9714670710500257e-17, + 2.9178546787002963e-34, 3.8140635508293278e-51), + qd_real( 6.5317284295377676e-01, 8.5695642060026238e-18, + -6.9165322305070633e-34, 2.3873751224185395e-50), + qd_real( 6.5549285299961535e-01, 3.5638734426385005e-17, + 1.2695365790889811e-33, 4.3984952865412050e-50), + qd_real( 6.5780669329707864e-01, 1.9580943058468545e-17, + -1.1944272256627192e-33, 2.8556402616436858e-50), + qd_real( 6.6011434206742048e-01, -1.3960054386823638e-19, + 6.1515777931494047e-36, 5.3510498875622660e-52), + qd_real( 6.6241577759017178e-01, -2.2615508885764591e-17, + 5.0177050318126862e-34, 2.9162532399530762e-50), + qd_real( 6.6471097820334490e-01, -3.6227793598034367e-17, + -9.0607934765540427e-34, 3.0917036342380213e-50), + qd_real( 6.6699992230363747e-01, 3.5284364997428166e-17, + -1.0382057232458238e-33, 7.3812756550167626e-50), + qd_real( 6.6928258834663612e-01, -5.4592652417447913e-17, + -2.5181014709695152e-33, -1.6867875999437174e-49), + qd_real( 6.7155895484701844e-01, -4.0489037749296692e-17, + 3.1995835625355681e-34, -1.4044414655670960e-50), + qd_real( 6.7382900037875604e-01, 2.3091901236161086e-17, + 5.7428037192881319e-34, 1.1240668354625977e-50), + qd_real( 6.7609270357531592e-01, 3.7256902248049466e-17, + 1.7059417895764375e-33, 9.7326347795300652e-50), + qd_real( 6.7835004312986147e-01, 1.8302093041863122e-17, + 9.5241675746813072e-34, 5.0328101116133503e-50), + qd_real( 6.8060099779545302e-01, 2.8473293354522047e-17, + 4.1331805977270903e-34, 4.2579030510748576e-50), + qd_real( 6.8284554638524808e-01, -1.2958058061524531e-17, + 1.8292386959330698e-34, 3.4536209116044487e-51), + qd_real( 6.8508366777270036e-01, 2.5948135194645137e-17, + -8.5030743129500702e-34, -6.9572086141009930e-50), + qd_real( 6.8731534089175916e-01, -5.5156158714917168e-17, + 1.1896489854266829e-33, -7.8505896218220662e-51), + qd_real( 6.8954054473706694e-01, -1.5889323294806790e-17, + 9.1242356240205712e-34, 3.8315454152267638e-50), + qd_real( 6.9175925836415775e-01, 2.7406078472410668e-17, + 1.3286508943202092e-33, 1.0651869129580079e-51), + qd_real( 6.9397146088965400e-01, 7.4345076956280137e-18, + 7.5061528388197460e-34, -1.5928000240686583e-50), + qd_real( 6.9617713149146299e-01, -4.1224081213582889e-17, + -3.1838716762083291e-35, -3.9625587412119131e-51), + qd_real( 6.9837624940897280e-01, 4.8988282435667768e-17, + 1.9134010413244152e-33, 2.6161153243793989e-50), + qd_real( 7.0056879394324834e-01, 3.1027960192992922e-17, + 9.5638250509179997e-34, 4.5896916138107048e-51), + qd_real( 7.0275474445722530e-01, 2.5278294383629822e-18, + -8.6985561210674942e-35, -5.6899862307812990e-51), + qd_real( 7.0493408037590488e-01, 2.7608725585748502e-17, + 2.9816599471629137e-34, 1.1533044185111206e-50), + qd_real( 7.0710678118654757e-01, -4.8336466567264567e-17, + 2.0693376543497068e-33, 2.4677734957341755e-50) +}; + +static const qd_real cos_table [] = { + qd_real( 9.9999529380957619e-01, -1.9668064285322189e-17, + -6.3053955095883481e-34, 5.3266110855726731e-52), + qd_real( 9.9998117528260111e-01, 3.3568103522895585e-17, + -1.4740132559368063e-35, 9.8603097594755596e-52), + qd_real( 9.9995764455196390e-01, -3.1527836866647287e-17, + 2.6363251186638437e-33, 1.0007504815488399e-49), + qd_real( 9.9992470183914450e-01, 3.7931082512668012e-17, + -8.5099918660501484e-35, -4.9956973223295153e-51), + qd_real( 9.9988234745421256e-01, -3.5477814872408538e-17, + 1.7102001035303974e-33, -1.0725388519026542e-49), + qd_real( 9.9983058179582340e-01, 1.8825140517551119e-17, + -5.1383513457616937e-34, -3.8378827995403787e-50), + qd_real( 9.9976940535121528e-01, 4.2681177032289012e-17, + 1.9062302359737099e-33, -6.0221153262881160e-50), + qd_real( 9.9969881869620425e-01, -2.9851486403799753e-17, + -1.9084787370733737e-33, 5.5980260344029202e-51), + qd_real( 9.9961882249517864e-01, -4.1181965521424734e-17, + 2.0915365593699916e-33, 8.1403390920903734e-50), + qd_real( 9.9952941750109314e-01, 2.0517917823755591e-17, + -4.7673802585706520e-34, -2.9443604198656772e-50), + qd_real( 9.9943060455546173e-01, 3.9644497752257798e-17, + -2.3757223716722428e-34, -1.2856759011361726e-51), + qd_real( 9.9932238458834954e-01, -4.2858538440845682e-17, + 3.3235101605146565e-34, -8.3554272377057543e-51), + qd_real( 9.9920475861836389e-01, 9.1796317110385693e-18, + 5.5416208185868570e-34, 8.0267046717615311e-52), + qd_real( 9.9907772775264536e-01, 2.1419007653587032e-17, + -7.9048203318529618e-34, -5.3166296181112712e-50), + qd_real( 9.9894129318685687e-01, -2.0610641910058638e-17, + -1.2546525485913485e-33, -7.5175888806157064e-50), + qd_real( 9.9879545620517241e-01, -1.2291693337075465e-17, + 2.4468446786491271e-34, 1.0723891085210268e-50), + qd_real( 9.9864021818026527e-01, -4.8690254312923302e-17, + -2.9470881967909147e-34, -1.3000650761346907e-50), + qd_real( 9.9847558057329477e-01, -2.2002931182778795e-17, + -1.2371509454944992e-33, -2.4911225131232065e-50), + qd_real( 9.9830154493389289e-01, -5.1869402702792278e-17, + 1.0480195493633452e-33, -2.8995649143155511e-50), + qd_real( 9.9811811290014918e-01, 2.7935487558113833e-17, + 2.4423341255830345e-33, -6.7646699175334417e-50), + qd_real( 9.9792528619859600e-01, 1.7143659778886362e-17, + 5.7885840902887460e-34, -9.2601432603894597e-51), + qd_real( 9.9772306664419164e-01, -2.6394475274898721e-17, + -1.6176223087661783e-34, -9.9924942889362281e-51), + qd_real( 9.9751145614030345e-01, 5.6007205919806937e-18, + -5.9477673514685690e-35, -1.4166807162743627e-54), + qd_real( 9.9729045667869021e-01, 9.1647695371101735e-18, + 6.7824134309739296e-34, -8.6191392795543357e-52), + qd_real( 9.9706007033948296e-01, 1.6734093546241963e-17, + -1.3169951440780028e-33, 1.0311048767952477e-50), + qd_real( 9.9682029929116567e-01, 4.7062820708615655e-17, + 2.8412041076474937e-33, -8.0006155670263622e-50), + qd_real( 9.9657114579055484e-01, 1.1707179088390986e-17, + -7.5934413263024663e-34, 2.8474848436926008e-50), + qd_real( 9.9631261218277800e-01, 1.1336497891624735e-17, + 3.4002458674414360e-34, 7.7419075921544901e-52), + qd_real( 9.9604470090125197e-01, 2.2870031707670695e-17, + -3.9184839405013148e-34, -3.7081260416246375e-50), + qd_real( 9.9576741446765982e-01, -2.3151908323094359e-17, + -1.6306512931944591e-34, -1.5925420783863192e-51), + qd_real( 9.9548075549192694e-01, 3.2084621412226554e-18, + -4.9501292146013023e-36, -2.7811428850878516e-52), + qd_real( 9.9518472667219693e-01, -4.2486913678304410e-17, + 1.3315510772504614e-33, 6.7927987417051888e-50), + qd_real( 9.9487933079480562e-01, 4.2130813284943662e-18, + -4.2062597488288452e-35, 2.5157064556087620e-51), + qd_real( 9.9456457073425542e-01, 3.6745069641528058e-17, + -3.0603304105471010e-33, 1.0397872280487526e-49), + qd_real( 9.9424044945318790e-01, 4.4129423472462673e-17, + -3.0107231708238066e-33, 7.4201582906861892e-50), + qd_real( 9.9390697000235606e-01, -1.8964849471123746e-17, + -1.5980853777937752e-35, -8.5374807150597082e-52), + qd_real( 9.9356413552059530e-01, 2.9752309927797428e-17, + -4.5066707331134233e-34, -3.3548191633805036e-50), + qd_real( 9.9321194923479450e-01, 3.3096906261272262e-17, + 1.5592811973249567e-33, 1.4373977733253592e-50), + qd_real( 9.9285041445986510e-01, -1.4094517733693302e-17, + -1.1954558131616916e-33, 1.8761873742867983e-50), + qd_real( 9.9247953459870997e-01, 3.1093055095428906e-17, + -1.8379594757818019e-33, -3.9885758559381314e-51), + qd_real( 9.9209931314219180e-01, -3.9431926149588778e-17, + -6.2758062911047230e-34, -1.2960929559212390e-50), + qd_real( 9.9170975366909953e-01, -2.3372891311883661e-18, + 2.7073298824968591e-35, -1.2569459441802872e-51), + qd_real( 9.9131085984611544e-01, -2.5192111583372105e-17, + -1.2852471567380887e-33, 5.2385212584310483e-50), + qd_real( 9.9090263542778001e-01, 1.5394565094566704e-17, + -1.0799984133184567e-33, 2.7451115960133595e-51), + qd_real( 9.9048508425645709e-01, -5.5411437553780867e-17, + -1.4614017210753585e-33, -3.8339374397387620e-50), + qd_real( 9.9005821026229712e-01, -1.7055485906233963e-17, + 1.3454939685758777e-33, 7.3117589137300036e-50), + qd_real( 9.8962201746320089e-01, -5.2398217968132530e-17, + 1.3463189211456219e-33, 5.8021640554894872e-50), + qd_real( 9.8917650996478101e-01, -4.0987309937047111e-17, + -4.4857560552048437e-34, -3.9414504502871125e-50), + qd_real( 9.8872169196032378e-01, -1.0976227206656125e-17, + 3.2311342577653764e-34, 9.6367946583575041e-51), + qd_real( 9.8825756773074946e-01, 2.7030607784372632e-17, + 7.7514866488601377e-35, 2.1019644956864938e-51), + qd_real( 9.8778414164457218e-01, -2.3600693397159021e-17, + -1.2323283769707861e-33, 3.0130900716803339e-50), + qd_real( 9.8730141815785843e-01, -5.2332261255715652e-17, + -2.7937644333152473e-33, 1.2074160567958408e-49), + qd_real( 9.8680940181418553e-01, -5.0287214351061075e-17, + -2.2681526238144461e-33, 4.4003694320169133e-50), + qd_real( 9.8630809724459867e-01, -2.1520877103013341e-17, + 1.1866528054187716e-33, -7.8532199199813836e-50), + qd_real( 9.8579750916756748e-01, -5.1439452979953012e-17, + 2.6276439309996725e-33, 7.5423552783286347e-50), + qd_real( 9.8527764238894122e-01, 2.3155637027900207e-17, + -7.5275971545764833e-34, 1.0582231660456094e-50), + qd_real( 9.8474850180190421e-01, 1.0548144061829957e-17, + 2.8786145266267306e-34, -3.6782210081466112e-51), + qd_real( 9.8421009238692903e-01, 4.7983922627050691e-17, + 2.2597419645070588e-34, 1.7573875814863400e-50), + qd_real( 9.8366241921173025e-01, 1.9864948201635255e-17, + -1.0743046281211033e-35, 1.7975662796558100e-52), + qd_real( 9.8310548743121629e-01, 4.2170007522888628e-17, + 8.2396265656440904e-34, -8.0803700139096561e-50), + qd_real( 9.8253930228744124e-01, 1.5149580813777224e-17, + -4.1802771422186237e-34, -2.2150174326226160e-50), + qd_real( 9.8196386910955524e-01, 2.1108443711513084e-17, + -1.5253013442896054e-33, -6.8388082079337969e-50), + qd_real( 9.8137919331375456e-01, 1.3428163260355633e-17, + -6.5294290469962986e-34, 2.7965412287456268e-51), + qd_real( 9.8078528040323043e-01, 1.8546939997825006e-17, + -1.0696564445530757e-33, 6.6668174475264961e-50), + qd_real( 9.8018213596811743e-01, -3.6801786963856159e-17, + 6.3245171387992842e-34, 1.8600176137175971e-50), + qd_real( 9.7956976568544052e-01, 1.5573991584990420e-17, + -1.3401066029782990e-33, -1.7263702199862149e-50), + qd_real( 9.7894817531906220e-01, -2.3817727961148053e-18, + -1.0694750370381661e-34, -8.2293047196087462e-51), + qd_real( 9.7831737071962765e-01, -2.1623082233344895e-17, + 1.0970403012028032e-33, 7.7091923099369339e-50), + qd_real( 9.7767735782450993e-01, 5.0514136167059628e-17, + -1.3254751701428788e-33, 7.0161254312124538e-50), + qd_real( 9.7702814265775439e-01, -4.3353875751555997e-17, + 5.4948839831535478e-34, -9.2755263105377306e-51), + qd_real( 9.7636973133002114e-01, 9.3093931526213780e-18, + -4.1184949155685665e-34, -3.1913926031393690e-50), + qd_real( 9.7570213003852857e-01, -2.5572556081259686e-17, + -9.3174244508942223e-34, -8.3675863211646863e-51), + qd_real( 9.7502534506699412e-01, 2.6642660651899135e-17, + 1.7819392739353853e-34, -3.3159625385648947e-51), + qd_real( 9.7433938278557586e-01, 2.3041221476151512e-18, + 1.0758686005031430e-34, 5.1074116432809478e-51), + qd_real( 9.7364424965081198e-01, -5.1729808691005871e-17, + -1.5508473005989887e-33, -1.6505125917675401e-49), + qd_real( 9.7293995220556018e-01, -3.1311211122281800e-17, + -2.6874087789006141e-33, -2.1652434818822145e-51), + qd_real( 9.7222649707893627e-01, 3.6461169785938221e-17, + 3.0309636883883133e-33, -1.2702716907967306e-51), + qd_real( 9.7150389098625178e-01, -7.9865421122289046e-18, + -4.3628417211263380e-34, 3.4307517798759352e-51), + qd_real( 9.7077214072895035e-01, -4.7992163325114922e-17, + 3.0347528910975783e-33, 8.5989199506479701e-50), + qd_real( 9.7003125319454397e-01, 1.8365300348428844e-17, + -1.4311097571944918e-33, 8.5846781998740697e-51), + qd_real( 9.6928123535654853e-01, -4.5663660261927896e-17, + 9.6147526917239387e-34, 8.1267605207871330e-51), + qd_real( 9.6852209427441727e-01, 4.9475074918244771e-17, + 2.8558738351911241e-33, 6.2948422316507461e-50), + qd_real( 9.6775383709347551e-01, -4.5512132825515820e-17, + -1.4127617988719093e-33, -8.4620609089704578e-50), + qd_real( 9.6697647104485207e-01, 3.8496228837337864e-17, + -5.3881631542745647e-34, -3.5221863171458959e-50), + qd_real( 9.6619000344541250e-01, 5.1298840401665493e-17, + 1.4564075904769808e-34, 1.0095973971377432e-50), + qd_real( 9.6539444169768940e-01, -2.3745389918392156e-17, + 5.9221515590053862e-34, -3.8811192556231094e-50), + qd_real( 9.6458979328981276e-01, -3.4189470735959786e-17, + 2.2982074155463522e-33, -4.5128791045607634e-50), + qd_real( 9.6377606579543984e-01, 2.6463950561220029e-17, + -2.9073234590199323e-36, -1.2938328629395601e-52), + qd_real( 9.6295326687368388e-01, 8.9341960404313634e-18, + -3.9071244661020126e-34, 1.6212091116847394e-50), + qd_real( 9.6212140426904158e-01, 1.5236770453846305e-17, + -1.3050173525597142e-33, 7.9016122394092666e-50), + qd_real( 9.6128048581132064e-01, 2.0933955216674039e-18, + 1.0768607469015692e-34, -5.9453639304361774e-51), + qd_real( 9.6043051941556579e-01, 2.4653904815317185e-17, + -1.3792169410906322e-33, -4.7726598378506903e-51), + qd_real( 9.5957151308198452e-01, 1.1000640085000957e-17, + -4.2036030828223975e-34, 4.0023704842606573e-51), + qd_real( 9.5870347489587160e-01, -4.3685014392372053e-17, + 2.2001800662729131e-33, -1.0553721324358075e-49), + qd_real( 9.5782641302753291e-01, -1.7696710075371263e-17, + 1.9164034110382190e-34, 8.1489235071754813e-51), + qd_real( 9.5694033573220882e-01, 4.0553869861875701e-17, + -1.7147013364302149e-33, 2.5736745295329455e-50), + qd_real( 9.5604525134999641e-01, 3.7705045279589067e-17, + 1.9678699997347571e-33, 8.5093177731230180e-50), + qd_real( 9.5514116830577067e-01, 5.0088652955014668e-17, + -2.6983181838059211e-33, 1.0102323575596493e-49), + qd_real( 9.5422809510910567e-01, -3.7545901690626874e-17, + 1.4951619241257764e-33, -8.2717333151394973e-50), + qd_real( 9.5330604035419386e-01, -2.5190738779919934e-17, + -1.4272239821134379e-33, -4.6717286809283155e-50), + qd_real( 9.5237501271976588e-01, -2.0269300462299272e-17, + -1.0635956887246246e-33, -3.5514537666487619e-50), + qd_real( 9.5143502096900834e-01, 3.1350584123266695e-17, + -2.4824833452737813e-33, 9.5450335525380613e-51), + qd_real( 9.5048607394948170e-01, 1.9410097562630436e-17, + -8.1559393949816789e-34, -1.0501209720164562e-50), + qd_real( 9.4952818059303667e-01, -7.5544151928043298e-18, + -5.1260245024046686e-34, 1.8093643389040406e-50), + qd_real( 9.4856134991573027e-01, 2.0668262262333232e-17, + -5.9440730243667306e-34, 1.4268853111554300e-50), + qd_real( 9.4758559101774109e-01, 4.3417993852125991e-17, + -2.7728667889840373e-34, 5.5709160196519968e-51), + qd_real( 9.4660091308328353e-01, 3.5056800210680730e-17, + 9.8578536940318117e-34, 6.6035911064585197e-50), + qd_real( 9.4560732538052128e-01, 4.6019102478523738e-17, + -6.2534384769452059e-34, 1.5758941215779961e-50), + qd_real( 9.4460483726148026e-01, 8.8100545476641165e-18, + 5.2291695602757842e-34, -3.3487256018407123e-50), + qd_real( 9.4359345816196039e-01, -2.4093127844404214e-17, + 1.0283279856803939e-34, -2.3398232614531355e-51), + qd_real( 9.4257319760144687e-01, 1.3235564806436886e-17, + -5.7048262885386911e-35, 3.9947050442753744e-51), + qd_real( 9.4154406518302081e-01, -2.7896379547698341e-17, + 1.6273236356733898e-33, -5.3075944708471203e-51), + qd_real( 9.4050607059326830e-01, 2.8610421567116268e-17, + 2.9261501147538827e-33, -2.6849867690896925e-50), + qd_real( 9.3945922360218992e-01, -7.0152867943098655e-18, + -5.6395693818011210e-34, 3.5568142678987651e-50), + qd_real( 9.3840353406310806e-01, 5.4242545044795490e-17, + -1.9039966607859759e-33, -1.5627792988341215e-49), + qd_real( 9.3733901191257496e-01, -3.6570926284362776e-17, + -1.1902940071273247e-33, -1.1215082331583223e-50), + qd_real( 9.3626566717027826e-01, -1.3013766145497654e-17, + 5.2229870061990595e-34, -3.3972777075634108e-51), + qd_real( 9.3518350993894761e-01, -3.2609395302485065e-17, + -8.1813015218875245e-34, 5.5642140024928139e-50), + qd_real( 9.3409255040425887e-01, 4.4662824360767511e-17, + -2.5903243047396916e-33, 8.1505209004343043e-50), + qd_real( 9.3299279883473885e-01, 4.2041415555384355e-17, + 9.0285896495521276e-34, 5.3019984977661259e-50), + qd_real( 9.3188426558166815e-01, -4.0785944377318095e-17, + 1.7631450298754169e-33, 2.5776403305507453e-50), + qd_real( 9.3076696107898371e-01, 1.9703775102838329e-17, + 6.5657908718278205e-34, -1.9480347966259524e-51), + qd_real( 9.2964089584318121e-01, 5.1282530016864107e-17, + 2.3719739891916261e-34, -1.7230065426917127e-50), + qd_real( 9.2850608047321559e-01, -2.3306639848485943e-17, + -7.7799084333208503e-34, -5.8597558009300305e-50), + qd_real( 9.2736252565040111e-01, -2.7677111692155437e-17, + 2.2110293450199576e-34, 2.0349190819680613e-50), + qd_real( 9.2621024213831138e-01, -3.7303754586099054e-17, + 2.0464457809993405e-33, 1.3831799631231817e-49), + qd_real( 9.2504924078267758e-01, 6.0529447412576159e-18, + -8.8256517760278541e-35, 1.8285462122388328e-51), + qd_real( 9.2387953251128674e-01, 1.7645047084336677e-17, + -5.0442537321586818e-34, -4.0478677716823890e-50), + qd_real( 9.2270112833387852e-01, 5.2963798918539814e-17, + -5.7135699628876685e-34, 3.0163671797219087e-50), + qd_real( 9.2151403934204190e-01, 4.1639843390684644e-17, + 1.1891485604702356e-33, 2.0862437594380324e-50), + qd_real( 9.2031827670911059e-01, -2.7806888779036837e-17, + 2.7011013677071274e-33, 1.1998578792455499e-49), + qd_real( 9.1911385169005777e-01, -2.6496484622344718e-17, + 6.5403604763461920e-34, -2.8997180201186078e-50), + qd_real( 9.1790077562139050e-01, -3.9074579680849515e-17, + 2.3004636541490264e-33, 3.9851762744443107e-50), + qd_real( 9.1667905992104270e-01, -4.1733978698287568e-17, + 1.2094444804381172e-33, 4.9356916826097816e-50), + qd_real( 9.1544871608826783e-01, -1.3591056692900894e-17, + 5.9923027475594735e-34, 2.1403295925962879e-50), + qd_real( 9.1420975570353069e-01, -3.6316182527814423e-17, + -1.9438819777122554e-33, 2.8340679287728316e-50), + qd_real( 9.1296219042839821e-01, -4.7932505228039469e-17, + -1.7753551889428638e-33, 4.0607782903868160e-51), + qd_real( 9.1170603200542988e-01, -2.6913273175034130e-17, + -5.1928101916162528e-35, 1.1338175936090630e-51), + qd_real( 9.1044129225806725e-01, -5.0433041673313820e-17, + 1.0938746257404305e-33, 9.5378272084170731e-51), + qd_real( 9.0916798309052238e-01, -3.6878564091359894e-18, + 2.9951330310507693e-34, -1.2225666136919926e-50), + qd_real( 9.0788611648766626e-01, -4.9459964301225840e-17, + -1.6599682707075313e-33, -5.1925202712634716e-50), + qd_real( 9.0659570451491533e-01, 3.0506718955442023e-17, + -1.4478836557141204e-33, 1.8906373784448725e-50), + qd_real( 9.0529675931811882e-01, -4.1153099826889901e-17, + 2.9859368705184223e-33, 5.1145293917439211e-50), + qd_real( 9.0398929312344334e-01, -6.6097544687484308e-18, + 1.2728013034680357e-34, -4.3026097234014823e-51), + qd_real( 9.0267331823725883e-01, -1.9250787033961483e-17, + 1.3242128993244527e-33, -5.2971030688703665e-50), + qd_real( 9.0134884704602203e-01, -1.3524789367698682e-17, + 6.3605353115880091e-34, 3.6227400654573828e-50), + qd_real( 9.0001589201616028e-01, -5.0639618050802273e-17, + 1.0783525384031576e-33, 2.8130016326515111e-50), + qd_real( 8.9867446569395382e-01, 2.6316906461033013e-17, + 3.7003137047796840e-35, -2.3447719900465938e-51), + qd_real( 8.9732458070541832e-01, -3.6396283314867290e-17, + -2.3611649895474815e-33, 1.1837247047900082e-49), + qd_real( 8.9596624975618511e-01, 4.9025099114811813e-17, + -1.9440489814795326e-33, -1.7070486667767033e-49), + qd_real( 8.9459948563138270e-01, -1.7516226396814919e-17, + -1.3200670047246923e-33, -1.5953009884324695e-50), + qd_real( 8.9322430119551532e-01, -4.1161239151908913e-18, + 2.5380253805715999e-34, 4.2849455510516192e-51), + qd_real( 8.9184070939234272e-01, 4.6690228137124547e-18, + 1.6150254286841982e-34, -3.9617448820725012e-51), + qd_real( 8.9044872324475788e-01, 1.1781931459051803e-17, + -1.3346142209571930e-34, -9.4982373530733431e-51), + qd_real( 8.8904835585466457e-01, -1.1164514966766675e-17, + -3.4797636107798736e-34, -1.5605079997040631e-50), + qd_real( 8.8763962040285393e-01, 1.2805091918587960e-17, + 3.9948742059584459e-35, 3.8940716325338136e-51), + qd_real( 8.8622253014888064e-01, -6.7307369600274315e-18, + 1.2385593432917413e-34, 2.0364014759133320e-51), + qd_real( 8.8479709843093779e-01, -9.4331469628972690e-18, + -5.7106541478701439e-34, 1.8260134111907397e-50), + qd_real( 8.8336333866573158e-01, 1.5822643380255127e-17, + -7.8921320007588250e-34, -1.4782321016179836e-50), + qd_real( 8.8192126434835505e-01, -1.9843248405890562e-17, + -7.0412114007673834e-34, -1.0636770169389104e-50), + qd_real( 8.8047088905216075e-01, 1.6311096602996350e-17, + -5.7541360594724172e-34, -4.0128611862170021e-50), + qd_real( 8.7901222642863353e-01, -4.7356837291118011e-17, + 1.4388771297975192e-33, -2.9085554304479134e-50), + qd_real( 8.7754529020726124e-01, 5.0113311846499550e-17, + 2.8382769008739543e-34, 1.5550640393164140e-50), + qd_real( 8.7607009419540660e-01, 5.8729024235147677e-18, + 2.7941144391738458e-34, -1.8536073846509828e-50), + qd_real( 8.7458665227817611e-01, -5.7216617730397065e-19, + -2.9705811503689596e-35, 8.7389593969796752e-52), + qd_real( 8.7309497841829009e-01, 7.8424672990129903e-18, + -4.8685015839797165e-34, -2.2815570587477527e-50), + qd_real( 8.7159508665595109e-01, -5.5272998038551050e-17, + -2.2104090204984907e-33, -9.7749763187643172e-50), + qd_real( 8.7008699110871146e-01, -4.1888510868549968e-17, + 7.0900185861878415e-34, 3.7600251115157260e-50), + qd_real( 8.6857070597134090e-01, 2.7192781689782903e-19, + -1.6710140396932428e-35, -1.2625514734637969e-51), + qd_real( 8.6704624551569265e-01, 3.0267859550930567e-18, + -1.1559438782171572e-34, -5.3580556397808012e-52), + qd_real( 8.6551362409056909e-01, -6.3723113549628899e-18, + 2.3725520321746832e-34, 1.5911880348395175e-50), + qd_real( 8.6397285612158670e-01, 4.1486355957361607e-17, + 2.2709976932210266e-33, -8.1228385659479984e-50), + qd_real( 8.6242395611104050e-01, 3.7008992527383130e-17, + 5.2128411542701573e-34, 2.6945600081026861e-50), + qd_real( 8.6086693863776731e-01, -3.0050048898573656e-17, + -8.8706183090892111e-34, 1.5005320558097301e-50), + qd_real( 8.5930181835700836e-01, 4.2435655816850687e-17, + 7.6181814059912025e-34, -3.9592127850658708e-50), + qd_real( 8.5772861000027212e-01, -4.8183447936336620e-17, + -1.1044130517687532e-33, -8.7400233444645562e-50), + qd_real( 8.5614732837519447e-01, 9.1806925616606261e-18, + 5.6328649785951470e-34, 2.3326646113217378e-51), + qd_real( 8.5455798836540053e-01, -1.2991124236396092e-17, + 1.2893407722948080e-34, -3.6506925747583053e-52), + qd_real( 8.5296060493036363e-01, 2.7152984251981370e-17, + 7.4336483283120719e-34, 4.2162417622350668e-50), + qd_real( 8.5135519310526520e-01, -5.3279874446016209e-17, + 2.2281156380919942e-33, -4.0281886404138477e-50), + qd_real( 8.4974176800085244e-01, 5.1812347659974015e-17, + 3.0810626087331275e-33, -2.5931308201994965e-50), + qd_real( 8.4812034480329723e-01, 1.8762563415239981e-17, + 1.4048773307919617e-33, -2.4915221509958691e-50), + qd_real( 8.4649093877405213e-01, -4.7969419958569345e-17, + -2.7518267097886703e-33, -7.3518959727313350e-50), + qd_real( 8.4485356524970712e-01, -4.3631360296879637e-17, + -2.0307726853367547e-33, 4.3097229819851761e-50), + qd_real( 8.4320823964184544e-01, 9.6536707005959077e-19, + 2.8995142431556364e-36, 9.6715076811480284e-53), + qd_real( 8.4155497743689844e-01, -3.4095465391321557e-17, + -8.4130208607579595e-34, -4.9447283960568686e-50), + qd_real( 8.3989379419599952e-01, -1.6673694881511411e-17, + -1.4759184141750289e-33, -7.5795098161914058e-50), + qd_real( 8.3822470555483808e-01, -3.5560085052855026e-17, + 1.1689791577022643e-33, -5.8627347359723411e-50), + qd_real( 8.3654772722351201e-01, -2.0899059027066533e-17, + -9.8104097821002585e-35, -3.1609177868229853e-51), + qd_real( 8.3486287498638001e-01, 4.6048430609159657e-17, + -5.1827423265239912e-34, -7.0505343435504109e-51), + qd_real( 8.3317016470191319e-01, 1.3275129507229764e-18, + 4.8589164115370863e-35, 4.5422281300506859e-51), + qd_real( 8.3146961230254524e-01, 1.4073856984728024e-18, + 4.6951315383980830e-35, 5.1431906049905658e-51), + qd_real( 8.2976123379452305e-01, -2.9349109376485597e-18, + 1.1496917934149818e-34, 3.5186665544980233e-51), + qd_real( 8.2804504525775580e-01, -4.4196593225871532e-17, + 2.7967864855211251e-33, 1.0030777287393502e-49), + qd_real( 8.2632106284566353e-01, -5.3957485453612902e-17, + 6.8976896130138550e-34, 3.8106164274199196e-50), + qd_real( 8.2458930278502529e-01, -2.6512360488868275e-17, + 1.6916964350914386e-34, 6.7693974813562649e-51), + qd_real( 8.2284978137582632e-01, 1.5193019034505495e-17, + 9.6890547246521685e-34, 5.6994562923653264e-50), + qd_real( 8.2110251499110465e-01, 3.0715131609697682e-17, + -1.7037168325855879e-33, -1.1149862443283853e-49), + qd_real( 8.1934752007679701e-01, -4.8200736995191133e-17, + -1.5574489646672781e-35, -9.5647853614522216e-53), + qd_real( 8.1758481315158371e-01, -1.4883149812426772e-17, + -7.8273262771298917e-34, 4.1332149161031594e-50), + qd_real( 8.1581441080673378e-01, 8.2652693782130871e-18, + -2.3028778135179471e-34, 1.5102071387249843e-50), + qd_real( 8.1403632970594841e-01, -5.2127351877042624e-17, + -1.9047670611316360e-33, -1.6937269585941507e-49), + qd_real( 8.1225058658520388e-01, 3.1054545609214803e-17, + 2.2649541922707251e-34, -7.4221684154649405e-51), + qd_real( 8.1045719825259477e-01, 2.3520367349840499e-17, + -7.7530070904846341e-34, -7.2792616357197140e-50), + qd_real( 8.0865618158817498e-01, 9.3251597879721674e-18, + -7.1823301933068394e-34, 2.3925440846132106e-50), + qd_real( 8.0684755354379922e-01, 4.9220603766095546e-17, + 2.9796016899903487e-33, 1.5220754223615788e-49), + qd_real( 8.0503133114296355e-01, 5.1368289568212149e-17, + 6.3082807402256524e-34, 7.3277646085129827e-51), + qd_real( 8.0320753148064494e-01, -3.3060609804814910e-17, + -1.2242726252420433e-33, 2.8413673268630117e-50), + qd_real( 8.0137617172314024e-01, -2.0958013413495834e-17, + -4.3798162198006931e-34, 2.0235690497752515e-50), + qd_real( 7.9953726910790501e-01, 2.0356723822005431e-17, + -9.7448513696896360e-34, 5.3608109599696008e-52), + qd_real( 7.9769084094339116e-01, -4.6730759884788944e-17, + 2.3075897077191757e-33, 3.1605567774640253e-51), + qd_real( 7.9583690460888357e-01, -3.0062724851910721e-17, + -2.2496210832042235e-33, -6.5881774117183040e-50), + qd_real( 7.9397547755433717e-01, -7.4194631759921416e-18, + 2.4124341304631069e-34, -4.9956808616244972e-51), + qd_real( 7.9210657730021239e-01, -3.7087850202326467e-17, + -1.4874457267228264e-33, 2.9323097289153505e-50), + qd_real( 7.9023022143731003e-01, 2.3056905954954492e-17, + 1.4481080533260193e-33, -7.6725237057203488e-50), + qd_real( 7.8834642762660623e-01, 3.4396993154059708e-17, + 1.7710623746737170e-33, 1.7084159098417402e-49), + qd_real( 7.8645521359908577e-01, -9.7841429939305265e-18, + 3.3906063272445472e-34, 5.7269505320382577e-51), + qd_real( 7.8455659715557524e-01, -8.5627965423173476e-18, + -2.1106834459001849e-34, -1.6890322182469603e-50), + qd_real( 7.8265059616657573e-01, 9.0745866975808825e-18, + 6.7623847404278666e-34, -1.7173237731987271e-50), + qd_real( 7.8073722857209449e-01, -9.9198782066678806e-18, + -2.1265794012162715e-36, 3.0772165598957647e-54), + qd_real( 7.7881651238147598e-01, -2.4891385579973807e-17, + 6.7665497024807980e-35, -6.5218594281701332e-52), + qd_real( 7.7688846567323244e-01, 7.7418602570672864e-18, + -5.9986517872157897e-34, 3.0566548232958972e-50), + qd_real( 7.7495310659487393e-01, -5.2209083189826433e-17, + -9.6653593393686612e-34, 3.7027750076562569e-50), + qd_real( 7.7301045336273699e-01, -3.2565907033649772e-17, + 1.3860807251523929e-33, -3.9971329917586022e-50), + qd_real( 7.7106052426181382e-01, -4.4558442347769265e-17, + -2.9863565614083783e-33, -6.8795262083596236e-50), + qd_real( 7.6910333764557959e-01, 5.1546455184564817e-17, + 2.6142829553524292e-33, -1.6199023632773298e-49), + qd_real( 7.6713891193582040e-01, -1.8885903683750782e-17, + -1.3659359331495433e-33, -2.2538834962921934e-50), + qd_real( 7.6516726562245896e-01, -3.2707225612534598e-17, + 1.1177117747079528e-33, -3.7005182280175715e-50), + qd_real( 7.6318841726338127e-01, 2.6314748416750748e-18, + 1.4048039063095910e-34, 8.9601886626630321e-52), + qd_real( 7.6120238548426178e-01, 3.5315510881690551e-17, + 1.2833566381864357e-33, 8.6221435180890613e-50), + qd_real( 7.5920918897838807e-01, -3.8558842175523123e-17, + 2.9720241208332759e-34, -1.2521388928220163e-50), + qd_real( 7.5720884650648457e-01, -1.9909098777335502e-17, + 3.9409283266158482e-34, 2.0744254207802976e-50), + qd_real( 7.5520137689653655e-01, -1.9402238001823017e-17, + -3.7756206444727573e-34, -2.1212242308178287e-50), + qd_real( 7.5318679904361252e-01, -3.7937789838736540e-17, + -6.7009539920231559e-34, -6.7128562115050214e-51), + qd_real( 7.5116513190968637e-01, 4.3499761158645868e-17, + 2.5227718971102212e-33, -6.5969709212757102e-50), + qd_real( 7.4913639452345937e-01, -4.4729078447011889e-17, + -2.4206025249983768e-33, 1.1336681351116422e-49), + qd_real( 7.4710060598018013e-01, 1.1874824875965430e-17, + 2.1992523849833518e-34, 1.1025018564644483e-50), + qd_real( 7.4505778544146595e-01, 1.5078686911877863e-17, + 8.0898987212942471e-34, 8.2677958765323532e-50), + qd_real( 7.4300795213512172e-01, -2.5144629669719265e-17, + 7.1128989512526157e-34, 3.0181629077821220e-50), + qd_real( 7.4095112535495911e-01, -1.4708616952297345e-17, + -4.9550433827142032e-34, 3.1434132533735671e-50), + qd_real( 7.3888732446061511e-01, 3.4324874808225091e-17, + -1.3706639444717610e-33, -3.3520827530718938e-51), + qd_real( 7.3681656887736990e-01, -2.8932468101656295e-17, + -3.4649887126202378e-34, -1.8484474476291476e-50), + qd_real( 7.3473887809596350e-01, -3.4507595976263941e-17, + -2.3718000676666409e-33, -3.9696090387165402e-50), + qd_real( 7.3265427167241282e-01, 1.8918673481573520e-17, + -1.5123719544119886e-33, -9.7922152011625728e-51), + qd_real( 7.3056276922782759e-01, -2.9689959904476928e-17, + -1.1276871244239744e-33, -3.0531520961539007e-50), + qd_real( 7.2846439044822520e-01, 1.1924642323370718e-19, + 5.9001892316611011e-36, 1.2178089069502704e-52), + qd_real( 7.2635915508434601e-01, -3.1917502443460542e-17, + 7.7047912412039396e-34, 4.1455880160182123e-50), + qd_real( 7.2424708295146689e-01, 2.9198471334403004e-17, + 2.3027324968739464e-33, -1.2928820533892183e-51), + qd_real( 7.2212819392921535e-01, -2.3871262053452047e-17, + 1.0636125432862273e-33, -4.4598638837802517e-50), + qd_real( 7.2000250796138165e-01, -2.5689658854462333e-17, + -9.1492566948567925e-34, 4.4403780801267786e-50), + qd_real( 7.1787004505573171e-01, 2.7006476062511453e-17, + -2.2854956580215348e-34, 9.1726903890287867e-51), + qd_real( 7.1573082528381871e-01, -5.1581018476410262e-17, + -1.3736271349300259e-34, -1.2734611344111297e-50), + qd_real( 7.1358486878079364e-01, -4.2342504403133584e-17, + -4.2690366101617268e-34, -2.6352370883066522e-50), + qd_real( 7.1143219574521643e-01, 7.9643298613856813e-18, + 2.9488239510721469e-34, 1.6985236437666356e-50), + qd_real( 7.0927282643886569e-01, -3.7597359110245730e-17, + 1.0613125954645119e-34, 8.9465480185486032e-51), + qd_real( 7.0710678118654757e-01, -4.8336466567264567e-17, + 2.0693376543497068e-33, 2.4677734957341755e-50) +}; + +/* Computes sin(a) and cos(a) using Taylor series. + Assumes |a| <= pi/2048. */ +static void sincos_taylor(const qd_real &a, + qd_real &sin_a, qd_real &cos_a) { + const double thresh = 0.5 * qd_real::_eps * std::abs(to_double(a)); + qd_real p, s, t, x; + + if (a.is_zero()) { + sin_a = 0.0; + cos_a = 1.0; + return; + } + + x = -sqr(a); + s = a; + p = a; + int i = 0; + do { + p *= x; + t = p * inv_fact[i]; + s += t; + i += 2; + } while (i < n_inv_fact && std::abs(to_double(t)) > thresh); + + sin_a = s; + cos_a = sqrt(1.0 - sqr(s)); +} + +static qd_real sin_taylor(const qd_real &a) { + const double thresh = 0.5 * qd_real::_eps * std::abs(to_double(a)); + qd_real p, s, t, x; + + if (a.is_zero()) { + return 0.0; + } + + x = -sqr(a); + s = a; + p = a; + int i = 0; + do { + p *= x; + t = p * inv_fact[i]; + s += t; + i += 2; + } while (i < n_inv_fact && std::abs(to_double(t)) > thresh); + + return s; +} + +static qd_real cos_taylor(const qd_real &a) { + const double thresh = 0.5 * qd_real::_eps; + qd_real p, s, t, x; + + if (a.is_zero()) { + return 1.0; + } + + x = -sqr(a); + s = 1.0 + mul_pwr2(x, 0.5); + p = x; + int i = 1; + do { + p *= x; + t = p * inv_fact[i]; + s += t; + i += 2; + } while (i < n_inv_fact && std::abs(to_double(t)) > thresh); + + return s; +} + +qd_real sin(const qd_real &a) { + + /* Strategy. To compute sin(x), we choose integers a, b so that + + x = s + a * (pi/2) + b * (pi/1024) + + and |s| <= pi/2048. Using a precomputed table of + sin(k pi / 1024) and cos(k pi / 1024), we can compute + sin(x) from sin(s) and cos(s). This greatly increases the + convergence of the sine Taylor series. */ + + if (a.is_zero()) { + return 0.0; + } + + // approximately reduce modulo 2*pi + qd_real z = nint(a / qd_real::_2pi); + qd_real r = a - qd_real::_2pi * z; + + // approximately reduce modulo pi/2 and then modulo pi/1024 + double q = std::floor(r.x[0] / qd_real::_pi2[0] + 0.5); + qd_real t = r - qd_real::_pi2 * q; + int j = static_cast(q); + q = std::floor(t.x[0] / _pi1024[0] + 0.5); + t -= _pi1024 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (j < -2 || j > 2) { + qd_real::error("(qd_real::sin): Cannot reduce modulo pi/2."); + return qd_real::_nan; + } + + if (abs_k > 256) { + qd_real::error("(qd_real::sin): Cannot reduce modulo pi/1024."); + return qd_real::_nan; + } + + if (k == 0) { + switch (j) { + case 0: + return sin_taylor(t); + case 1: + return cos_taylor(t); + case -1: + return -cos_taylor(t); + default: + return -sin_taylor(t); + } + } + + qd_real sin_t, cos_t; + qd_real u = cos_table[abs_k-1]; + qd_real v = sin_table[abs_k-1]; + sincos_taylor(t, sin_t, cos_t); + + if (j == 0) { + if (k > 0) { + r = u * sin_t + v * cos_t; + } else { + r = u * sin_t - v * cos_t; + } + } else if (j == 1) { + if (k > 0) { + r = u * cos_t - v * sin_t; + } else { + r = u * cos_t + v * sin_t; + } + } else if (j == -1) { + if (k > 0) { + r = v * sin_t - u * cos_t; + } else { + r = - u * cos_t - v * sin_t; + } + } else { + if (k > 0) { + r = - u * sin_t - v * cos_t; + } else { + r = v * cos_t - u * sin_t; + } + } + + return r; +} + +qd_real cos(const qd_real &a) { + + if (a.is_zero()) { + return 1.0; + } + + // approximately reduce modulo 2*pi + qd_real z = nint(a / qd_real::_2pi); + qd_real r = a - qd_real::_2pi * z; + + // approximately reduce modulo pi/2 and then modulo pi/1024 + double q = std::floor(r.x[0] / qd_real::_pi2.x[0] + 0.5); + qd_real t = r - qd_real::_pi2 * q; + int j = static_cast(q); + q = std::floor(t.x[0] / _pi1024.x[0] + 0.5); + t -= _pi1024 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (j < -2 || j > 2) { + qd_real::error("(qd_real::cos): Cannot reduce modulo pi/2."); + return qd_real::_nan; + } + + if (abs_k > 256) { + qd_real::error("(qd_real::cos): Cannot reduce modulo pi/1024."); + return qd_real::_nan; + } + + if (k == 0) { + switch (j) { + case 0: + return cos_taylor(t); + case 1: + return -sin_taylor(t); + case -1: + return sin_taylor(t); + default: + return -cos_taylor(t); + } + } + + qd_real sin_t, cos_t; + sincos_taylor(t, sin_t, cos_t); + + qd_real u = cos_table[abs_k-1]; + qd_real v = sin_table[abs_k-1]; + + if (j == 0) { + if (k > 0) { + r = u * cos_t - v * sin_t; + } else { + r = u * cos_t + v * sin_t; + } + } else if (j == 1) { + if (k > 0) { + r = - u * sin_t - v * cos_t; + } else { + r = v * cos_t - u * sin_t; + } + } else if (j == -1) { + if (k > 0) { + r = u * sin_t + v * cos_t; + } else { + r = u * sin_t - v * cos_t; + } + } else { + if (k > 0) { + r = v * sin_t - u * cos_t; + } else { + r = - u * cos_t - v * sin_t; + } + } + + return r; +} + +void sincos(const qd_real &a, qd_real &sin_a, qd_real &cos_a) { + + if (a.is_zero()) { + sin_a = 0.0; + cos_a = 1.0; + return; + } + + // approximately reduce by 2*pi + qd_real z = nint(a / qd_real::_2pi); + qd_real t = a - qd_real::_2pi * z; + + // approximately reduce by pi/2 and then by pi/1024. + double q = std::floor(t.x[0] / qd_real::_pi2.x[0] + 0.5); + t -= qd_real::_pi2 * q; + int j = static_cast(q); + q = std::floor(t.x[0] / _pi1024.x[0] + 0.5); + t -= _pi1024 * q; + int k = static_cast(q); + int abs_k = std::abs(k); + + if (j < -2 || j > 2) { + qd_real::error("(qd_real::sincos): Cannot reduce modulo pi/2."); + cos_a = sin_a = qd_real::_nan; + return; + } + + if (abs_k > 256) { + qd_real::error("(qd_real::sincos): Cannot reduce modulo pi/1024."); + cos_a = sin_a = qd_real::_nan; + return; + } + + qd_real sin_t, cos_t; + sincos_taylor(t, sin_t, cos_t); + + if (k == 0) { + if (j == 0) { + sin_a = sin_t; + cos_a = cos_t; + } else if (j == 1) { + sin_a = cos_t; + cos_a = -sin_t; + } else if (j == -1) { + sin_a = -cos_t; + cos_a = sin_t; + } else { + sin_a = -sin_t; + cos_a = -cos_t; + } + return; + } + + qd_real u = cos_table[abs_k-1]; + qd_real v = sin_table[abs_k-1]; + + if (j == 0) { + if (k > 0) { + sin_a = u * sin_t + v * cos_t; + cos_a = u * cos_t - v * sin_t; + } else { + sin_a = u * sin_t - v * cos_t; + cos_a = u * cos_t + v * sin_t; + } + } else if (j == 1) { + if (k > 0) { + cos_a = - u * sin_t - v * cos_t; + sin_a = u * cos_t - v * sin_t; + } else { + cos_a = v * cos_t - u * sin_t; + sin_a = u * cos_t + v * sin_t; + } + } else if (j == -1) { + if (k > 0) { + cos_a = u * sin_t + v * cos_t; + sin_a = v * sin_t - u * cos_t; + } else { + cos_a = u * sin_t - v * cos_t; + sin_a = - u * cos_t - v * sin_t; + } + } else { + if (k > 0) { + sin_a = - u * sin_t - v * cos_t; + cos_a = v * sin_t - u * cos_t; + } else { + sin_a = v * cos_t - u * sin_t; + cos_a = - u * cos_t - v * sin_t; + } + } +} + +qd_real atan(const qd_real &a) { + return atan2(a, qd_real(1.0)); +} + +qd_real atan2(const qd_real &y, const qd_real &x) { + /* Strategy: Instead of using Taylor series to compute + arctan, we instead use Newton's iteration to solve + the equation + + sin(z) = y/r or cos(z) = x/r + + where r = sqrt(x^2 + y^2). + The iteration is given by + + z' = z + (y - sin(z)) / cos(z) (for equation 1) + z' = z - (x - cos(z)) / sin(z) (for equation 2) + + Here, x and y are normalized so that x^2 + y^2 = 1. + If |x| > |y|, then first iteration is used since the + denominator is larger. Otherwise, the second is used. + */ + + if (x.is_zero()) { + + if (y.is_zero()) { + /* Both x and y is zero. */ + qd_real::error("(qd_real::atan2): Both arguments zero."); + return qd_real::_nan; + } + + return (y.is_positive()) ? qd_real::_pi2 : -qd_real::_pi2; + } else if (y.is_zero()) { + return (x.is_positive()) ? qd_real(0.0) : qd_real::_pi; + } + + if (x == y) { + return (y.is_positive()) ? qd_real::_pi4 : -qd_real::_3pi4; + } + + if (x == -y) { + return (y.is_positive()) ? qd_real::_3pi4 : -qd_real::_pi4; + } + + qd_real r = sqrt(sqr(x) + sqr(y)); + qd_real xx = x / r; + qd_real yy = y / r; + + /* Compute double precision approximation to atan. */ + qd_real z = std::atan2(to_double(y), to_double(x)); + qd_real sin_z, cos_z; + + if (std::abs(xx.x[0]) > std::abs(yy.x[0])) { + /* Use Newton iteration 1. z' = z + (y - sin(z)) / cos(z) */ + sincos(z, sin_z, cos_z); + z += (yy - sin_z) / cos_z; + sincos(z, sin_z, cos_z); + z += (yy - sin_z) / cos_z; + sincos(z, sin_z, cos_z); + z += (yy - sin_z) / cos_z; + } else { + /* Use Newton iteration 2. z' = z - (x - cos(z)) / sin(z) */ + sincos(z, sin_z, cos_z); + z -= (xx - cos_z) / sin_z; + sincos(z, sin_z, cos_z); + z -= (xx - cos_z) / sin_z; + sincos(z, sin_z, cos_z); + z -= (xx - cos_z) / sin_z; + } + + return z; +} + + +qd_real drem(const qd_real &a, const qd_real &b) { + qd_real n = nint(a/b); + return (a - n * b); +} + +qd_real divrem(const qd_real &a, const qd_real &b, qd_real &r) { + qd_real n = nint(a/b); + r = a - n * b; + return n; +} + +qd_real tan(const qd_real &a) { + qd_real s, c; + sincos(a, s, c); + return s/c; +} + +qd_real asin(const qd_real &a) { + qd_real abs_a = abs(a); + + if (abs_a > 1.0) { + qd_real::error("(qd_real::asin): Argument out of domain."); + return qd_real::_nan; + } + + if (abs_a.is_one()) { + return (a.is_positive()) ? qd_real::_pi2 : -qd_real::_pi2; + } + + return atan2(a, sqrt(1.0 - sqr(a))); +} + +qd_real acos(const qd_real &a) { + qd_real abs_a = abs(a); + + if (abs_a > 1.0) { + qd_real::error("(qd_real::acos): Argument out of domain."); + return qd_real::_nan; + } + + if (abs_a.is_one()) { + return (a.is_positive()) ? qd_real(0.0) : qd_real::_pi; + } + + return atan2(sqrt(1.0 - sqr(a)), a); +} + +qd_real sinh(const qd_real &a) { + if (a.is_zero()) { + return 0.0; + } + + if (abs(a) > 0.05) { + qd_real ea = exp(a); + return mul_pwr2(ea - inv(ea), 0.5); + } + + /* Since a is small, using the above formula gives + a lot of cancellation. So use Taylor series. */ + qd_real s = a; + qd_real t = a; + qd_real r = sqr(t); + double m = 1.0; + double thresh = std::abs(to_double(a) * qd_real::_eps); + + do { + m += 2.0; + t *= r; + t /= (m-1) * m; + + s += t; + } while (abs(t) > thresh); + + return s; +} + +qd_real cosh(const qd_real &a) { + if (a.is_zero()) { + return 1.0; + } + + qd_real ea = exp(a); + return mul_pwr2(ea + inv(ea), 0.5); +} + +qd_real tanh(const qd_real &a) { + if (a.is_zero()) { + return 0.0; + } + + if (std::abs(to_double(a)) > 0.05) { + qd_real ea = exp(a); + qd_real inv_ea = inv(ea); + return (ea - inv_ea) / (ea + inv_ea); + } else { + qd_real s, c; + s = sinh(a); + c = sqrt(1.0 + sqr(s)); + return s / c; + } +} + +void sincosh(const qd_real &a, qd_real &s, qd_real &c) { + if (std::abs(to_double(a)) <= 0.05) { + s = sinh(a); + c = sqrt(1.0 + sqr(s)); + } else { + qd_real ea = exp(a); + qd_real inv_ea = inv(ea); + s = mul_pwr2(ea - inv_ea, 0.5); + c = mul_pwr2(ea + inv_ea, 0.5); + } +} + +qd_real asinh(const qd_real &a) { + return log(a + sqrt(sqr(a) + 1.0)); +} + +qd_real acosh(const qd_real &a) { + if (a < 1.0) { + qd_real::error("(qd_real::acosh): Argument out of domain."); + return qd_real::_nan; + } + + return log(a + sqrt(sqr(a) - 1.0)); +} + +qd_real atanh(const qd_real &a) { + if (abs(a) >= 1.0) { + qd_real::error("(qd_real::atanh): Argument out of domain."); + return qd_real::_nan; + } + + return mul_pwr2(log((1.0 + a) / (1.0 - a)), 0.5); +} + +QD_API qd_real fmod(const qd_real &a, const qd_real &b) { + qd_real n = aint(a / b); + return (a - b * n); +} + +QD_API qd_real qdrand() { + static const double m_const = 4.6566128730773926e-10; /* = 2^{-31} */ + double m = m_const; + qd_real r = 0.0; + double d; + + /* Strategy: Generate 31 bits at a time, using lrand48 + random number generator. Shift the bits, and repeat + 7 times. */ + + for (int i = 0; i < 7; i++, m *= m_const) { + d = std::rand() * m; + r += d; + } + + return r; +} + + +/* polyeval(c, n, x) + Evaluates the given n-th degree polynomial at x. + The polynomial is given by the array of (n+1) coefficients. */ +qd_real polyeval(const qd_real *c, int n, const qd_real &x) { + /* Just use Horner's method of polynomial evaluation. */ + qd_real r = c[n]; + + for (int i = n-1; i >= 0; i--) { + r *= x; + r += c[i]; + } + + return r; +} + +/* polyroot(c, n, x0) + Given an n-th degree polynomial, finds a root close to + the given guess x0. Note that this uses simple Newton + iteration scheme, and does not work for multiple roots. */ +QD_API qd_real polyroot(const qd_real *c, int n, + const qd_real &x0, int max_iter, double thresh) { + qd_real x = x0; + qd_real f; + qd_real *d = new qd_real[n]; + bool conv = false; + int i; + double max_c = std::abs(to_double(c[0])); + double v; + + if (thresh == 0.0) thresh = qd_real::_eps; + + /* Compute the coefficients of the derivatives. */ + for (i = 1; i <= n; i++) { + v = std::abs(to_double(c[i])); + if (v > max_c) max_c = v; + d[i-1] = c[i] * static_cast(i); + } + thresh *= max_c; + + /* Newton iteration. */ + for (i = 0; i < max_iter; i++) { + f = polyeval(c, n, x); + + if (abs(f) < thresh) { + conv = true; + break; + } + x -= (f / polyeval(d, n-1, x)); + } + delete [] d; + + if (!conv) { + qd_real::error("(qd_real::polyroot): Failed to converge."); + return qd_real::_nan; + } + + return x; +} + +qd_real qd_real::debug_rand() { + if (std::rand() % 2 == 0) + return qdrand(); + + int expn = 0; + qd_real a = 0.0; + double d; + for (int i = 0; i < 4; i++) { + d = std::ldexp(std::rand() / static_cast(RAND_MAX), -expn); + a += d; + expn = expn + 54 + std::rand() % 200; + } + return a; +} + diff --git a/src/external/PackedCSparse/qd/qd_real.h b/src/external/PackedCSparse/qd/qd_real.h new file mode 100644 index 00000000..1ecfc732 --- /dev/null +++ b/src/external/PackedCSparse/qd/qd_real.h @@ -0,0 +1,296 @@ +/* + * include/qd_real.h + * + * This work was supported by the Director, Office of Science, Division + * of Mathematical, Information, and Computational Sciences of the + * U.S. Department of Energy under contract number DE-AC03-76SF00098. + * + * Copyright (c) 2000-2007 + * + * Quad-double precision (>= 212-bit significand) floating point arithmetic + * package, written in ANSI C++, taking full advantage of operator overloading. + * Uses similar techniques as that of David Bailey's double-double package + * and that of Jonathan Shewchuk's adaptive precision floating point + * arithmetic package. See + * + * http://www.nersc.gov/~dhbailey/mpdist/mpdist.html + * http://www.cs.cmu.edu/~quake/robust.html + * + * for more details. + * + * Yozo Hida + */ +#ifndef _QD_QD_REAL_H +#define _QD_QD_REAL_H + +#include +#include +#include +#include "qd_config.h" +#include "dd_real.h" + +struct QD_API qd_real { + double x[4]; /* The Components. */ + + /* Eliminates any zeros in the middle component(s). */ + void zero_elim(); + void zero_elim(double &e); + + void renorm(); + void renorm(double &e); + + void quick_accum(double d, double &e); + void quick_prod_accum(double a, double b, double &e); + + qd_real(double x0, double x1, double x2, double x3); + explicit qd_real(const double *xx); + + static const qd_real _2pi; + static const qd_real _pi; + static const qd_real _3pi4; + static const qd_real _pi2; + static const qd_real _pi4; + static const qd_real _e; + static const qd_real _log2; + static const qd_real _log10; + static const qd_real _nan; + static const qd_real _inf; + + static const double _eps; + static const double _min_normalized; + static const qd_real _max; + static const qd_real _safe_max; + static const int _ndigits; + + qd_real(); + qd_real(const char *s); + qd_real(const dd_real &dd); + qd_real(double d); + qd_real(int i); + + double operator[](int i) const; + double &operator[](int i); + + static void error(const char *msg); + + bool isnan() const; + bool isfinite() const { return QD_ISFINITE(x[0]); } + bool isinf() const { return QD_ISINF(x[0]); } + + static qd_real ieee_add(const qd_real &a, const qd_real &b); + static qd_real sloppy_add(const qd_real &a, const qd_real &b); + + qd_real &operator+=(double a); + qd_real &operator+=(const dd_real &a); + qd_real &operator+=(const qd_real &a); + + qd_real &operator-=(double a); + qd_real &operator-=(const dd_real &a); + qd_real &operator-=(const qd_real &a); + + static qd_real sloppy_mul(const qd_real &a, const qd_real &b); + static qd_real accurate_mul(const qd_real &a, const qd_real &b); + + qd_real &operator*=(double a); + qd_real &operator*=(const dd_real &a); + qd_real &operator*=(const qd_real &a); + + static qd_real sloppy_div(const qd_real &a, const dd_real &b); + static qd_real accurate_div(const qd_real &a, const dd_real &b); + static qd_real sloppy_div(const qd_real &a, const qd_real &b); + static qd_real accurate_div(const qd_real &a, const qd_real &b); + + qd_real &operator/=(double a); + qd_real &operator/=(const dd_real &a); + qd_real &operator/=(const qd_real &a); + + qd_real operator^(int n) const; + + qd_real operator-() const; + + qd_real &operator=(double a); + qd_real &operator=(const dd_real &a); + qd_real &operator=(const char *s); + + bool is_zero() const; + bool is_one() const; + bool is_positive() const; + bool is_negative() const; + + explicit operator bool() const; // new + explicit operator double() const; // new + + static qd_real rand(void); + + void to_digits(char *s, int &expn, int precision = _ndigits) const; + void write(char *s, int len, int precision = _ndigits, + bool showpos = false, bool uppercase = false) const; + std::string to_string(int precision = _ndigits, int width = 0, + std::ios_base::fmtflags fmt = static_cast(0), + bool showpos = false, bool uppercase = false, char fill = ' ') const; + static int read(const char *s, qd_real &a); + + /* Debugging methods */ + void dump(const std::string &name, std::ostream &os = std::cerr) const; + void dump_bits(const std::string &name, + std::ostream &os = std::cerr) const; + + static qd_real debug_rand(); + +}; + +namespace std { + template <> + class numeric_limits : public numeric_limits { + public: + inline static double epsilon() { return qd_real::_eps; } + inline static double min() { return qd_real::_min_normalized; } + inline static qd_real max() { return qd_real::_max; } + inline static qd_real safe_max() { return qd_real::_safe_max; } + static const int digits = 209; + static const int digits10 = 62; + }; +} + +QD_API qd_real polyeval(const qd_real *c, int n, const qd_real &x); +QD_API qd_real polyroot(const qd_real *c, int n, + const qd_real &x0, int max_iter = 64, double thresh = 0.0); + +QD_API qd_real qdrand(void); +QD_API qd_real sqrt(const qd_real &a); + +QD_API inline bool isnan(const qd_real &a) { return a.isnan(); } +QD_API inline bool isfinite(const qd_real &a) { return a.isfinite(); } +QD_API inline bool isinf(const qd_real &a) { return a.isinf(); } + +/* Computes qd * d where d is known to be a power of 2. + This can be done component wise. */ +QD_API qd_real mul_pwr2(const qd_real &qd, double d); + +QD_API qd_real operator+(const qd_real &a, const qd_real &b); +QD_API qd_real operator+(const dd_real &a, const qd_real &b); +QD_API qd_real operator+(const qd_real &a, const dd_real &b); +QD_API qd_real operator+(const qd_real &a, double b); +QD_API qd_real operator+(double a, const qd_real &b); + +QD_API qd_real operator-(const qd_real &a, const qd_real &b); +QD_API qd_real operator-(const dd_real &a, const qd_real &b); +QD_API qd_real operator-(const qd_real &a, const dd_real &b); +QD_API qd_real operator-(const qd_real &a, double b); +QD_API qd_real operator-(double a, const qd_real &b); + +QD_API qd_real operator*(const qd_real &a, const qd_real &b); +QD_API qd_real operator*(const dd_real &a, const qd_real &b); +QD_API qd_real operator*(const qd_real &a, const dd_real &b); +QD_API qd_real operator*(const qd_real &a, double b); +QD_API qd_real operator*(double a, const qd_real &b); + +QD_API qd_real operator/(const qd_real &a, const qd_real &b); +QD_API qd_real operator/(const dd_real &a, const qd_real &b); +QD_API qd_real operator/(const qd_real &a, const dd_real &b); +QD_API qd_real operator/(const qd_real &a, double b); +QD_API qd_real operator/(double a, const qd_real &b); + +QD_API qd_real sqr(const qd_real &a); +QD_API qd_real sqrt(const qd_real &a); +QD_API qd_real pow(const qd_real &a, int n); +QD_API qd_real pow(const qd_real &a, const qd_real &b); +QD_API qd_real npwr(const qd_real &a, int n); + +QD_API qd_real nroot(const qd_real &a, int n); + +QD_API qd_real rem(const qd_real &a, const qd_real &b); +QD_API qd_real drem(const qd_real &a, const qd_real &b); +QD_API qd_real divrem(const qd_real &a, const qd_real &b, qd_real &r); + +dd_real to_dd_real(const qd_real &a); +double to_double(const qd_real &a); +int to_int(const qd_real &a); + +QD_API bool operator==(const qd_real &a, const qd_real &b); +QD_API bool operator==(const qd_real &a, const dd_real &b); +QD_API bool operator==(const dd_real &a, const qd_real &b); +QD_API bool operator==(double a, const qd_real &b); +QD_API bool operator==(const qd_real &a, double b); + +QD_API bool operator<(const qd_real &a, const qd_real &b); +QD_API bool operator<(const qd_real &a, const dd_real &b); +QD_API bool operator<(const dd_real &a, const qd_real &b); +QD_API bool operator<(double a, const qd_real &b); +QD_API bool operator<(const qd_real &a, double b); + +QD_API bool operator>(const qd_real &a, const qd_real &b); +QD_API bool operator>(const qd_real &a, const dd_real &b); +QD_API bool operator>(const dd_real &a, const qd_real &b); +QD_API bool operator>(double a, const qd_real &b); +QD_API bool operator>(const qd_real &a, double b); + +QD_API bool operator<=(const qd_real &a, const qd_real &b); +QD_API bool operator<=(const qd_real &a, const dd_real &b); +QD_API bool operator<=(const dd_real &a, const qd_real &b); +QD_API bool operator<=(double a, const qd_real &b); +QD_API bool operator<=(const qd_real &a, double b); + +QD_API bool operator>=(const qd_real &a, const qd_real &b); +QD_API bool operator>=(const qd_real &a, const dd_real &b); +QD_API bool operator>=(const dd_real &a, const qd_real &b); +QD_API bool operator>=(double a, const qd_real &b); +QD_API bool operator>=(const qd_real &a, double b); + +QD_API bool operator!=(const qd_real &a, const qd_real &b); +QD_API bool operator!=(const qd_real &a, const dd_real &b); +QD_API bool operator!=(const dd_real &a, const qd_real &b); +QD_API bool operator!=(double a, const qd_real &b); +QD_API bool operator!=(const qd_real &a, double b); + +QD_API qd_real fabs(const qd_real &a); +QD_API qd_real abs(const qd_real &a); /* same as fabs */ + +QD_API qd_real ldexp(const qd_real &a, int n); + +QD_API qd_real nint(const qd_real &a); +QD_API qd_real quick_nint(const qd_real &a); +QD_API qd_real floor(const qd_real &a); +QD_API qd_real ceil(const qd_real &a); +QD_API qd_real aint(const qd_real &a); + +QD_API qd_real sin(const qd_real &a); +QD_API qd_real cos(const qd_real &a); +QD_API qd_real tan(const qd_real &a); +QD_API void sincos(const qd_real &a, qd_real &s, qd_real &c); + +QD_API qd_real asin(const qd_real &a); +QD_API qd_real acos(const qd_real &a); +QD_API qd_real atan(const qd_real &a); +QD_API qd_real atan2(const qd_real &y, const qd_real &x); + +QD_API qd_real exp(const qd_real &a); +QD_API qd_real log(const qd_real &a); +QD_API qd_real log10(const qd_real &a); + +QD_API qd_real sinh(const qd_real &a); +QD_API qd_real cosh(const qd_real &a); +QD_API qd_real tanh(const qd_real &a); +QD_API void sincosh(const qd_real &a, qd_real &sin_qd, qd_real &cos_qd); + +QD_API qd_real asinh(const qd_real &a); +QD_API qd_real acosh(const qd_real &a); +QD_API qd_real atanh(const qd_real &a); + +QD_API qd_real qdrand(void); + +QD_API qd_real max(const qd_real &a, const qd_real &b); +QD_API qd_real max(const qd_real &a, const qd_real &b, const qd_real &c); +QD_API qd_real min(const qd_real &a, const qd_real &b); +QD_API qd_real min(const qd_real &a, const qd_real &b, const qd_real &c); + +QD_API qd_real fmod(const qd_real &a, const qd_real &b); + +QD_API std::ostream &operator<<(std::ostream &s, const qd_real &a); +QD_API std::istream &operator>>(std::istream &s, qd_real &a); +#ifdef QD_INLINE +#include "qd_inline.h" +#endif + +#endif /* _QD_QD_REAL_H */ + diff --git a/src/external/PackedCSparse/qd/util.cc b/src/external/PackedCSparse/qd/util.cc new file mode 100644 index 00000000..ab962081 --- /dev/null +++ b/src/external/PackedCSparse/qd/util.cc @@ -0,0 +1,22 @@ +#include +#include "util.h" + +void append_expn(std::string &str, int expn) { + int k; + + str += (expn < 0 ? '-' : '+'); + expn = std::abs(expn); + + if (expn >= 100) { + k = (expn / 100); + str += '0' + k; + expn -= 100*k; + } + + k = (expn / 10); + str += '0' + k; + expn -= 10*k; + + str += '0' + expn; +} + diff --git a/src/external/PackedCSparse/qd/util.h b/src/external/PackedCSparse/qd/util.h new file mode 100644 index 00000000..7de35836 --- /dev/null +++ b/src/external/PackedCSparse/qd/util.h @@ -0,0 +1,4 @@ +#include + +void append_expn(std::string &str, int expn); + diff --git a/src/external/PackedCSparse/transpose.h b/src/external/PackedCSparse/transpose.h new file mode 100644 index 00000000..6e1502e3 --- /dev/null +++ b/src/external/PackedCSparse/transpose.h @@ -0,0 +1,90 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2022 Ioannis Iakovidis + +// This file is converted from PolytopeSamplerMatlab +//(https://github.com/ConstrainedSampler/PolytopeSamplerMatlab/blob/master/code/solver/PackedCSparse/PackedChol.h) by Ioannis Iakovidis +#pragma once +#include "SparseMatrix.h" + +// Problem: +// Compute M = A' + +// Algorithm: +// We precompute the mapping from entries of A to entries of At + +namespace PackedCSparse { + template + struct TransposeOutput : SparseMatrix + { + UniquePtr forward; + + template + void initialize(const SparseMatrix& A) + { + pcs_assert(A.initialized(), "transpose: bad inputs."); + SparseMatrix::initialize(A.n, A.m, A.nnz()); + + Ti Am = A.m, An = A.n, * Ap = A.p.get(), * Ai = A.i.get(); + Ti Bm = this->m, Bn = this->n, * Bp = this->p.get(), * Bi = this->i.get(); + Ti nz = A.nnz(); + + // compute row counts of A + Ti* count = new Ti[Bn + 1](); + + for (Ti p = 0; p < nz; p++) + count[Ai[p]]++; + + // compute this->p + Bp[0] = 0; + for (Ti i = 0; i < Bn; i++) + { + Bp[i + 1] = Bp[i] + count[i]; + count[i] = Bp[i]; // Now, cnt[i] stores the index of the first element in the i-th row + } + + // compute i and forward + if (!std::is_same::value) + forward.reset(new Ti[nz]); + for (Ti j = 0; j < An; j++) + { + for (Ti p = Ap[j]; p < Ap[j + 1]; p++) + { + Ti q = count[Ai[p]]; + Bi[q] = j; + if (!std::is_same::value) + forward[p] = q; + count[Ai[p]]++; + } + } + + delete[] count; + } + }; + + template + void transpose(TransposeOutput& o, const SparseMatrix& A) + { + if (!o.initialized()) + o.initialize(A); + + Tx* Ax = A.x.get(); Tx2 *Bx = o.x.get(); + Ti nz = o.nnz(), *forward = o.forward.get(); + + if (!std::is_same::value) + { + for (Ti s = 0; s < nz; s++) + Bx[forward[s]] = Tx2(Ax[s]); + } + } + + template + TransposeOutput transpose(const SparseMatrix& A) + { + TransposeOutput o; + transpose(o, A); + return o; + } +} diff --git a/src/external/Spectra/include/Spectra/GenEigsBase.h b/src/external/Spectra/include/Spectra/GenEigsBase.h new file mode 100644 index 00000000..19b12c15 --- /dev/null +++ b/src/external/Spectra/include/Spectra/GenEigsBase.h @@ -0,0 +1,479 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef GEN_EIGS_BASE_H +#define GEN_EIGS_BASE_H + +#include +#include // std::vector +#include // std::abs, std::pow, std::sqrt +#include // std::min, std::copy +#include // std::complex, std::conj, std::norm, std::abs +#include // std::invalid_argument + +#include "Util/TypeTraits.h" +#include "Util/SelectionRule.h" +#include "Util/CompInfo.h" +#include "Util/SimpleRandom.h" +#include "MatOp/internal/ArnoldiOp.h" +#include "LinAlg/UpperHessenbergQR.h" +#include "LinAlg/DoubleShiftQR.h" +#include "LinAlg/UpperHessenbergEigen.h" +#include "LinAlg/Arnoldi.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This is the base class for general eigen solvers, mainly for internal use. +/// It is kept here to provide the documentation for member functions of concrete eigen solvers +/// such as GenEigsSolver and GenEigsRealShiftSolver. +/// +template < typename Scalar, + int SelectionRule, + typename OpType, + typename BOpType > +class GenEigsBase +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Array Array; + typedef Eigen::Array BoolArray; + typedef Eigen::Map MapMat; + typedef Eigen::Map MapVec; + typedef Eigen::Map MapConstVec; + + typedef std::complex Complex; + typedef Eigen::Matrix ComplexMatrix; + typedef Eigen::Matrix ComplexVector; + + typedef ArnoldiOp ArnoldiOpType; + typedef Arnoldi ArnoldiFac; + +protected: + OpType* m_op; // object to conduct matrix operation, + // e.g. matrix-vector product + const Index m_n; // dimension of matrix A + const Index m_nev; // number of eigenvalues requested + const Index m_ncv; // dimension of Krylov subspace in the Arnoldi method + Index m_nmatop; // number of matrix operations called + Index m_niter; // number of restarting iterations + + ArnoldiFac m_fac; // Arnoldi factorization + + ComplexVector m_ritz_val; // Ritz values + ComplexMatrix m_ritz_vec; // Ritz vectors + ComplexVector m_ritz_est; // last row of m_ritz_vec + +private: + BoolArray m_ritz_conv; // indicator of the convergence of Ritz values + int m_info; // status of the computation + + const Scalar m_near_0; // a very small value, but 1.0 / m_near_0 does not overflow + // ~= 1e-307 for the "double" type + const Scalar m_eps; // the machine precision, ~= 1e-16 for the "double" type + const Scalar m_eps23; // m_eps^(2/3), used to test the convergence + + // Real Ritz values calculated from UpperHessenbergEigen have exact zero imaginary part + // Complex Ritz values have exact conjugate pairs + // So we use exact tests here + static bool is_complex(const Complex& v) { return v.imag() != Scalar(0); } + static bool is_conj(const Complex& v1, const Complex& v2) { return v1 == Eigen::numext::conj(v2); } + + // Implicitly restarted Arnoldi factorization + void restart(Index k) + { + using std::norm; + + if(k >= m_ncv) + return; + + DoubleShiftQR decomp_ds(m_ncv); + UpperHessenbergQR decomp_hb(m_ncv); + Matrix Q = Matrix::Identity(m_ncv, m_ncv); + + for(Index i = k; i < m_ncv; i++) + { + if(is_complex(m_ritz_val[i]) && is_conj(m_ritz_val[i], m_ritz_val[i + 1])) + { + // H - mu * I = Q1 * R1 + // H <- R1 * Q1 + mu * I = Q1' * H * Q1 + // H - conj(mu) * I = Q2 * R2 + // H <- R2 * Q2 + conj(mu) * I = Q2' * H * Q2 + // + // (H - mu * I) * (H - conj(mu) * I) = Q1 * Q2 * R2 * R1 = Q * R + const Scalar s = Scalar(2) * m_ritz_val[i].real(); + const Scalar t = norm(m_ritz_val[i]); + + decomp_ds.compute(m_fac.matrix_H(), s, t); + + // Q -> Q * Qi + decomp_ds.apply_YQ(Q); + // H -> Q'HQ + // Matrix Q = Matrix::Identity(m_ncv, m_ncv); + // decomp_ds.apply_YQ(Q); + // m_fac_H = Q.transpose() * m_fac_H * Q; + m_fac.compress_H(decomp_ds); + + i++; + } else { + // QR decomposition of H - mu * I, mu is real + decomp_hb.compute(m_fac.matrix_H(), m_ritz_val[i].real()); + + // Q -> Q * Qi + decomp_hb.apply_YQ(Q); + // H -> Q'HQ = RQ + mu * I + m_fac.compress_H(decomp_hb); + } + } + + m_fac.compress_V(Q); + m_fac.factorize_from(k, m_ncv, m_nmatop); + + retrieve_ritzpair(); + } + + // Calculates the number of converged Ritz values + Index num_converged(Scalar tol) + { + // thresh = tol * max(m_eps23, abs(theta)), theta for Ritz value + Array thresh = tol * m_ritz_val.head(m_nev).array().abs().max(m_eps23); + Array resid = m_ritz_est.head(m_nev).array().abs() * m_fac.f_norm(); + // Converged "wanted" Ritz values + m_ritz_conv = (resid < thresh); + + return m_ritz_conv.cast().sum(); + } + + // Returns the adjusted nev for restarting + Index nev_adjusted(Index nconv) + { + using std::abs; + + Index nev_new = m_nev; + for(Index i = m_nev; i < m_ncv; i++) + if(abs(m_ritz_est[i]) < m_near_0) nev_new++; + + // Adjust nev_new, according to dnaup2.f line 660~674 in ARPACK + nev_new += std::min(nconv, (m_ncv - nev_new) / 2); + if(nev_new == 1 && m_ncv >= 6) + nev_new = m_ncv / 2; + else if(nev_new == 1 && m_ncv > 3) + nev_new = 2; + + if(nev_new > m_ncv - 2) + nev_new = m_ncv - 2; + + // Increase nev by one if ritz_val[nev - 1] and + // ritz_val[nev] are conjugate pairs + if(is_complex(m_ritz_val[nev_new - 1]) && + is_conj(m_ritz_val[nev_new - 1], m_ritz_val[nev_new])) + { + nev_new++; + } + + return nev_new; + } + + // Retrieves and sorts Ritz values and Ritz vectors + void retrieve_ritzpair() + { + UpperHessenbergEigen decomp(m_fac.matrix_H()); + const ComplexVector& evals = decomp.eigenvalues(); + ComplexMatrix evecs = decomp.eigenvectors(); + + SortEigenvalue sorting(evals.data(), evals.size()); + std::vector ind = sorting.index(); + + // Copy the Ritz values and vectors to m_ritz_val and m_ritz_vec, respectively + for(Index i = 0; i < m_ncv; i++) + { + m_ritz_val[i] = evals[ind[i]]; + m_ritz_est[i] = evecs(m_ncv - 1, ind[i]); + } + for(Index i = 0; i < m_nev; i++) + { + m_ritz_vec.col(i).noalias() = evecs.col(ind[i]); + } + } + +protected: + // Sorts the first nev Ritz pairs in the specified order + // This is used to return the final results + virtual void sort_ritzpair(int sort_rule) + { + // First make sure that we have a valid index vector + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + std::vector ind = sorting.index(); + + switch(sort_rule) + { + case LARGEST_MAGN: + break; + case LARGEST_REAL: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case LARGEST_IMAG: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case SMALLEST_MAGN: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case SMALLEST_REAL: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case SMALLEST_IMAG: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + default: + throw std::invalid_argument("unsupported sorting rule"); + } + + ComplexVector new_ritz_val(m_ncv); + ComplexMatrix new_ritz_vec(m_ncv, m_nev); + BoolArray new_ritz_conv(m_nev); + + for(Index i = 0; i < m_nev; i++) + { + new_ritz_val[i] = m_ritz_val[ind[i]]; + new_ritz_vec.col(i).noalias() = m_ritz_vec.col(ind[i]); + new_ritz_conv[i] = m_ritz_conv[ind[i]]; + } + + m_ritz_val.swap(new_ritz_val); + m_ritz_vec.swap(new_ritz_vec); + m_ritz_conv.swap(new_ritz_conv); + } + +public: + /// \cond + + GenEigsBase(OpType* op, BOpType* Bop, Index nev, Index ncv) : + m_op(op), + m_n(m_op->rows()), + m_nev(nev), + m_ncv(ncv > m_n ? m_n : ncv), + m_nmatop(0), + m_niter(0), + m_fac(ArnoldiOpType(op, Bop), m_ncv), + m_info(NOT_COMPUTED), + m_near_0(TypeTraits::min() * Scalar(10)), + m_eps(Eigen::NumTraits::epsilon()), + m_eps23(Eigen::numext::pow(m_eps, Scalar(2.0) / 3)) + { + if(nev < 1 || nev > m_n - 2) + throw std::invalid_argument("nev must satisfy 1 <= nev <= n - 2, n is the size of matrix"); + + if(ncv < nev + 2 || ncv > m_n) + throw std::invalid_argument("ncv must satisfy nev + 2 <= ncv <= n, n is the size of matrix"); + } + + /// + /// Virtual destructor + /// + virtual ~GenEigsBase() {} + + /// \endcond + + /// + /// Initializes the solver by providing an initial residual vector. + /// + /// \param init_resid Pointer to the initial residual vector. + /// + /// **Spectra** (and also **ARPACK**) uses an iterative algorithm + /// to find eigenvalues. This function allows the user to provide the initial + /// residual vector. + /// + void init(const Scalar* init_resid) + { + // Reset all matrices/vectors to zero + m_ritz_val.resize(m_ncv); + m_ritz_vec.resize(m_ncv, m_nev); + m_ritz_est.resize(m_ncv); + m_ritz_conv.resize(m_nev); + + m_ritz_val.setZero(); + m_ritz_vec.setZero(); + m_ritz_est.setZero(); + m_ritz_conv.setZero(); + + m_nmatop = 0; + m_niter = 0; + + // Initialize the Arnoldi factorization + MapConstVec v0(init_resid, m_n); + m_fac.init(v0, m_nmatop); + } + + /// + /// Initializes the solver by providing a random initial residual vector. + /// + /// This overloaded function generates a random initial residual vector + /// (with a fixed random seed) for the algorithm. Elements in the vector + /// follow independent Uniform(-0.5, 0.5) distribution. + /// + void init() + { + SimpleRandom rng(0); + Vector init_resid = rng.random_vec(m_n); + init(init_resid.data()); + } + + /// + /// Conducts the major computation procedure. + /// + /// \param maxit Maximum number of iterations allowed in the algorithm. + /// \param tol Precision parameter for the calculated eigenvalues. + /// \param sort_rule Rule to sort the eigenvalues and eigenvectors. + /// Supported values are + /// `Spectra::LARGEST_MAGN`, `Spectra::LARGEST_REAL`, + /// `Spectra::LARGEST_IMAG`, `Spectra::SMALLEST_MAGN`, + /// `Spectra::SMALLEST_REAL` and `Spectra::SMALLEST_IMAG`, + /// for example `LARGEST_MAGN` indicates that eigenvalues + /// with largest magnitude come first. + /// Note that this argument is only used to + /// **sort** the final result, and the **selection** rule + /// (e.g. selecting the largest or smallest eigenvalues in the + /// full spectrum) is specified by the template parameter + /// `SelectionRule` of GenEigsSolver. + /// + /// \return Number of converged eigenvalues. + /// + Index compute(Index maxit = 1000, Scalar tol = 1e-10, int sort_rule = LARGEST_MAGN) + { + // The m-step Arnoldi factorization + m_fac.factorize_from(1, m_ncv, m_nmatop); + retrieve_ritzpair(); + // Restarting + Index i, nconv = 0, nev_adj; + for(i = 0; i < maxit; i++) + { + nconv = num_converged(tol); + if(nconv >= m_nev) + break; + + nev_adj = nev_adjusted(nconv); + restart(nev_adj); + } + // Sorting results + sort_ritzpair(sort_rule); + + m_niter += i + 1; + m_info = (nconv >= m_nev) ? SUCCESSFUL : NOT_CONVERGING; + + return std::min(m_nev, nconv); + } + + /// + /// Returns the status of the computation. + /// The full list of enumeration values can be found in \ref Enumerations. + /// + int info() const { return m_info; } + + /// + /// Returns the number of iterations used in the computation. + /// + Index num_iterations() const { return m_niter; } + + /// + /// Returns the number of matrix operations used in the computation. + /// + Index num_operations() const { return m_nmatop; } + + /// + /// Returns the converged eigenvalues. + /// + /// \return A complex-valued vector containing the eigenvalues. + /// Returned vector type will be `Eigen::Vector, ...>`, depending on + /// the template parameter `Scalar` defined. + /// + ComplexVector eigenvalues() const + { + const Index nconv = m_ritz_conv.cast().sum(); + ComplexVector res(nconv); + + if(!nconv) + return res; + + Index j = 0; + for(Index i = 0; i < m_nev; i++) + { + if(m_ritz_conv[i]) + { + res[j] = m_ritz_val[i]; + j++; + } + } + + return res; + } + + /// + /// Returns the eigenvectors associated with the converged eigenvalues. + /// + /// \param nvec The number of eigenvectors to return. + /// + /// \return A complex-valued matrix containing the eigenvectors. + /// Returned matrix type will be `Eigen::Matrix, ...>`, + /// depending on the template parameter `Scalar` defined. + /// + ComplexMatrix eigenvectors(Index nvec) const + { + const Index nconv = m_ritz_conv.cast().sum(); + nvec = std::min(nvec, nconv); + ComplexMatrix res(m_n, nvec); + + if(!nvec) + return res; + + ComplexMatrix ritz_vec_conv(m_ncv, nvec); + Index j = 0; + for(Index i = 0; i < m_nev && j < nvec; i++) + { + if(m_ritz_conv[i]) + { + ritz_vec_conv.col(j).noalias() = m_ritz_vec.col(i); + j++; + } + } + + res.noalias() = m_fac.matrix_V() * ritz_vec_conv; + + return res; + } + + /// + /// Returns all converged eigenvectors. + /// + ComplexMatrix eigenvectors() const + { + return eigenvectors(m_nev); + } +}; + + +} // namespace Spectra + +#endif // GEN_EIGS_BASE_H diff --git a/src/external/Spectra/include/Spectra/GenEigsComplexShiftSolver.h b/src/external/Spectra/include/Spectra/GenEigsComplexShiftSolver.h new file mode 100644 index 00000000..2c1aee7f --- /dev/null +++ b/src/external/Spectra/include/Spectra/GenEigsComplexShiftSolver.h @@ -0,0 +1,156 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef GEN_EIGS_COMPLEX_SHIFT_SOLVER_H +#define GEN_EIGS_COMPLEX_SHIFT_SOLVER_H + +#include + +#include "GenEigsBase.h" +#include "Util/SelectionRule.h" +#include "MatOp/DenseGenComplexShiftSolve.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This class implements the eigen solver for general real matrices with +/// a complex shift value in the **shift-and-invert mode**. The background +/// knowledge of the shift-and-invert mode can be found in the documentation +/// of the SymEigsShiftSolver class. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the shifted-and-inverted eigenvalues. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class. Users could either +/// use the DenseGenComplexShiftSolve wrapper class, or define their +/// own that implements all the public member functions as in +/// DenseGenComplexShiftSolve. +/// +template > +class GenEigsComplexShiftSolver: public GenEigsBase +{ +private: + typedef Eigen::Index Index; + typedef std::complex Complex; + typedef Eigen::Matrix Vector; + typedef Eigen::Matrix ComplexVector; + + const Scalar m_sigmar; + const Scalar m_sigmai; + + // First transform back the Ritz values, and then sort + void sort_ritzpair(int sort_rule) + { + using std::abs; + using std::sqrt; + using std::norm; + + // The eigenvalues we get from the iteration is + // nu = 0.5 * (1 / (lambda - sigma) + 1 / (lambda - conj(sigma))) + // So the eigenvalues of the original problem is + // 1 \pm sqrt(1 - 4 * nu^2 * sigmai^2) + // lambda = sigmar + ----------------------------------- + // 2 * nu + // We need to pick the correct root + // Let (lambdaj, vj) be the j-th eigen pair, then A * vj = lambdaj * vj + // and inv(A - r * I) * vj = 1 / (lambdaj - r) * vj + // where r is any shift value. + // We can use this identity to determine lambdaj + // + // op(v) computes Re(inv(A - r * I) * v) for any real v + // If r is real, then op(v) is also real. Let a = Re(vj), b = Im(vj), + // then op(vj) = op(a) + op(b) * i + // By comparing op(vj) and [1 / (lambdaj - r) * vj], we can determine + // which one is the correct root + + // Select a random shift value + SimpleRandom rng(0); + const Scalar shiftr = rng.random() * m_sigmar + rng.random(); + const Complex shift = Complex(shiftr, Scalar(0)); + this->m_op->set_shift(shiftr, Scalar(0)); + + // Calculate inv(A - r * I) * vj + Vector v_real(this->m_n), v_imag(this->m_n), OPv_real(this->m_n), OPv_imag(this->m_n); + const Scalar eps = Eigen::NumTraits::epsilon(); + for(Index i = 0; i < this->m_nev; i++) + { + v_real.noalias() = this->m_fac.matrix_V() * this->m_ritz_vec.col(i).real(); + v_imag.noalias() = this->m_fac.matrix_V() * this->m_ritz_vec.col(i).imag(); + this->m_op->perform_op(v_real.data(), OPv_real.data()); + this->m_op->perform_op(v_imag.data(), OPv_imag.data()); + + // Two roots computed from the quadratic equation + const Complex nu = this->m_ritz_val[i]; + const Complex root_part1 = m_sigmar + Scalar(0.5) / nu; + const Complex root_part2 = Scalar(0.5) * sqrt(Scalar(1) - Scalar(4) * m_sigmai * m_sigmai * (nu * nu)) / nu; + const Complex root1 = root_part1 + root_part2; + const Complex root2 = root_part1 - root_part2; + + // Test roots + Scalar err1 = Scalar(0), err2 = Scalar(0); + for(int k = 0; k < this->m_n; k++) + { + const Complex rhs1 = Complex(v_real[k], v_imag[k]) / (root1 - shift); + const Complex rhs2 = Complex(v_real[k], v_imag[k]) / (root2 - shift); + const Complex OPv = Complex(OPv_real[k], OPv_imag[k]); + err1 += norm(OPv - rhs1); + err2 += norm(OPv - rhs2); + } + + const Complex lambdaj = (err1 < err2) ? root1 : root2; + this->m_ritz_val[i] = lambdaj; + + if(abs(Eigen::numext::imag(lambdaj)) > eps) + { + this->m_ritz_val[i + 1] = Eigen::numext::conj(lambdaj); + i++; + } else { + this->m_ritz_val[i] = Complex(Eigen::numext::real(lambdaj), Scalar(0)); + } + } + + GenEigsBase::sort_ritzpair(sort_rule); + } +public: + /// + /// Constructor to create a eigen solver object using the shift-and-invert mode. + /// + /// \param op Pointer to the matrix operation object. This class should implement + /// the complex shift-solve operation of \f$A\f$: calculating + /// \f$\mathrm{Re}\{(A-\sigma I)^{-1}v\}\f$ for any vector \f$v\f$. Users could either + /// create the object from the DenseGenComplexShiftSolve wrapper class, or + /// define their own that implements all the public member functions + /// as in DenseGenComplexShiftSolve. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-2\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev+2 \le ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev + 1\f$. + /// \param sigmar The real part of the shift. + /// \param sigmai The imaginary part of the shift. + /// + GenEigsComplexShiftSolver(OpType* op, Index nev, Index ncv, const Scalar& sigmar, const Scalar& sigmai) : + GenEigsBase(op, NULL, nev, ncv), + m_sigmar(sigmar), m_sigmai(sigmai) + { + this->m_op->set_shift(m_sigmar, m_sigmai); + } +}; + + +} // namespace Spectra + +#endif // GEN_EIGS_COMPLEX_SHIFT_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/GenEigsRealShiftSolver.h b/src/external/Spectra/include/Spectra/GenEigsRealShiftSolver.h new file mode 100644 index 00000000..a7e3da8e --- /dev/null +++ b/src/external/Spectra/include/Spectra/GenEigsRealShiftSolver.h @@ -0,0 +1,90 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef GEN_EIGS_REAL_SHIFT_SOLVER_H +#define GEN_EIGS_REAL_SHIFT_SOLVER_H + +#include + +#include "GenEigsBase.h" +#include "Util/SelectionRule.h" +#include "MatOp/DenseGenRealShiftSolve.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This class implements the eigen solver for general real matrices with +/// a real shift value in the **shift-and-invert mode**. The background +/// knowledge of the shift-and-invert mode can be found in the documentation +/// of the SymEigsShiftSolver class. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the shifted-and-inverted eigenvalues. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class. Users could either +/// use the wrapper classes such as DenseGenRealShiftSolve and +/// SparseGenRealShiftSolve, or define their +/// own that implements all the public member functions as in +/// DenseGenRealShiftSolve. +/// +template > +class GenEigsRealShiftSolver: public GenEigsBase +{ +private: + typedef Eigen::Index Index; + typedef std::complex Complex; + typedef Eigen::Array ComplexArray; + + const Scalar m_sigma; + + // First transform back the Ritz values, and then sort + void sort_ritzpair(int sort_rule) + { + // The eigenvalues we get from the iteration is nu = 1 / (lambda - sigma) + // So the eigenvalues of the original problem is lambda = 1 / nu + sigma + ComplexArray ritz_val_org = Scalar(1.0) / this->m_ritz_val.head(this->m_nev).array() + m_sigma; + this->m_ritz_val.head(this->m_nev) = ritz_val_org; + GenEigsBase::sort_ritzpair(sort_rule); + } +public: + /// + /// Constructor to create a eigen solver object using the shift-and-invert mode. + /// + /// \param op Pointer to the matrix operation object. This class should implement + /// the shift-solve operation of \f$A\f$: calculating + /// \f$(A-\sigma I)^{-1}v\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper class such as DenseGenRealShiftSolve, or + /// define their own that implements all the public member functions + /// as in DenseGenRealShiftSolve. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-2\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev+2 \le ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev + 1\f$. + /// \param sigma The real-valued shift. + /// + GenEigsRealShiftSolver(OpType* op, Index nev, Index ncv, Scalar sigma) : + GenEigsBase(op, NULL, nev, ncv), + m_sigma(sigma) + { + this->m_op->set_shift(m_sigma); + } +}; + + +} // namespace Spectra + +#endif // GEN_EIGS_REAL_SHIFT_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/GenEigsSolver.h b/src/external/Spectra/include/Spectra/GenEigsSolver.h new file mode 100644 index 00000000..a6960acf --- /dev/null +++ b/src/external/Spectra/include/Spectra/GenEigsSolver.h @@ -0,0 +1,160 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef GEN_EIGS_SOLVER_H +#define GEN_EIGS_SOLVER_H + +#include + +#include "GenEigsBase.h" +#include "Util/SelectionRule.h" +#include "MatOp/DenseGenMatProd.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This class implements the eigen solver for general real matrices, i.e., +/// to solve \f$Ax=\lambda x\f$ for a possibly non-symmetric \f$A\f$ matrix. +/// +/// Most of the background information documented in the SymEigsSolver class +/// also applies to the GenEigsSolver class here, except that the eigenvalues +/// and eigenvectors of a general matrix can now be complex-valued. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the requested eigenvalues, for example `LARGEST_MAGN` +/// to retrieve eigenvalues with the largest magnitude. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class. Users could either +/// use the wrapper classes such as DenseGenMatProd and +/// SparseGenMatProd, or define their +/// own that implements all the public member functions as in +/// DenseGenMatProd. +/// +/// An example that illustrates the usage of GenEigsSolver is give below: +/// +/// \code{.cpp} +/// #include +/// #include +/// // is implicitly included +/// #include +/// +/// using namespace Spectra; +/// +/// int main() +/// { +/// // We are going to calculate the eigenvalues of M +/// Eigen::MatrixXd M = Eigen::MatrixXd::Random(10, 10); +/// +/// // Construct matrix operation object using the wrapper class +/// DenseGenMatProd op(M); +/// +/// // Construct eigen solver object, requesting the largest +/// // (in magnitude, or norm) three eigenvalues +/// GenEigsSolver< double, LARGEST_MAGN, DenseGenMatProd > eigs(&op, 3, 6); +/// +/// // Initialize and compute +/// eigs.init(); +/// int nconv = eigs.compute(); +/// +/// // Retrieve results +/// Eigen::VectorXcd evalues; +/// if(eigs.info() == SUCCESSFUL) +/// evalues = eigs.eigenvalues(); +/// +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// +/// return 0; +/// } +/// \endcode +/// +/// And also an example for sparse matrices: +/// +/// \code{.cpp} +/// #include +/// #include +/// #include +/// #include +/// #include +/// +/// using namespace Spectra; +/// +/// int main() +/// { +/// // A band matrix with 1 on the main diagonal, 2 on the below-main subdiagonal, +/// // and 3 on the above-main subdiagonal +/// const int n = 10; +/// Eigen::SparseMatrix M(n, n); +/// M.reserve(Eigen::VectorXi::Constant(n, 3)); +/// for(int i = 0; i < n; i++) +/// { +/// M.insert(i, i) = 1.0; +/// if(i > 0) +/// M.insert(i - 1, i) = 3.0; +/// if(i < n - 1) +/// M.insert(i + 1, i) = 2.0; +/// } +/// +/// // Construct matrix operation object using the wrapper class SparseGenMatProd +/// SparseGenMatProd op(M); +/// +/// // Construct eigen solver object, requesting the largest three eigenvalues +/// GenEigsSolver< double, LARGEST_MAGN, SparseGenMatProd > eigs(&op, 3, 6); +/// +/// // Initialize and compute +/// eigs.init(); +/// int nconv = eigs.compute(); +/// +/// // Retrieve results +/// Eigen::VectorXcd evalues; +/// if(eigs.info() == SUCCESSFUL) +/// evalues = eigs.eigenvalues(); +/// +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// +/// return 0; +/// } +/// \endcode +template < typename Scalar = double, + int SelectionRule = LARGEST_MAGN, + typename OpType = DenseGenMatProd > +class GenEigsSolver: public GenEigsBase +{ +private: + typedef Eigen::Index Index; + +public: + /// + /// Constructor to create a solver object. + /// + /// \param op Pointer to the matrix operation object, which should implement + /// the matrix-vector multiplication operation of \f$A\f$: + /// calculating \f$Av\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper class such as DenseGenMatProd, or + /// define their own that implements all the public member functions + /// as in DenseGenMatProd. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-2\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev+2 \le ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev + 1\f$. + /// + GenEigsSolver(OpType* op, Index nev, Index ncv) : + GenEigsBase(op, NULL, nev, ncv) + {} +}; + + +} // namespace Spectra + +#endif // GEN_EIGS_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/Arnoldi.h b/src/external/Spectra/include/Spectra/LinAlg/Arnoldi.h new file mode 100644 index 00000000..b9fa75b5 --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/Arnoldi.h @@ -0,0 +1,283 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef ARNOLDI_H +#define ARNOLDI_H + +#include +#include // std::sqrt +#include // std::invalid_argument +#include // std::stringstream + +#include "../MatOp/internal/ArnoldiOp.h" +#include "../Util/TypeTraits.h" +#include "../Util/SimpleRandom.h" +#include "UpperHessenbergQR.h" +#include "DoubleShiftQR.h" + +namespace Spectra { + + +// Arnoldi factorization A * V = V * H + f * e' +// A: n x n +// V: n x k +// H: k x k +// f: n x 1 +// e: [0, ..., 0, 1] +// V and H are allocated of dimension m, so the maximum value of k is m +template +class Arnoldi +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapMat; + typedef Eigen::Map MapVec; + typedef Eigen::Map MapConstMat; + typedef Eigen::Map MapConstVec; + +protected: + ArnoldiOpType m_op; // Operators for the Arnoldi factorization + + const Index m_n; // dimension of A + const Index m_m; // maximum dimension of subspace V + Index m_k; // current dimension of subspace V + + Matrix m_fac_V; // V matrix in the Arnoldi factorization + Matrix m_fac_H; // H matrix in the Arnoldi factorization + Vector m_fac_f; // residual in the Arnoldi factorization + Scalar m_beta; // ||f||, B-norm of f + + const Scalar m_near_0; // a very small value, but 1.0 / m_near_0 does not overflow + // ~= 1e-307 for the "double" type + const Scalar m_eps; // the machine precision, ~= 1e-16 for the "double" type + + // Given orthonormal basis functions V, find a nonzero vector f such that V'Bf = 0 + // Assume that f has been properly allocated + void expand_basis(MapConstMat& V, const Index seed, Vector& f, Scalar& fnorm) + { + using std::sqrt; + + const Scalar thresh = m_eps * sqrt(Scalar(m_n)); + Vector Vf(V.cols()); + for(Index iter = 0; iter < 5; iter++) + { + // Randomly generate a new vector and orthogonalize it against V + SimpleRandom rng(seed + 123 * iter); + f.noalias() = rng.random_vec(m_n); + // f <- f - V * V'Bf, so that f is orthogonal to V in B-norm + m_op.trans_product(V, f, Vf); + f.noalias() -= V * Vf; + // fnorm <- ||f|| + fnorm = m_op.norm(f); + + // If fnorm is too close to zero, we try a new random vector, + // otherwise return the result + if(fnorm >= thresh) + return; + } + } + +public: + Arnoldi(const ArnoldiOpType& op, Index m) : + m_op(op), m_n(op.rows()), m_m(m), m_k(0), + m_near_0(TypeTraits::min() * Scalar(10)), + m_eps(Eigen::NumTraits::epsilon()) + {} + + virtual ~Arnoldi() {} + + // Const-reference to internal structures + const Matrix& matrix_V() const { return m_fac_V; } + const Matrix& matrix_H() const { return m_fac_H; } + const Vector& vector_f() const { return m_fac_f; } + Scalar f_norm() const { return m_beta; } + Index subspace_dim() const { return m_k; } + + // Initialize with an operator and an initial vector + void init(MapConstVec& v0, Index& op_counter) + { + m_fac_V.resize(m_n, m_m); + m_fac_H.resize(m_m, m_m); + m_fac_f.resize(m_n); + m_fac_H.setZero(); + + // Verify the initial vector + const Scalar v0norm = m_op.norm(v0); + if(v0norm < m_near_0) + throw std::invalid_argument("initial residual vector cannot be zero"); + + // Points to the first column of V + MapVec v(m_fac_V.data(), m_n); + + // Normalize + v.noalias() = v0 / v0norm; + + // Compute H and f + Vector w(m_n); + m_op.perform_op(v.data(), w.data()); + op_counter++; + + m_fac_H(0, 0) = m_op.inner_product(v, w); + m_fac_f.noalias() = w - v * m_fac_H(0, 0); + + // In some cases f is zero in exact arithmetics, but due to rounding errors + // it may contain tiny fluctuations. When this happens, we force f to be zero + if(m_fac_f.cwiseAbs().maxCoeff() < m_eps) + { + m_fac_f.setZero(); + m_beta = Scalar(0); + } else { + m_beta = m_op.norm(m_fac_f); + } + + // Indicate that this is a step-1 factorization + m_k = 1; + } + + // Arnoldi factorization starting from step-k + virtual void factorize_from(Index from_k, Index to_m, Index& op_counter) + { + using std::sqrt; + + if(to_m <= from_k) return; + + if(from_k > m_k) + { + std::stringstream msg; + msg << "Arnoldi: from_k (= " << from_k << + ") is larger than the current subspace dimension (= " << + m_k << ")"; + throw std::invalid_argument(msg.str()); + } + + const Scalar beta_thresh = m_eps * sqrt(Scalar(m_n)); + + // Pre-allocate vectors + Vector Vf(to_m); + Vector w(m_n); + + // Keep the upperleft k x k submatrix of H and set other elements to 0 + m_fac_H.rightCols(m_m - from_k).setZero(); + m_fac_H.block(from_k, 0, m_m - from_k, from_k).setZero(); + + for(Index i = from_k; i <= to_m - 1; i++) + { + bool restart = false; + // If beta = 0, then the next V is not full rank + // We need to generate a new residual vector that is orthogonal + // to the current V, which we call a restart + if(m_beta < m_near_0) + { + MapConstMat V(m_fac_V.data(), m_n, i); // The first i columns + expand_basis(V, 2 * i, m_fac_f, m_beta); + restart = true; + } + + // v <- f / ||f|| + m_fac_V.col(i).noalias() = m_fac_f / m_beta; // The (i+1)-th column + + // Note that H[i+1, i] equals to the unrestarted beta + m_fac_H(i, i - 1) = restart ? Scalar(0) : m_beta; + + // w <- A * v, v = m_fac_V.col(i) + m_op.perform_op(&m_fac_V(0, i), w.data()); + op_counter++; + + const Index i1 = i + 1; + // First i+1 columns of V + MapConstMat Vs(m_fac_V.data(), m_n, i1); + // h = m_fac_H(0:i, i) + MapVec h(&m_fac_H(0, i), i1); + // h <- V'Bw + m_op.trans_product(Vs, w, h); + + // f <- w - V * h + m_fac_f.noalias() = w - Vs * h; + m_beta = m_op.norm(m_fac_f); + + if(m_beta > Scalar(0.717) * m_op.norm(h)) + continue; + + // f/||f|| is going to be the next column of V, so we need to test + // whether V'B(f/||f||) ~= 0 + m_op.trans_product(Vs, m_fac_f, Vf.head(i1)); + Scalar ortho_err = Vf.head(i1).cwiseAbs().maxCoeff(); + // If not, iteratively correct the residual + int count = 0; + while(count < 5 && ortho_err > m_eps * m_beta) + { + // There is an edge case: when beta=||f|| is close to zero, f mostly consists + // of noises of rounding errors, so the test [ortho_err < eps * beta] is very + // likely to fail. In particular, if beta=0, then the test is ensured to fail. + // Hence when this happens, we force f to be zero, and then restart in the + // next iteration. + if(m_beta < beta_thresh) + { + m_fac_f.setZero(); + m_beta = Scalar(0); + break; + } + + // f <- f - V * Vf + m_fac_f.noalias() -= Vs * Vf.head(i1); + // h <- h + Vf + h.noalias() += Vf.head(i1); + // beta <- ||f|| + m_beta = m_op.norm(m_fac_f); + + m_op.trans_product(Vs, m_fac_f, Vf.head(i1)); + ortho_err = Vf.head(i1).cwiseAbs().maxCoeff(); + count++; + } + } + + // Indicate that this is a step-m factorization + m_k = to_m; + } + + // Apply H -> Q'HQ, where Q is from a double shift QR decomposition + void compress_H(const DoubleShiftQR& decomp) + { + decomp.matrix_QtHQ(m_fac_H); + m_k -= 2; + } + + // Apply H -> Q'HQ, where Q is from an upper Hessenberg QR decomposition + void compress_H(const UpperHessenbergQR& decomp) + { + decomp.matrix_QtHQ(m_fac_H); + m_k--; + } + + // Apply V -> VQ and compute the new f. + // Should be called after compress_H(), since m_k is updated there. + // Only need to update the first k+1 columns of V + // The first (m - k + i) elements of the i-th column of Q are non-zero, + // and the rest are zero + void compress_V(const Matrix& Q) + { + Matrix Vs(m_n, m_k + 1); + for(Index i = 0; i < m_k; i++) + { + const Index nnz = m_m - m_k + i + 1; + MapConstVec q(&Q(0, i), nnz); + Vs.col(i).noalias() = m_fac_V.leftCols(nnz) * q; + } + Vs.col(m_k).noalias() = m_fac_V * Q.col(m_k); + m_fac_V.leftCols(m_k + 1).noalias() = Vs; + + Vector fk = m_fac_f * Q(m_m - 1, m_k - 1) + m_fac_V.col(m_k) * m_fac_H(m_k, m_k - 1); + m_fac_f.swap(fk); + m_beta = m_op.norm(m_fac_f); + } +}; + + +} // namespace Spectra + +#endif // ARNOLDI_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/BKLDLT.h b/src/external/Spectra/include/Spectra/LinAlg/BKLDLT.h new file mode 100644 index 00000000..5509749b --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/BKLDLT.h @@ -0,0 +1,522 @@ +// Copyright (C) 2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef BK_LDLT_H +#define BK_LDLT_H + +#include +#include +#include + +#include "../Util/CompInfo.h" + +namespace Spectra { + + +// Bunch-Kaufman LDLT decomposition +// References: +// 1. Bunch, J. R., & Kaufman, L. (1977). Some stable methods for calculating inertia and solving symmetric linear systems. +// Mathematics of computation, 31(137), 163-179. +// 2. Golub, G. H., & Van Loan, C. F. (2012). Matrix computations (Vol. 3). JHU press. Section 4.4. +// 3. Bunch-Parlett diagonal pivoting +// 4. Ashcraft, C., Grimes, R. G., & Lewis, J. G. (1998). Accurate symmetric indefinite linear equation solvers. +// SIAM Journal on Matrix Analysis and Applications, 20(2), 513-561. +template +class BKLDLT +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapVec; + typedef Eigen::Map MapConstVec; + + typedef Eigen::Matrix IntVector; + typedef Eigen::Ref GenericVector; + typedef Eigen::Ref GenericMatrix; + typedef const Eigen::Ref ConstGenericMatrix; + typedef const Eigen::Ref ConstGenericVector; + + Index m_n; + Vector m_data; // storage for a lower-triangular matrix + std::vector m_colptr; // pointers to columns + IntVector m_perm; // [-2, -1, 3, 1, 4, 5]: 0 <-> 2, 1 <-> 1, 2 <-> 3, 3 <-> 1, 4 <-> 4, 5 <-> 5 + std::vector< std::pair > m_permc; // compressed version of m_perm: [(0, 2), (2, 3), (3, 1)] + + bool m_computed; + int m_info; + + // Access to elements + // Pointer to the k-th column + Scalar* col_pointer(Index k) { return m_colptr[k]; } + // A[i, j] -> m_colptr[j][i - j], i >= j + Scalar& coeff(Index i, Index j) { return m_colptr[j][i - j]; } + const Scalar& coeff(Index i, Index j) const { return m_colptr[j][i - j]; } + // A[i, i] -> m_colptr[i][0] + Scalar& diag_coeff(Index i) { return m_colptr[i][0]; } + const Scalar& diag_coeff(Index i) const { return m_colptr[i][0]; } + + // Compute column pointers + void compute_pointer() + { + m_colptr.clear(); + m_colptr.reserve(m_n); + Scalar* head = m_data.data(); + + for(Index i = 0; i < m_n; i++) + { + m_colptr.push_back(head); + head += (m_n - i); + } + } + + // Copy mat - shift * I to m_data + void copy_data(ConstGenericMatrix& mat, int uplo, const Scalar& shift) + { + if(uplo == Eigen::Lower) + { + for(Index j = 0; j < m_n; j++) + { + const Scalar* begin = &mat.coeffRef(j, j); + const Index len = m_n - j; + std::copy(begin, begin + len, col_pointer(j)); + diag_coeff(j) -= shift; + } + } else { + Scalar* dest = m_data.data(); + for(Index i = 0; i < m_n; i++) + { + for(Index j = i; j < m_n; j++, dest++) + { + *dest = mat.coeff(i, j); + } + diag_coeff(i) -= shift; + } + } + } + + // Compute compressed permutations + void compress_permutation() + { + for(Index i = 0; i < m_n; i++) + { + // Recover the permutation action + const Index perm = (m_perm[i] >= 0) ? (m_perm[i]) : (-m_perm[i] - 1); + if(perm != i) + m_permc.push_back(std::make_pair(i, perm)); + } + } + + // Working on the A[k:end, k:end] submatrix + // Exchange k <-> r + // Assume r >= k + void pivoting_1x1(Index k, Index r) + { + // No permutation + if(k == r) + { + m_perm[k] = r; + return; + } + + // A[k, k] <-> A[r, r] + std::swap(diag_coeff(k), diag_coeff(r)); + + // A[(r+1):end, k] <-> A[(r+1):end, r] + std::swap_ranges(&coeff(r + 1, k), col_pointer(k + 1), &coeff(r + 1, r)); + + // A[(k+1):(r-1), k] <-> A[r, (k+1):(r-1)] + Scalar* src = &coeff(k + 1, k); + for(Index j = k + 1; j < r; j++, src++) + { + std::swap(*src, coeff(r, j)); + } + + m_perm[k] = r; + } + + // Working on the A[k:end, k:end] submatrix + // Exchange [k+1, k] <-> [r, p] + // Assume p >= k, r >= k+1 + void pivoting_2x2(Index k, Index r, Index p) + { + pivoting_1x1(k, p); + pivoting_1x1(k + 1, r); + + // A[k+1, k] <-> A[r, k] + std::swap(coeff(k + 1, k), coeff(r, k)); + + // Use negative signs to indicate a 2x2 block + // Also minus one to distinguish a negative zero from a positive zero + m_perm[k] = -m_perm[k] - 1; + m_perm[k + 1] = -m_perm[k + 1] - 1; + } + + // A[r1, c1:c2] <-> A[r2, c1:c2] + // Assume r2 >= r1 > c2 >= c1 + void interchange_rows(Index r1, Index r2, Index c1, Index c2) + { + if(r1 == r2) + return; + + for(Index j = c1; j <= c2; j++) + { + std::swap(coeff(r1, j), coeff(r2, j)); + } + } + + // lambda = |A[r, k]| = max{|A[k+1, k]|, ..., |A[end, k]|} + // Largest (in magnitude) off-diagonal element in the first column of the current reduced matrix + // r is the row index + // Assume k < end + Scalar find_lambda(Index k, Index& r) + { + using std::abs; + + const Scalar* head = col_pointer(k); // => A[k, k] + const Scalar* end = col_pointer(k + 1); + // Start with r=k+1, lambda=A[k+1, k] + r = k + 1; + Scalar lambda = abs(head[1]); + // Scan remaining elements + for(const Scalar* ptr = head + 2; ptr < end; ptr++) + { + const Scalar abs_elem = abs(*ptr); + if(lambda < abs_elem) + { + lambda = abs_elem; + r = k + (ptr - head); + } + } + + return lambda; + } + + // sigma = |A[p, r]| = max {|A[k, r]|, ..., |A[end, r]|} \ {A[r, r]} + // Largest (in magnitude) off-diagonal element in the r-th column of the current reduced matrix + // p is the row index + // Assume k < r < end + Scalar find_sigma(Index k, Index r, Index& p) + { + using std::abs; + + // First search A[r+1, r], ..., A[end, r], which has the same task as find_lambda() + // If r == end, we skip this search + Scalar sigma = Scalar(-1); + if(r < m_n - 1) + sigma = find_lambda(r, p); + + // Then search A[k, r], ..., A[r-1, r], which maps to A[r, k], ..., A[r, r-1] + for(Index j = k; j < r; j++) + { + const Scalar abs_elem = abs(coeff(r, j)); + if(sigma < abs_elem) + { + sigma = abs_elem; + p = j; + } + } + + return sigma; + } + + // Generate permutations and apply to A + // Return true if the resulting pivoting is 1x1, and false if 2x2 + bool permutate_mat(Index k, const Scalar& alpha) + { + using std::abs; + + Index r = k, p = k; + const Scalar lambda = find_lambda(k, r); + + // If lambda=0, no need to interchange + if(lambda > Scalar(0)) + { + const Scalar abs_akk = abs(diag_coeff(k)); + // If |A[k, k]| >= alpha * lambda, no need to interchange + if(abs_akk < alpha * lambda) + { + const Scalar sigma = find_sigma(k, r, p); + + // If sigma * |A[k, k]| >= alpha * lambda^2, no need to interchange + if(sigma * abs_akk < alpha * lambda * lambda) + { + if(abs_akk >= alpha * sigma) + { + // Permutation on A + pivoting_1x1(k, r); + + // Permutation on L + interchange_rows(k, r, 0, k - 1); + return true; + } else { + // There are two versions of permutation here + // 1. A[k+1, k] <-> A[r, k] + // 2. A[k+1, k] <-> A[r, p], where p >= k and r >= k+1 + // + // Version 1 and 2 are used by Ref[1] and Ref[2], respectively + + // Version 1 implementation + p = k; + + // Version 2 implementation + // [r, p] and [p, r] are symmetric, but we need to make sure + // p >= k and r >= k+1, so it is safe to always make r > p + // One exception is when min{r,p} == k+1, in which case we make + // r = k+1, so that only one permutation needs to be performed + /* const Index rp_min = std::min(r, p); + const Index rp_max = std::max(r, p); + if(rp_min == k + 1) + { + r = rp_min; p = rp_max; + } else { + r = rp_max; p = rp_min; + } */ + + // Right now we use Version 1 since it reduces the overhead of interchange + + // Permutation on A + pivoting_2x2(k, r, p); + // Permutation on L + interchange_rows(k, p, 0, k - 1); + interchange_rows(k + 1, r, 0, k - 1); + return false; + } + } + } + } + + return true; + } + + // E = [e11, e12] + // [e21, e22] + // Overwrite E with inv(E) + void inverse_inplace_2x2(Scalar& e11, Scalar& e21, Scalar& e22) const + { + // inv(E) = [d11, d12], d11 = e22/delta, d21 = -e21/delta, d22 = e11/delta + // [d21, d22] + const Scalar delta = e11 * e22 - e21 * e21; + std::swap(e11, e22); + e11 /= delta; + e22 /= delta; + e21 = -e21 / delta; + } + + // Return value is the status, SUCCESSFUL/NUMERICAL_ISSUE + int gaussian_elimination_1x1(Index k) + { + // D = 1 / A[k, k] + const Scalar akk = diag_coeff(k); + // Return NUMERICAL_ISSUE if not invertible + if(akk == Scalar(0)) + return NUMERICAL_ISSUE; + + diag_coeff(k) = Scalar(1) / akk; + + // B -= l * l' / A[k, k], B := A[(k+1):end, (k+1):end], l := L[(k+1):end, k] + Scalar* lptr = col_pointer(k) + 1; + const Index ldim = m_n - k - 1; + MapVec l(lptr, ldim); + for(Index j = 0; j < ldim; j++) + { + MapVec(col_pointer(j + k + 1), ldim - j).noalias() -= (lptr[j] / akk) * l.tail(ldim - j); + } + + // l /= A[k, k] + l /= akk; + + return SUCCESSFUL; + } + + // Return value is the status, SUCCESSFUL/NUMERICAL_ISSUE + int gaussian_elimination_2x2(Index k) + { + // D = inv(E) + Scalar& e11 = diag_coeff(k); + Scalar& e21 = coeff(k + 1, k); + Scalar& e22 = diag_coeff(k + 1); + // Return NUMERICAL_ISSUE if not invertible + if(e11 * e22 - e21 * e21 == Scalar(0)) + return NUMERICAL_ISSUE; + + inverse_inplace_2x2(e11, e21, e22); + + // X = l * inv(E), l := L[(k+2):end, k:(k+1)] + Scalar* l1ptr = &coeff(k + 2, k); + Scalar* l2ptr = &coeff(k + 2, k + 1); + const Index ldim = m_n - k - 2; + MapVec l1(l1ptr, ldim), l2(l2ptr, ldim); + + Eigen::Matrix X(ldim, 2); + X.col(0).noalias() = l1 * e11 + l2 * e21; + X.col(1).noalias() = l1 * e21 + l2 * e22; + + // B -= l * inv(E) * l' = X * l', B = A[(k+2):end, (k+2):end] + for(Index j = 0; j < ldim; j++) + { + MapVec(col_pointer(j + k + 2), ldim - j).noalias() -= (X.col(0).tail(ldim - j) * l1ptr[j] + X.col(1).tail(ldim - j) * l2ptr[j]); + } + + // l = X + l1.noalias() = X.col(0); + l2.noalias() = X.col(1); + + return SUCCESSFUL; + } + +public: + BKLDLT() : + m_n(0), m_computed(false), m_info(NOT_COMPUTED) + {} + + // Factorize mat - shift * I + BKLDLT(ConstGenericMatrix& mat, int uplo = Eigen::Lower, const Scalar& shift = Scalar(0)) : + m_n(mat.rows()), m_computed(false), m_info(NOT_COMPUTED) + { + compute(mat, uplo, shift); + } + + void compute(ConstGenericMatrix& mat, int uplo = Eigen::Lower, const Scalar& shift = Scalar(0)) + { + using std::abs; + + m_n = mat.rows(); + if(m_n != mat.cols()) + throw std::invalid_argument("BKLDLT: matrix must be square"); + + m_perm.setLinSpaced(m_n, 0, m_n - 1); + m_permc.clear(); + + // Copy data + m_data.resize((m_n * (m_n + 1)) / 2); + compute_pointer(); + copy_data(mat, uplo, shift); + + const Scalar alpha = (1.0 + std::sqrt(17.0)) / 8.0; + Index k = 0; + for(k = 0; k < m_n - 1; k++) + { + // 1. Interchange rows and columns of A, and save the result to m_perm + bool is_1x1 = permutate_mat(k, alpha); + + // 2. Gaussian elimination + if(is_1x1) + { + m_info = gaussian_elimination_1x1(k); + } else { + m_info = gaussian_elimination_2x2(k); + k++; + } + + // 3. Check status + if(m_info != SUCCESSFUL) + break; + } + // Invert the last 1x1 block if it exists + if(k == m_n - 1) + { + const Scalar akk = diag_coeff(k); + if(akk == Scalar(0)) + m_info = NUMERICAL_ISSUE; + + diag_coeff(k) = Scalar(1) / diag_coeff(k); + } + + compress_permutation(); + + m_computed = true; + } + + // Solve Ax=b + void solve_inplace(GenericVector b) const + { + if(!m_computed) + throw std::logic_error("BKLDLT: need to call compute() first"); + + // PAP' = LDL' + // 1. b -> Pb + Scalar* x = b.data(); + MapVec res(x, m_n); + Index npermc = m_permc.size(); + for(Index i = 0; i < npermc; i++) + { + std::swap(x[m_permc[i].first], x[m_permc[i].second]); + } + + // 2. Lz = Pb + // If m_perm[end] < 0, then end with m_n - 3, otherwise end with m_n - 2 + const Index end = (m_perm[m_n - 1] < 0) ? (m_n - 3) : (m_n - 2); + for(Index i = 0; i <= end; i++) + { + const Index b1size = m_n - i - 1; + const Index b2size = b1size - 1; + if(m_perm[i] >= 0) + { + MapConstVec l(&coeff(i + 1, i), b1size); + res.segment(i + 1, b1size).noalias() -= l * x[i]; + } else { + MapConstVec l1(&coeff(i + 2, i), b2size); + MapConstVec l2(&coeff(i + 2, i + 1), b2size); + res.segment(i + 2, b2size).noalias() -= (l1 * x[i] + l2 * x[i + 1]); + i++; + } + } + + // 3. Dw = z + for(Index i = 0; i < m_n; i++) + { + const Scalar e11 = diag_coeff(i); + if(m_perm[i] >= 0) + { + x[i] *= e11; + } else { + const Scalar e21 = coeff(i + 1, i), e22 = diag_coeff(i + 1); + const Scalar wi = x[i] * e11 + x[i + 1] * e21; + x[i + 1] = x[i] * e21 + x[i + 1] * e22; + x[i] = wi; + i++; + } + } + + // 4. L'y = w + // If m_perm[end] < 0, then start with m_n - 3, otherwise start with m_n - 2 + Index i = (m_perm[m_n - 1] < 0) ? (m_n - 3) : (m_n - 2); + for(; i >= 0; i--) + { + const Index ldim = m_n - i - 1; + MapConstVec l(&coeff(i + 1, i), ldim); + x[i] -= res.segment(i + 1, ldim).dot(l); + + if(m_perm[i] < 0) + { + MapConstVec l2(&coeff(i + 1, i - 1), ldim); + x[i - 1] -= res.segment(i + 1, ldim).dot(l2); + i--; + } + } + + // 5. x = P'y + for(Index i = npermc - 1; i >= 0; i--) + { + std::swap(x[m_permc[i].first], x[m_permc[i].second]); + } + } + + Vector solve(ConstGenericVector& b) const + { + Vector res = b; + solve_inplace(res); + return res; + } + + int info() const { return m_info; } +}; + + +} // namespace Spectra + +#endif // BK_LDLT_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/DoubleShiftQR.h b/src/external/Spectra/include/Spectra/LinAlg/DoubleShiftQR.h new file mode 100644 index 00000000..2191909a --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/DoubleShiftQR.h @@ -0,0 +1,378 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DOUBLE_SHIFT_QR_H +#define DOUBLE_SHIFT_QR_H + +#include +#include // std::vector +#include // std::min, std::fill, std::copy +#include // std::abs, std::sqrt, std::pow +#include // std::invalid_argument, std::logic_error + +#include "../Util/TypeTraits.h" + +namespace Spectra { + + +template +class DoubleShiftQR +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Matrix3X; + typedef Eigen::Matrix Vector; + typedef Eigen::Array IntArray; + + typedef Eigen::Ref GenericMatrix; + typedef const Eigen::Ref ConstGenericMatrix; + + Index m_n; // Dimension of the matrix + Matrix m_mat_H; // A copy of the matrix to be factorized + Scalar m_shift_s; // Shift constant + Scalar m_shift_t; // Shift constant + Matrix3X m_ref_u; // Householder reflectors + IntArray m_ref_nr; // How many rows does each reflector affects + // 3 - A general reflector + // 2 - A Givens rotation + // 1 - An identity transformation + const Scalar m_near_0; // a very small value, but 1.0 / m_safe_min does not overflow + // ~= 1e-307 for the "double" type + const Scalar m_eps; // the machine precision, + // e.g. ~= 1e-16 for the "double" type + const Scalar m_eps_rel; + const Scalar m_eps_abs; + bool m_computed; // Whether matrix has been factorized + + void compute_reflector(const Scalar& x1, const Scalar& x2, const Scalar& x3, Index ind) + { + using std::abs; + + Scalar* u = &m_ref_u.coeffRef(0, ind); + unsigned char* nr = m_ref_nr.data(); + // In general case the reflector affects 3 rows + nr[ind] = 3; + Scalar x2x3 = Scalar(0); + // If x3 is zero, decrease nr by 1 + if(abs(x3) < m_near_0) + { + // If x2 is also zero, nr will be 1, and we can exit this function + if(abs(x2) < m_near_0) + { + nr[ind] = 1; + return; + } else { + nr[ind] = 2; + } + x2x3 = abs(x2); + } else { + x2x3 = Eigen::numext::hypot(x2, x3); + } + + // x1' = x1 - rho * ||x|| + // rho = -sign(x1), if x1 == 0, we choose rho = 1 + Scalar x1_new = x1 - ((x1 <= 0) - (x1 > 0)) * Eigen::numext::hypot(x1, x2x3); + Scalar x_norm = Eigen::numext::hypot(x1_new, x2x3); + // Double check the norm of new x + if(x_norm < m_near_0) + { + nr[ind] = 1; + return; + } + u[0] = x1_new / x_norm; + u[1] = x2 / x_norm; + u[2] = x3 / x_norm; + } + + void compute_reflector(const Scalar* x, Index ind) + { + compute_reflector(x[0], x[1], x[2], ind); + } + + // Update the block X = H(il:iu, il:iu) + void update_block(Index il, Index iu) + { + // Block size + const Index bsize = iu - il + 1; + + // If block size == 1, there is no need to apply reflectors + if(bsize == 1) + { + m_ref_nr.coeffRef(il) = 1; + return; + } + + const Scalar x00 = m_mat_H.coeff(il, il), + x01 = m_mat_H.coeff(il, il + 1), + x10 = m_mat_H.coeff(il + 1, il), + x11 = m_mat_H.coeff(il + 1, il + 1); + // m00 = x00 * (x00 - s) + x01 * x10 + t + const Scalar m00 = x00 * (x00 - m_shift_s) + x01 * x10 + m_shift_t; + // m10 = x10 * (x00 + x11 - s) + const Scalar m10 = x10 * (x00 + x11 - m_shift_s); + + // For block size == 2, do a Givens rotation on M = X * X - s * X + t * I + if(bsize == 2) + { + // This causes nr=2 + compute_reflector(m00, m10, 0, il); + // Apply the reflector to X + apply_PX(m_mat_H.block(il, il, 2, m_n - il), m_n, il); + apply_XP(m_mat_H.block(0, il, il + 2, 2), m_n, il); + + m_ref_nr.coeffRef(il + 1) = 1; + return; + } + + // For block size >=3, use the regular strategy + // m20 = x21 * x10 + const Scalar m20 = m_mat_H.coeff(il + 2, il + 1) * m_mat_H.coeff(il + 1, il); + compute_reflector(m00, m10, m20, il); + + // Apply the first reflector + apply_PX(m_mat_H.block(il, il, 3, m_n - il), m_n, il); + apply_XP(m_mat_H.block(0, il, il + std::min(bsize, Index(4)), 3), m_n, il); + + // Calculate the following reflectors + // If entering this loop, block size is at least 4. + for(Index i = 1; i < bsize - 2; i++) + { + compute_reflector(&m_mat_H.coeffRef(il + i, il + i - 1), il + i); + // Apply the reflector to X + apply_PX(m_mat_H.block(il + i, il + i - 1, 3, m_n - il - i + 1), m_n, il + i); + apply_XP(m_mat_H.block(0, il + i, il + std::min(bsize, Index(i + 4)), 3), m_n, il + i); + } + + // The last reflector + // This causes nr=2 + compute_reflector(m_mat_H.coeff(iu - 1, iu - 2), m_mat_H.coeff(iu, iu - 2), 0, iu - 1); + // Apply the reflector to X + apply_PX(m_mat_H.block(iu - 1, iu - 2, 2, m_n - iu + 2), m_n, iu - 1); + apply_XP(m_mat_H.block(0, iu - 1, il + bsize, 2), m_n, iu - 1); + + m_ref_nr.coeffRef(iu) = 1; + } + + // P = I - 2 * u * u' = P' + // PX = X - 2 * u * (u'X) + void apply_PX(GenericMatrix X, Index stride, Index u_ind) const + { + const Index nr = m_ref_nr.coeff(u_ind); + if(nr == 1) + return; + + const Scalar u0 = m_ref_u.coeff(0, u_ind), + u1 = m_ref_u.coeff(1, u_ind); + const Scalar u0_2 = Scalar(2) * u0, + u1_2 = Scalar(2) * u1; + + const Index nrow = X.rows(); + const Index ncol = X.cols(); + + Scalar* xptr = X.data(); + if(nr == 2 || nrow == 2) + { + for(Index i = 0; i < ncol; i++, xptr += stride) + { + const Scalar tmp = u0_2 * xptr[0] + u1_2 * xptr[1]; + xptr[0] -= tmp * u0; + xptr[1] -= tmp * u1; + } + } else { + const Scalar u2 = m_ref_u.coeff(2, u_ind); + const Scalar u2_2 = Scalar(2) * u2; + for(Index i = 0; i < ncol; i++, xptr += stride) + { + const Scalar tmp = u0_2 * xptr[0] + u1_2 * xptr[1] + u2_2 * xptr[2]; + xptr[0] -= tmp * u0; + xptr[1] -= tmp * u1; + xptr[2] -= tmp * u2; + } + } + } + + // x is a pointer to a vector + // Px = x - 2 * dot(x, u) * u + void apply_PX(Scalar* x, Index u_ind) const + { + const Index nr = m_ref_nr.coeff(u_ind); + if(nr == 1) + return; + + const Scalar u0 = m_ref_u.coeff(0, u_ind), + u1 = m_ref_u.coeff(1, u_ind), + u2 = m_ref_u.coeff(2, u_ind); + + // When the reflector only contains two elements, u2 has been set to zero + const bool nr_is_2 = (nr == 2); + const Scalar dot2 = Scalar(2) * (x[0] * u0 + x[1] * u1 + (nr_is_2 ? 0 : (x[2] * u2))); + x[0] -= dot2 * u0; + x[1] -= dot2 * u1; + if(!nr_is_2) + x[2] -= dot2 * u2; + } + + // XP = X - 2 * (X * u) * u' + void apply_XP(GenericMatrix X, Index stride, Index u_ind) const + { + const Index nr = m_ref_nr.coeff(u_ind); + if(nr == 1) + return; + + const Scalar u0 = m_ref_u.coeff(0, u_ind), + u1 = m_ref_u.coeff(1, u_ind); + const Scalar u0_2 = Scalar(2) * u0, + u1_2 = Scalar(2) * u1; + + const int nrow = X.rows(); + const int ncol = X.cols(); + Scalar *X0 = X.data(), *X1 = X0 + stride; // X0 => X.col(0), X1 => X.col(1) + + if(nr == 2 || ncol == 2) + { + // tmp = 2 * u0 * X0 + 2 * u1 * X1 + // X0 => X0 - u0 * tmp + // X1 => X1 - u1 * tmp + for(Index i = 0; i < nrow; i++) + { + const Scalar tmp = u0_2 * X0[i] + u1_2 * X1[i]; + X0[i] -= tmp * u0; + X1[i] -= tmp * u1; + } + } else { + Scalar* X2 = X1 + stride; // X2 => X.col(2) + const Scalar u2 = m_ref_u.coeff(2, u_ind); + const Scalar u2_2 = Scalar(2) * u2; + for(Index i = 0; i < nrow; i++) + { + const Scalar tmp = u0_2 * X0[i] + u1_2 * X1[i] + u2_2 * X2[i]; + X0[i] -= tmp * u0; + X1[i] -= tmp * u1; + X2[i] -= tmp * u2; + } + } + } + +public: + DoubleShiftQR(Index size) : + m_n(size), + m_near_0(TypeTraits::min() * Scalar(10)), + m_eps(Eigen::NumTraits::epsilon()), + m_eps_rel(m_eps), + m_eps_abs(m_near_0 * (m_n / m_eps)), + m_computed(false) + {} + + DoubleShiftQR(ConstGenericMatrix& mat, const Scalar& s, const Scalar& t) : + m_n(mat.rows()), + m_mat_H(m_n, m_n), + m_shift_s(s), + m_shift_t(t), + m_ref_u(3, m_n), + m_ref_nr(m_n), + m_near_0(TypeTraits::min() * Scalar(10)), + m_eps(Eigen::NumTraits::epsilon()), + m_eps_rel(m_eps), + m_eps_abs(m_near_0 * (m_n / m_eps)), + m_computed(false) + { + compute(mat, s, t); + } + + void compute(ConstGenericMatrix& mat, const Scalar& s, const Scalar& t) + { + using std::abs; + + m_n = mat.rows(); + if(m_n != mat.cols()) + throw std::invalid_argument("DoubleShiftQR: matrix must be square"); + + m_mat_H.resize(m_n, m_n); + m_shift_s = s; + m_shift_t = t; + m_ref_u.resize(3, m_n); + m_ref_nr.resize(m_n); + + // Make a copy of mat + std::copy(mat.data(), mat.data() + mat.size(), m_mat_H.data()); + + // Obtain the indices of zero elements in the subdiagonal, + // so that H can be divided into several blocks + std::vector zero_ind; + zero_ind.reserve(m_n - 1); + zero_ind.push_back(0); + Scalar* Hii = m_mat_H.data(); + for(Index i = 0; i < m_n - 2; i++, Hii += (m_n + 1)) + { + // Hii[1] => m_mat_H(i + 1, i) + const Scalar h = abs(Hii[1]); + if(h <= 0 || h <= m_eps_rel * (abs(Hii[0]) + abs(Hii[m_n + 1]))) + { + Hii[1] = 0; + zero_ind.push_back(i + 1); + } + // Make sure m_mat_H is upper Hessenberg + // Zero the elements below m_mat_H(i + 1, i) + std::fill(Hii + 2, Hii + m_n - i, Scalar(0)); + } + zero_ind.push_back(m_n); + + for(std::vector::size_type i = 0; i < zero_ind.size() - 1; i++) + { + const Index start = zero_ind[i]; + const Index end = zero_ind[i + 1] - 1; + // Compute refelctors and update each block + update_block(start, end); + } + + m_computed = true; + } + + void matrix_QtHQ(Matrix& dest) const + { + if(!m_computed) + throw std::logic_error("DoubleShiftQR: need to call compute() first"); + + dest.noalias() = m_mat_H; + } + + // Q = P0 * P1 * ... + // Q'y = P_{n-2} * ... * P1 * P0 * y + void apply_QtY(Vector& y) const + { + if(!m_computed) + throw std::logic_error("DoubleShiftQR: need to call compute() first"); + + Scalar* y_ptr = y.data(); + const Index n1 = m_n - 1; + for(Index i = 0; i < n1; i++, y_ptr++) + { + apply_PX(y_ptr, i); + } + } + + // Q = P0 * P1 * ... + // YQ = Y * P0 * P1 * ... + void apply_YQ(GenericMatrix Y) const + { + if(!m_computed) + throw std::logic_error("DoubleShiftQR: need to call compute() first"); + + const Index nrow = Y.rows(); + const Index n2 = m_n - 2; + for(Index i = 0; i < n2; i++) + { + apply_XP(Y.block(0, i, nrow, 3), nrow, i); + } + apply_XP(Y.block(0, n2, nrow, 2), nrow, n2); + } +}; + + +} // namespace Spectra + +#endif // DOUBLE_SHIFT_QR_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/Lanczos.h b/src/external/Spectra/include/Spectra/LinAlg/Lanczos.h new file mode 100644 index 00000000..2301dd30 --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/Lanczos.h @@ -0,0 +1,170 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef LANCZOS_H +#define LANCZOS_H + +#include +#include // std::sqrt +#include // std::invalid_argument +#include // std::stringstream + +#include "Arnoldi.h" + +namespace Spectra { + + +// Lanczos factorization A * V = V * H + f * e' +// A: n x n +// V: n x k +// H: k x k +// f: n x 1 +// e: [0, ..., 0, 1] +// V and H are allocated of dimension m, so the maximum value of k is m +template +class Lanczos: public Arnoldi +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapMat; + typedef Eigen::Map MapVec; + typedef Eigen::Map MapConstMat; + typedef Eigen::Map MapConstVec; + + using Arnoldi::m_op; + using Arnoldi::m_n; + using Arnoldi::m_m; + using Arnoldi::m_k; + using Arnoldi::m_fac_V; + using Arnoldi::m_fac_H; + using Arnoldi::m_fac_f; + using Arnoldi::m_beta; + using Arnoldi::m_near_0; + using Arnoldi::m_eps; + +public: + Lanczos(const ArnoldiOpType& op, Index m) : + Arnoldi(op, m) + {} + + // Lanczos factorization starting from step-k + void factorize_from(Index from_k, Index to_m, Index& op_counter) + { + using std::sqrt; + + if(to_m <= from_k) return; + + if(from_k > m_k) + { + std::stringstream msg; + msg << "Lanczos: from_k (= " << from_k << + ") is larger than the current subspace dimension (= " << + m_k << ")"; + throw std::invalid_argument(msg.str()); + } + + const Scalar beta_thresh = m_eps * sqrt(Scalar(m_n)); + + // Pre-allocate vectors + Vector Vf(to_m); + Vector w(m_n); + + // Keep the upperleft k x k submatrix of H and set other elements to 0 + m_fac_H.rightCols(m_m - from_k).setZero(); + m_fac_H.block(from_k, 0, m_m - from_k, from_k).setZero(); + + for(Index i = from_k; i <= to_m - 1; i++) + { + bool restart = false; + // If beta = 0, then the next V is not full rank + // We need to generate a new residual vector that is orthogonal + // to the current V, which we call a restart + if(m_beta < m_near_0) + { + MapConstMat V(m_fac_V.data(), m_n, i); // The first i columns + this->expand_basis(V, 2 * i, m_fac_f, m_beta); + restart = true; + } + + // v <- f / ||f|| + MapVec v(&m_fac_V(0, i), m_n); // The (i+1)-th column + v.noalias() = m_fac_f / m_beta; + + // Note that H[i+1, i] equals to the unrestarted beta + m_fac_H(i, i - 1) = restart ? Scalar(0) : m_beta; + + // w <- A * v + m_op.perform_op(v.data(), w.data()); + op_counter++; + + // H[i+1, i+1] = = v'Bw + m_fac_H(i - 1, i) = m_fac_H(i, i - 1); // Due to symmetry + m_fac_H(i, i) = m_op.inner_product(v, w); + + // f <- w - V * V'Bw = w - H[i+1, i] * V{i} - H[i+1, i+1] * V{i+1} + // If restarting, we know that H[i+1, i] = 0 + if(restart) + m_fac_f.noalias() = w - m_fac_H(i, i) * v; + else + m_fac_f.noalias() = w - m_fac_H(i, i - 1) * m_fac_V.col(i - 1) - m_fac_H(i, i) * v; + + m_beta = m_op.norm(m_fac_f); + + // f/||f|| is going to be the next column of V, so we need to test + // whether V'B(f/||f||) ~= 0 + const Index i1 = i + 1; + MapMat Vs(m_fac_V.data(), m_n, i1); // The first (i+1) columns + m_op.trans_product(Vs, m_fac_f, Vf.head(i1)); + Scalar ortho_err = Vf.head(i1).cwiseAbs().maxCoeff(); + // If not, iteratively correct the residual + int count = 0; + while(count < 5 && ortho_err > m_eps * m_beta) + { + // There is an edge case: when beta=||f|| is close to zero, f mostly consists + // of noises of rounding errors, so the test [ortho_err < eps * beta] is very + // likely to fail. In particular, if beta=0, then the test is ensured to fail. + // Hence when this happens, we force f to be zero, and then restart in the + // next iteration. + if(m_beta < beta_thresh) + { + m_fac_f.setZero(); + m_beta = Scalar(0); + break; + } + + // f <- f - V * Vf + m_fac_f.noalias() -= Vs * Vf.head(i1); + // h <- h + Vf + m_fac_H(i - 1, i) += Vf[i - 1]; + m_fac_H(i, i - 1) = m_fac_H(i - 1, i); + m_fac_H(i, i) += Vf[i]; + // beta <- ||f|| + m_beta = m_op.norm(m_fac_f); + + m_op.trans_product(Vs, m_fac_f, Vf.head(i1)); + ortho_err = Vf.head(i1).cwiseAbs().maxCoeff(); + count++; + } + } + + // Indicate that this is a step-m factorization + m_k = to_m; + } + + // Apply H -> Q'HQ, where Q is from a tridiagonal QR decomposition + void compress_H(const TridiagQR& decomp) + { + decomp.matrix_QtHQ(m_fac_H); + m_k--; + } +}; + + +} // namespace Spectra + +#endif // LANCZOS_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/TridiagEigen.h b/src/external/Spectra/include/Spectra/LinAlg/TridiagEigen.h new file mode 100644 index 00000000..b79fe8d1 --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/TridiagEigen.h @@ -0,0 +1,219 @@ +// The code was adapted from Eigen/src/Eigenvaleus/SelfAdjointEigenSolver.h +// +// Copyright (C) 2008-2010 Gael Guennebaud +// Copyright (C) 2010 Jitse Niesen +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef TRIDIAG_EIGEN_H +#define TRIDIAG_EIGEN_H + +#include +#include +#include + +#include "../Util/TypeTraits.h" + +namespace Spectra { + + +template +class TridiagEigen +{ +private: + typedef Eigen::Index Index; + // For convenience in adapting the tridiagonal_qr_step() function + typedef Scalar RealScalar; + + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + typedef Eigen::Ref GenericMatrix; + typedef const Eigen::Ref ConstGenericMatrix; + + Index m_n; + Vector m_main_diag; // Main diagonal elements of the matrix + Vector m_sub_diag; // Sub-diagonal elements of the matrix + Matrix m_evecs; // To store eigenvectors + + bool m_computed; + const Scalar m_near_0; // a very small value, ~= 1e-307 for the "double" type + + // Adapted from Eigen/src/Eigenvaleus/SelfAdjointEigenSolver.h + static void tridiagonal_qr_step(RealScalar* diag, + RealScalar* subdiag, Index start, + Index end, Scalar* matrixQ, + Index n) + { + using std::abs; + + RealScalar td = (diag[end-1] - diag[end]) * RealScalar(0.5); + RealScalar e = subdiag[end-1]; + // Note that thanks to scaling, e^2 or td^2 cannot overflow, however they can still + // underflow thus leading to inf/NaN values when using the following commented code: + // RealScalar e2 = numext::abs2(subdiag[end-1]); + // RealScalar mu = diag[end] - e2 / (td + (td>0 ? 1 : -1) * sqrt(td*td + e2)); + // This explain the following, somewhat more complicated, version: + RealScalar mu = diag[end]; + if(td == Scalar(0)) + mu -= abs(e); + else + { + RealScalar e2 = Eigen::numext::abs2(subdiag[end-1]); + RealScalar h = Eigen::numext::hypot(td, e); + if(e2==RealScalar(0)) mu -= (e / (td + (td>RealScalar(0) ? RealScalar(1) : RealScalar(-1)))) * (e / h); + else mu -= e2 / (td + (td>RealScalar(0) ? h : -h)); + } + + RealScalar x = diag[start] - mu; + RealScalar z = subdiag[start]; + Eigen::Map q(matrixQ, n, n); + for(Index k = start; k < end; ++k) + { + Eigen::JacobiRotation rot; + rot.makeGivens(x, z); + + const RealScalar s = rot.s(); + const RealScalar c = rot.c(); + + // do T = G' T G + RealScalar sdk = s * diag[k] + c * subdiag[k]; + RealScalar dkp1 = s * subdiag[k] + c * diag[k + 1]; + + diag[k] = c * (c * diag[k] - s * subdiag[k]) - s * (c * subdiag[k] - s * diag[k + 1]); + diag[k + 1] = s * sdk + c * dkp1; + subdiag[k] = c * sdk - s * dkp1; + + if(k > start) + subdiag[k - 1] = c * subdiag[k - 1] - s * z; + + x = subdiag[k]; + + if(k < end - 1) + { + z = -s * subdiag[k+1]; + subdiag[k + 1] = c * subdiag[k + 1]; + } + + // apply the givens rotation to the unit matrix Q = Q * G + if(matrixQ) + q.applyOnTheRight(k, k + 1, rot); + } + } + +public: + TridiagEigen() : + m_n(0), m_computed(false), + m_near_0(TypeTraits::min() * Scalar(10)) + {} + + TridiagEigen(ConstGenericMatrix& mat) : + m_n(mat.rows()), m_computed(false), + m_near_0(TypeTraits::min() * Scalar(10)) + { + compute(mat); + } + + void compute(ConstGenericMatrix& mat) + { + using std::abs; + + m_n = mat.rows(); + if(m_n != mat.cols()) + throw std::invalid_argument("TridiagEigen: matrix must be square"); + + m_main_diag.resize(m_n); + m_sub_diag.resize(m_n - 1); + m_evecs.resize(m_n, m_n); + m_evecs.setIdentity(); + + // Scale matrix to improve stability + const Scalar scale = std::max(mat.diagonal().cwiseAbs().maxCoeff(), + mat.diagonal(-1).cwiseAbs().maxCoeff()); + // If scale=0, mat is a zero matrix, so we can early stop + if(scale < m_near_0) + { + // m_main_diag contains eigenvalues + m_main_diag.setZero(); + // m_evecs has been set identity + // m_evecs.setIdentity(); + m_computed = true; + return; + } + m_main_diag.noalias() = mat.diagonal() / scale; + m_sub_diag.noalias() = mat.diagonal(-1) / scale; + + Scalar* diag = m_main_diag.data(); + Scalar* subdiag = m_sub_diag.data(); + + Index end = m_n - 1; + Index start = 0; + Index iter = 0; // total number of iterations + int info = 0; // 0 for success, 1 for failure + + const Scalar considerAsZero = TypeTraits::min(); + const Scalar precision = Scalar(2) * Eigen::NumTraits::epsilon(); + + while(end > 0) + { + for(Index i = start; i < end; i++) + if(abs(subdiag[i]) <= considerAsZero || + abs(subdiag[i]) <= (abs(diag[i]) + abs(diag[i + 1])) * precision) + subdiag[i] = 0; + + // find the largest unreduced block + while(end > 0 && subdiag[end - 1] == Scalar(0)) + end--; + + if(end <= 0) + break; + + // if we spent too many iterations, we give up + iter++; + if(iter > 30 * m_n) + { + info = 1; + break; + } + + start = end - 1; + while(start > 0 && subdiag[start - 1] != Scalar(0)) + start--; + + tridiagonal_qr_step(diag, subdiag, start, end, m_evecs.data(), m_n); + } + + if(info > 0) + throw std::runtime_error("TridiagEigen: eigen decomposition failed"); + + // Scale eigenvalues back + m_main_diag *= scale; + + m_computed = true; + } + + const Vector& eigenvalues() const + { + if(!m_computed) + throw std::logic_error("TridiagEigen: need to call compute() first"); + + // After calling compute(), main_diag will contain the eigenvalues. + return m_main_diag; + } + + const Matrix& eigenvectors() const + { + if(!m_computed) + throw std::logic_error("TridiagEigen: need to call compute() first"); + + return m_evecs; + } +}; + + +} // namespace Spectra + +#endif // TRIDIAG_EIGEN_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergEigen.h b/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergEigen.h new file mode 100644 index 00000000..4e099f56 --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergEigen.h @@ -0,0 +1,317 @@ +// The code was adapted from Eigen/src/Eigenvaleus/EigenSolver.h +// +// Copyright (C) 2008 Gael Guennebaud +// Copyright (C) 2010,2012 Jitse Niesen +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef UPPER_HESSENBERG_EIGEN_H +#define UPPER_HESSENBERG_EIGEN_H + +#include +#include +#include + +namespace Spectra { + + +template +class UpperHessenbergEigen +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + typedef Eigen::Ref GenericMatrix; + typedef const Eigen::Ref ConstGenericMatrix; + + typedef std::complex Complex; + typedef Eigen::Matrix ComplexMatrix; + typedef Eigen::Matrix ComplexVector; + + Index m_n; // Size of the matrix + Eigen::RealSchur m_realSchur; // Schur decomposition solver + Matrix m_matT; // Schur T matrix + Matrix m_eivec; // Storing eigenvectors + ComplexVector m_eivalues; // Eigenvalues + + bool m_computed; + + void doComputeEigenvectors() + { + using std::abs; + + const Index size = m_eivec.cols(); + const Scalar eps = Eigen::NumTraits::epsilon(); + + // inefficient! this is already computed in RealSchur + Scalar norm(0); + for(Index j = 0; j < size; ++j) + { + norm += m_matT.row(j).segment((std::max)(j-1, Index(0)), size-(std::max)(j-1, Index(0))).cwiseAbs().sum(); + } + + // Backsubstitute to find vectors of upper triangular form + if(norm == Scalar(0)) + return; + + for(Index n = size - 1; n >= 0; n--) + { + Scalar p = m_eivalues.coeff(n).real(); + Scalar q = m_eivalues.coeff(n).imag(); + + // Scalar vector + if(q == Scalar(0)) + { + Scalar lastr(0), lastw(0); + Index l = n; + + m_matT.coeffRef(n,n) = Scalar(1); + for(Index i = n-1; i >= 0; i--) + { + Scalar w = m_matT.coeff(i,i) - p; + Scalar r = m_matT.row(i).segment(l,n-l+1).dot(m_matT.col(n).segment(l, n-l+1)); + + if(m_eivalues.coeff(i).imag() < Scalar(0)) + { + lastw = w; + lastr = r; + } else { + l = i; + if(m_eivalues.coeff(i).imag() == Scalar(0)) + { + if (w != Scalar(0)) + m_matT.coeffRef(i,n) = -r / w; + else + m_matT.coeffRef(i,n) = -r / (eps * norm); + } + else // Solve real equations + { + Scalar x = m_matT.coeff(i,i+1); + Scalar y = m_matT.coeff(i+1,i); + Scalar denom = (m_eivalues.coeff(i).real() - p) * (m_eivalues.coeff(i).real() - p) + m_eivalues.coeff(i).imag() * m_eivalues.coeff(i).imag(); + Scalar t = (x * lastr - lastw * r) / denom; + m_matT.coeffRef(i,n) = t; + if(abs(x) > abs(lastw)) + m_matT.coeffRef(i+1,n) = (-r - w * t) / x; + else + m_matT.coeffRef(i+1,n) = (-lastr - y * t) / lastw; + } + + // Overflow control + Scalar t = abs(m_matT.coeff(i,n)); + if((eps * t) * t > Scalar(1)) + m_matT.col(n).tail(size-i) /= t; + } + } + } else if(q < Scalar(0) && n > 0) { // Complex vector + Scalar lastra(0), lastsa(0), lastw(0); + Index l = n-1; + + // Last vector component imaginary so matrix is triangular + if(abs(m_matT.coeff(n,n-1)) > abs(m_matT.coeff(n-1,n))) + { + m_matT.coeffRef(n-1,n-1) = q / m_matT.coeff(n,n-1); + m_matT.coeffRef(n-1,n) = -(m_matT.coeff(n,n) - p) / m_matT.coeff(n,n-1); + } + else + { + Complex cc = Complex(Scalar(0),-m_matT.coeff(n-1,n)) / Complex(m_matT.coeff(n-1,n-1)-p,q); + m_matT.coeffRef(n-1,n-1) = Eigen::numext::real(cc); + m_matT.coeffRef(n-1,n) = Eigen::numext::imag(cc); + } + m_matT.coeffRef(n,n-1) = Scalar(0); + m_matT.coeffRef(n,n) = Scalar(1); + for(Index i = n-2; i >= 0; i--) + { + Scalar ra = m_matT.row(i).segment(l, n-l+1).dot(m_matT.col(n-1).segment(l, n-l+1)); + Scalar sa = m_matT.row(i).segment(l, n-l+1).dot(m_matT.col(n).segment(l, n-l+1)); + Scalar w = m_matT.coeff(i,i) - p; + + if(m_eivalues.coeff(i).imag() < Scalar(0)) + { + lastw = w; + lastra = ra; + lastsa = sa; + } + else + { + l = i; + if(m_eivalues.coeff(i).imag() == Scalar(0)) + { + Complex cc = Complex(-ra,-sa) / Complex(w,q); + m_matT.coeffRef(i,n-1) = Eigen::numext::real(cc); + m_matT.coeffRef(i,n) = Eigen::numext::imag(cc); + } + else + { + // Solve complex equations + Scalar x = m_matT.coeff(i,i+1); + Scalar y = m_matT.coeff(i+1,i); + Scalar vr = (m_eivalues.coeff(i).real() - p) * (m_eivalues.coeff(i).real() - p) + m_eivalues.coeff(i).imag() * m_eivalues.coeff(i).imag() - q * q; + Scalar vi = (m_eivalues.coeff(i).real() - p) * Scalar(2) * q; + if((vr == Scalar(0)) && (vi == Scalar(0))) + vr = eps * norm * (abs(w) + abs(q) + abs(x) + abs(y) + abs(lastw)); + + Complex cc = Complex(x*lastra-lastw*ra+q*sa,x*lastsa-lastw*sa-q*ra) / Complex(vr,vi); + m_matT.coeffRef(i,n-1) = Eigen::numext::real(cc); + m_matT.coeffRef(i,n) = Eigen::numext::imag(cc); + if(abs(x) > (abs(lastw) + abs(q))) + { + m_matT.coeffRef(i+1,n-1) = (-ra - w * m_matT.coeff(i,n-1) + q * m_matT.coeff(i,n)) / x; + m_matT.coeffRef(i+1,n) = (-sa - w * m_matT.coeff(i,n) - q * m_matT.coeff(i,n-1)) / x; + } + else + { + cc = Complex(-lastra-y*m_matT.coeff(i,n-1),-lastsa-y*m_matT.coeff(i,n)) / Complex(lastw,q); + m_matT.coeffRef(i+1,n-1) = Eigen::numext::real(cc); + m_matT.coeffRef(i+1,n) = Eigen::numext::imag(cc); + } + } + + // Overflow control + Scalar t = std::max(abs(m_matT.coeff(i,n-1)), abs(m_matT.coeff(i,n))); + if((eps * t) * t > Scalar(1)) + m_matT.block(i, n-1, size-i, 2) /= t; + + } + } + + // We handled a pair of complex conjugate eigenvalues, so need to skip them both + n--; + } + } + + // Back transformation to get eigenvectors of original matrix + Vector m_tmp(size); + for(Index j = size-1; j >= 0; j--) + { + m_tmp.noalias() = m_eivec.leftCols(j+1) * m_matT.col(j).segment(0, j+1); + m_eivec.col(j) = m_tmp; + } + } + +public: + + UpperHessenbergEigen() : + m_n(0), m_computed(false) + {} + + UpperHessenbergEigen(ConstGenericMatrix& mat) : + m_n(mat.rows()), m_computed(false) + { + compute(mat); + } + + void compute(ConstGenericMatrix& mat) + { + using std::abs; + using std::sqrt; + + if(mat.rows() != mat.cols()) + throw std::invalid_argument("UpperHessenbergEigen: matrix must be square"); + + m_n = mat.rows(); + // Scale matrix prior to the Schur decomposition + const Scalar scale = mat.cwiseAbs().maxCoeff(); + + // Reduce to real Schur form + Matrix Q = Matrix::Identity(m_n, m_n); + m_realSchur.computeFromHessenberg(mat / scale, Q, true); + if(m_realSchur.info() != Eigen::Success) + throw std::runtime_error("UpperHessenbergEigen: eigen decomposition failed"); + + m_matT = m_realSchur.matrixT(); + m_eivec = m_realSchur.matrixU(); + + // Compute eigenvalues from matT + m_eivalues.resize(m_n); + Index i = 0; + while(i < m_n) + { + // Real eigenvalue + if(i == m_n - 1 || m_matT.coeff(i+1, i) == Scalar(0)) + { + m_eivalues.coeffRef(i) = m_matT.coeff(i, i); + ++i; + } + else // Complex eigenvalues + { + Scalar p = Scalar(0.5) * (m_matT.coeff(i, i) - m_matT.coeff(i+1, i+1)); + Scalar z; + // Compute z = sqrt(abs(p * p + m_matT.coeff(i+1, i) * m_matT.coeff(i, i+1))); + // without overflow + { + Scalar t0 = m_matT.coeff(i+1, i); + Scalar t1 = m_matT.coeff(i, i+1); + Scalar maxval = std::max(abs(p), std::max(abs(t0), abs(t1))); + t0 /= maxval; + t1 /= maxval; + Scalar p0 = p / maxval; + z = maxval * sqrt(abs(p0 * p0 + t0 * t1)); + } + m_eivalues.coeffRef(i) = Complex(m_matT.coeff(i+1, i+1) + p, z); + m_eivalues.coeffRef(i+1) = Complex(m_matT.coeff(i+1, i+1) + p, -z); + i += 2; + } + } + + // Compute eigenvectors + doComputeEigenvectors(); + + // Scale eigenvalues back + m_eivalues *= scale; + + m_computed = true; + } + + const ComplexVector& eigenvalues() const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergEigen: need to call compute() first"); + + return m_eivalues; + } + + ComplexMatrix eigenvectors() + { + using std::abs; + + if(!m_computed) + throw std::logic_error("UpperHessenbergEigen: need to call compute() first"); + + Index n = m_eivec.cols(); + ComplexMatrix matV(n, n); + for(Index j = 0; j < n; ++j) + { + // imaginary part of real eigenvalue is already set to exact zero + if(Eigen::numext::imag(m_eivalues.coeff(j)) == Scalar(0) || j + 1 == n) + { + // we have a real eigen value + matV.col(j) = m_eivec.col(j).template cast(); + matV.col(j).normalize(); + } else { + // we have a pair of complex eigen values + for(Index i = 0; i < n; ++i) + { + matV.coeffRef(i,j) = Complex(m_eivec.coeff(i,j), m_eivec.coeff(i,j+1)); + matV.coeffRef(i,j+1) = Complex(m_eivec.coeff(i,j), -m_eivec.coeff(i,j+1)); + } + matV.col(j).normalize(); + matV.col(j+1).normalize(); + ++j; + } + } + + return matV; + } +}; + + +} // namespace Spectra + +#endif // UPPER_HESSENBERG_EIGEN_H diff --git a/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergQR.h b/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergQR.h new file mode 100644 index 00000000..a66d9598 --- /dev/null +++ b/src/external/Spectra/include/Spectra/LinAlg/UpperHessenbergQR.h @@ -0,0 +1,670 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef UPPER_HESSENBERG_QR_H +#define UPPER_HESSENBERG_QR_H + +#include +#include // std::sqrt +#include // std::fill, std::copy +#include // std::logic_error + +namespace Spectra { + + +/// +/// \defgroup Internals Internal Classes +/// +/// Classes for internal use. May be useful to developers. +/// + +/// +/// \ingroup Internals +/// @{ +/// + +/// +/// \defgroup LinearAlgebra Linear Algebra +/// +/// A number of classes for linear algebra operations. +/// + +/// +/// \ingroup LinearAlgebra +/// +/// Perform the QR decomposition of an upper Hessenberg matrix. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// +template +class UpperHessenbergQR +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Matrix RowVector; + typedef Eigen::Array Array; + + typedef Eigen::Ref GenericMatrix; + typedef const Eigen::Ref ConstGenericMatrix; + + Matrix m_mat_T; + +protected: + Index m_n; + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + // Q = G1 * G2 * ... * G_{n-1} + Scalar m_shift; + Array m_rot_cos; + Array m_rot_sin; + bool m_computed; + + // Given x and y, compute 1) r = sqrt(x^2 + y^2), 2) c = x / r, 3) s = -y / r + // If both x and y are zero, set c = 1 and s = 0 + // We must implement it in a numerically stable way + static void compute_rotation(const Scalar& x, const Scalar& y, Scalar& r, Scalar& c, Scalar& s) + { + using std::sqrt; + + const Scalar xsign = (x > Scalar(0)) - (x < Scalar(0)); + const Scalar ysign = (y > Scalar(0)) - (y < Scalar(0)); + const Scalar xabs = x * xsign; + const Scalar yabs = y * ysign; + if(xabs > yabs) + { + // In this case xabs != 0 + const Scalar ratio = yabs / xabs; // so that 0 <= ratio < 1 + const Scalar common = sqrt(Scalar(1) + ratio * ratio); + c = xsign / common; + r = xabs * common; + s = -y / r; + } else { + if(yabs == Scalar(0)) + { + r = Scalar(0); c = Scalar(1); s = Scalar(0); + return; + } + const Scalar ratio = xabs / yabs; // so that 0 <= ratio <= 1 + const Scalar common = sqrt(Scalar(1) + ratio * ratio); + s = -ysign / common; + r = yabs * common; + c = x / r; + } + } + +public: + /// + /// Constructor to preallocate memory. Computation can + /// be performed later by calling the compute() method. + /// + UpperHessenbergQR(Index size) : + m_n(size), + m_rot_cos(m_n - 1), + m_rot_sin(m_n - 1), + m_computed(false) + {} + + /// + /// Constructor to create an object that performs and stores the + /// QR decomposition of an upper Hessenberg matrix `mat`, with an + /// optional shift: \f$H-sI=QR\f$. Here \f$H\f$ stands for the matrix + /// `mat`, and \f$s\f$ is the shift. + /// + /// \param mat Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// Only the upper triangular and the lower subdiagonal parts of + /// the matrix are used. + /// + UpperHessenbergQR(ConstGenericMatrix& mat, const Scalar& shift = Scalar(0)) : + m_n(mat.rows()), + m_shift(shift), + m_rot_cos(m_n - 1), + m_rot_sin(m_n - 1), + m_computed(false) + { + compute(mat, shift); + } + + /// + /// Virtual destructor. + /// + virtual ~UpperHessenbergQR() {}; + + /// + /// Conduct the QR factorization of an upper Hessenberg matrix with + /// an optional shift. + /// + /// \param mat Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// Only the upper triangular and the lower subdiagonal parts of + /// the matrix are used. + /// + virtual void compute(ConstGenericMatrix& mat, const Scalar& shift = Scalar(0)) + { + m_n = mat.rows(); + if(m_n != mat.cols()) + throw std::invalid_argument("UpperHessenbergQR: matrix must be square"); + + m_shift = shift; + m_mat_T.resize(m_n, m_n); + m_rot_cos.resize(m_n - 1); + m_rot_sin.resize(m_n - 1); + + // Make a copy of mat - s * I + std::copy(mat.data(), mat.data() + mat.size(), m_mat_T.data()); + m_mat_T.diagonal().array() -= m_shift; + + Scalar xi, xj, r, c, s; + Scalar *Tii, *ptr; + const Index n1 = m_n - 1; + for(Index i = 0; i < n1; i++) + { + Tii = &m_mat_T.coeffRef(i, i); + + // Make sure mat_T is upper Hessenberg + // Zero the elements below mat_T(i + 1, i) + std::fill(Tii + 2, Tii + m_n - i, Scalar(0)); + + xi = Tii[0]; // mat_T(i, i) + xj = Tii[1]; // mat_T(i + 1, i) + compute_rotation(xi, xj, r, c, s); + m_rot_cos[i] = c; + m_rot_sin[i] = s; + + // For a complete QR decomposition, + // we first obtain the rotation matrix + // G = [ cos sin] + // [-sin cos] + // and then do T[i:(i + 1), i:(n - 1)] = G' * T[i:(i + 1), i:(n - 1)] + + // Gt << c, -s, s, c; + // m_mat_T.block(i, i, 2, m_n - i) = Gt * m_mat_T.block(i, i, 2, m_n - i); + Tii[0] = r; // m_mat_T(i, i) => r + Tii[1] = 0; // m_mat_T(i + 1, i) => 0 + ptr = Tii + m_n; // m_mat_T(i, k), k = i+1, i+2, ..., n-1 + for(Index j = i + 1; j < m_n; j++, ptr += m_n) + { + Scalar tmp = ptr[0]; + ptr[0] = c * tmp - s * ptr[1]; + ptr[1] = s * tmp + c * ptr[1]; + } + + // If we do not need to calculate the R matrix, then + // only the cos and sin sequences are required. + // In such case we only update T[i + 1, (i + 1):(n - 1)] + // m_mat_T.block(i + 1, i + 1, 1, m_n - i - 1) *= c; + // m_mat_T.block(i + 1, i + 1, 1, m_n - i - 1) += s * mat_T.block(i, i + 1, 1, m_n - i - 1); + } + + m_computed = true; + } + + /// + /// Return the \f$R\f$ matrix in the QR decomposition, which is an + /// upper triangular matrix. + /// + /// \return Returned matrix type will be `Eigen::Matrix`, depending on + /// the template parameter `Scalar` defined. + /// + virtual Matrix matrix_R() const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + return m_mat_T; + } + + /// + /// Overwrite `dest` with \f$Q'HQ = RQ + sI\f$, where \f$H\f$ is the input matrix `mat`, + /// and \f$s\f$ is the shift. The result is an upper Hessenberg matrix. + /// + /// \param mat The matrix to be overwritten, whose type should be `Eigen::Matrix`, + /// depending on the template parameter `Scalar` defined. + /// + virtual void matrix_QtHQ(Matrix& dest) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + // Make a copy of the R matrix + dest.resize(m_n, m_n); + std::copy(m_mat_T.data(), m_mat_T.data() + m_mat_T.size(), dest.data()); + + // Compute the RQ matrix + const Index n1 = m_n - 1; + for(Index i = 0; i < n1; i++) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // RQ[, i:(i + 1)] = RQ[, i:(i + 1)] * Gi + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + Scalar *Yi, *Yi1; + Yi = &dest.coeffRef(0, i); + Yi1 = Yi + m_n; // RQ(0, i + 1) + const Index i2 = i + 2; + for(Index j = 0; j < i2; j++) + { + const Scalar tmp = Yi[j]; + Yi[j] = c * tmp - s * Yi1[j]; + Yi1[j] = s * tmp + c * Yi1[j]; + } + + /* Vector dest = RQ.block(0, i, i + 2, 1); + dest.block(0, i, i + 2, 1) = c * Yi - s * dest.block(0, i + 1, i + 2, 1); + dest.block(0, i + 1, i + 2, 1) = s * Yi + c * dest.block(0, i + 1, i + 2, 1); */ + } + + // Add the shift to the diagonal + dest.diagonal().array() += m_shift; + } + + /// + /// Apply the \f$Q\f$ matrix to a vector \f$y\f$. + /// + /// \param Y A vector that will be overwritten by the matrix product \f$Qy\f$. + /// + /// Vector type can be `Eigen::Vector`, depending on + /// the template parameter `Scalar` defined. + /// + // Y -> QY = G1 * G2 * ... * Y + void apply_QY(Vector& Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + for(Index i = m_n - 2; i >= 0; i--) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[i:(i + 1)] = Gi * Y[i:(i + 1)] + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + const Scalar tmp = Y[i]; + Y[i] = c * tmp + s * Y[i + 1]; + Y[i + 1] = -s * tmp + c * Y[i + 1]; + } + } + + /// + /// Apply the \f$Q\f$ matrix to a vector \f$y\f$. + /// + /// \param Y A vector that will be overwritten by the matrix product \f$Q'y\f$. + /// + /// Vector type can be `Eigen::Vector`, depending on + /// the template parameter `Scalar` defined. + /// + // Y -> Q'Y = G_{n-1}' * ... * G2' * G1' * Y + void apply_QtY(Vector& Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + const Index n1 = m_n - 1; + for(Index i = 0; i < n1; i++) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[i:(i + 1)] = Gi' * Y[i:(i + 1)] + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + const Scalar tmp = Y[i]; + Y[i] = c * tmp - s * Y[i + 1]; + Y[i + 1] = s * tmp + c * Y[i + 1]; + } + } + + /// + /// Apply the \f$Q\f$ matrix to another matrix \f$Y\f$. + /// + /// \param Y A matrix that will be overwritten by the matrix product \f$QY\f$. + /// + /// Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + // Y -> QY = G1 * G2 * ... * Y + void apply_QY(GenericMatrix Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + RowVector Yi(Y.cols()), Yi1(Y.cols()); + for(Index i = m_n - 2; i >= 0; i--) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[i:(i + 1), ] = Gi * Y[i:(i + 1), ] + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + Yi.noalias() = Y.row(i); + Yi1.noalias() = Y.row(i + 1); + Y.row(i) = c * Yi + s * Yi1; + Y.row(i + 1) = -s * Yi + c * Yi1; + } + } + + /// + /// Apply the \f$Q\f$ matrix to another matrix \f$Y\f$. + /// + /// \param Y A matrix that will be overwritten by the matrix product \f$Q'Y\f$. + /// + /// Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + // Y -> Q'Y = G_{n-1}' * ... * G2' * G1' * Y + void apply_QtY(GenericMatrix Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + RowVector Yi(Y.cols()), Yi1(Y.cols()); + const Index n1 = m_n - 1; + for(Index i = 0; i < n1; i++) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[i:(i + 1), ] = Gi' * Y[i:(i + 1), ] + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + Yi.noalias() = Y.row(i); + Yi1.noalias() = Y.row(i + 1); + Y.row(i) = c * Yi - s * Yi1; + Y.row(i + 1) = s * Yi + c * Yi1; + } + } + + /// + /// Apply the \f$Q\f$ matrix to another matrix \f$Y\f$. + /// + /// \param Y A matrix that will be overwritten by the matrix product \f$YQ\f$. + /// + /// Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + // Y -> YQ = Y * G1 * G2 * ... + void apply_YQ(GenericMatrix Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + /*Vector Yi(Y.rows()); + for(Index i = 0; i < m_n - 1; i++) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[, i:(i + 1)] = Y[, i:(i + 1)] * Gi + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + Yi.noalias() = Y.col(i); + Y.col(i) = c * Yi - s * Y.col(i + 1); + Y.col(i + 1) = s * Yi + c * Y.col(i + 1); + }*/ + Scalar *Y_col_i, *Y_col_i1; + const Index n1 = m_n - 1; + const Index nrow = Y.rows(); + for(Index i = 0; i < n1; i++) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + + Y_col_i = &Y.coeffRef(0, i); + Y_col_i1 = &Y.coeffRef(0, i + 1); + for(Index j = 0; j < nrow; j++) + { + Scalar tmp = Y_col_i[j]; + Y_col_i[j] = c * tmp - s * Y_col_i1[j]; + Y_col_i1[j] = s * tmp + c * Y_col_i1[j]; + } + } + } + + /// + /// Apply the \f$Q\f$ matrix to another matrix \f$Y\f$. + /// + /// \param Y A matrix that will be overwritten by the matrix product \f$YQ'\f$. + /// + /// Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + // Y -> YQ' = Y * G_{n-1}' * ... * G2' * G1' + void apply_YQt(GenericMatrix Y) const + { + if(!m_computed) + throw std::logic_error("UpperHessenbergQR: need to call compute() first"); + + Vector Yi(Y.rows()); + for(Index i = m_n - 2; i >= 0; i--) + { + const Scalar c = m_rot_cos.coeff(i); + const Scalar s = m_rot_sin.coeff(i); + // Y[, i:(i + 1)] = Y[, i:(i + 1)] * Gi' + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + Yi.noalias() = Y.col(i); + Y.col(i) = c * Yi + s * Y.col(i + 1); + Y.col(i + 1) = -s * Yi + c * Y.col(i + 1); + } + } +}; + + + +/// +/// \ingroup LinearAlgebra +/// +/// Perform the QR decomposition of a tridiagonal matrix, a special +/// case of upper Hessenberg matrices. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// +template +class TridiagQR: public UpperHessenbergQR +{ +private: + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef const Eigen::Ref ConstGenericMatrix; + + typedef typename Matrix::Index Index; + + Vector m_T_diag; // diagonal elements of T + Vector m_T_lsub; // lower subdiagonal of T + Vector m_T_usub; // upper subdiagonal of T + Vector m_T_usub2; // 2nd upper subdiagonal of T + +public: + /// + /// Constructor to preallocate memory. Computation can + /// be performed later by calling the compute() method. + /// + TridiagQR(Index size) : + UpperHessenbergQR(size) + {} + + /// + /// Constructor to create an object that performs and stores the + /// QR decomposition of an upper Hessenberg matrix `mat`, with an + /// optional shift: \f$H-sI=QR\f$. Here \f$H\f$ stands for the matrix + /// `mat`, and \f$s\f$ is the shift. + /// + /// \param mat Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// Only the major- and sub- diagonal parts of + /// the matrix are used. + /// + TridiagQR(ConstGenericMatrix& mat, const Scalar& shift = Scalar(0)) : + UpperHessenbergQR(mat.rows()) + { + this->compute(mat, shift); + } + + /// + /// Conduct the QR factorization of a tridiagonal matrix with an + /// optional shift. + /// + /// \param mat Matrix type can be `Eigen::Matrix` (e.g. + /// `Eigen::MatrixXd` and `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// Only the major- and sub- diagonal parts of + /// the matrix are used. + /// + void compute(ConstGenericMatrix& mat, const Scalar& shift = Scalar(0)) + { + this->m_n = mat.rows(); + if(this->m_n != mat.cols()) + throw std::invalid_argument("TridiagQR: matrix must be square"); + + this->m_shift = shift; + m_T_diag.resize(this->m_n); + m_T_lsub.resize(this->m_n - 1); + m_T_usub.resize(this->m_n - 1); + m_T_usub2.resize(this->m_n - 2); + this->m_rot_cos.resize(this->m_n - 1); + this->m_rot_sin.resize(this->m_n - 1); + + m_T_diag.array() = mat.diagonal().array() - this->m_shift; + m_T_lsub.noalias() = mat.diagonal(-1); + m_T_usub.noalias() = m_T_lsub; + + // A number of pointers to avoid repeated address calculation + Scalar *c = this->m_rot_cos.data(), // pointer to the cosine vector + *s = this->m_rot_sin.data(), // pointer to the sine vector + r; + const Index n1 = this->m_n - 1; + for(Index i = 0; i < n1; i++) + { + // diag[i] == T[i, i] + // lsub[i] == T[i + 1, i] + // r = sqrt(T[i, i]^2 + T[i + 1, i]^2) + // c = T[i, i] / r, s = -T[i + 1, i] / r + this->compute_rotation(m_T_diag.coeff(i), m_T_lsub.coeff(i), r, *c, *s); + + // For a complete QR decomposition, + // we first obtain the rotation matrix + // G = [ cos sin] + // [-sin cos] + // and then do T[i:(i + 1), i:(i + 2)] = G' * T[i:(i + 1), i:(i + 2)] + + // Update T[i, i] and T[i + 1, i] + // The updated value of T[i, i] is known to be r + // The updated value of T[i + 1, i] is known to be 0 + m_T_diag.coeffRef(i) = r; + m_T_lsub.coeffRef(i) = Scalar(0); + // Update T[i, i + 1] and T[i + 1, i + 1] + // usub[i] == T[i, i + 1] + // diag[i + 1] == T[i + 1, i + 1] + const Scalar tmp = m_T_usub.coeff(i); + m_T_usub.coeffRef(i) = (*c) * tmp - (*s) * m_T_diag.coeff(i + 1); + m_T_diag.coeffRef(i + 1) = (*s) * tmp + (*c) * m_T_diag.coeff(i + 1); + // Update T[i, i + 2] and T[i + 1, i + 2] + // usub2[i] == T[i, i + 2] + // usub[i + 1] == T[i + 1, i + 2] + if(i < n1 - 1) + { + m_T_usub2.coeffRef(i) = -(*s) * m_T_usub.coeff(i + 1); + m_T_usub.coeffRef(i + 1) *= (*c); + } + + c++; + s++; + + // If we do not need to calculate the R matrix, then + // only the cos and sin sequences are required. + // In such case we only update T[i + 1, (i + 1):(i + 2)] + // T[i + 1, i + 1] = c * T[i + 1, i + 1] + s * T[i, i + 1]; + // T[i + 1, i + 2] *= c; + } + + this->m_computed = true; + } + + /// + /// Return the \f$R\f$ matrix in the QR decomposition, which is an + /// upper triangular matrix. + /// + /// \return Returned matrix type will be `Eigen::Matrix`, depending on + /// the template parameter `Scalar` defined. + /// + Matrix matrix_R() const + { + if(!this->m_computed) + throw std::logic_error("TridiagQR: need to call compute() first"); + + Matrix R = Matrix::Zero(this->m_n, this->m_n); + R.diagonal().noalias() = m_T_diag; + R.diagonal(1).noalias() = m_T_usub; + R.diagonal(2).noalias() = m_T_usub2; + + return R; + } + + /// + /// Overwrite `dest` with \f$Q'HQ = RQ + sI\f$, where \f$H\f$ is the input matrix `mat`, + /// and \f$s\f$ is the shift. The result is a tridiagonal matrix. + /// + /// \param mat The matrix to be overwritten, whose type should be `Eigen::Matrix`, + /// depending on the template parameter `Scalar` defined. + /// + void matrix_QtHQ(Matrix& dest) const + { + if(!this->m_computed) + throw std::logic_error("TridiagQR: need to call compute() first"); + + // Make a copy of the R matrix + dest.resize(this->m_n, this->m_n); + dest.setZero(); + dest.diagonal().noalias() = m_T_diag; + // The upper diagonal refers to m_T_usub + // The 2nd upper subdiagonal will be zero in RQ + + // Compute the RQ matrix + // [m11 m12] points to RQ[i:(i+1), i:(i+1)] + // [0 m22] + // + // Gi = [ cos[i] sin[i]] + // [-sin[i] cos[i]] + const Index n1 = this->m_n - 1; + for(Index i = 0; i < n1; i++) + { + const Scalar c = this->m_rot_cos.coeff(i); + const Scalar s = this->m_rot_sin.coeff(i); + const Scalar m11 = dest.coeff(i, i), + m12 = m_T_usub.coeff(i), + m22 = m_T_diag.coeff(i + 1); + + // Update the diagonal and the lower subdiagonal of dest + dest.coeffRef(i , i ) = c * m11 - s * m12; + dest.coeffRef(i + 1, i ) = - s * m22; + dest.coeffRef(i + 1, i + 1) = c * m22; + } + + // Copy the lower subdiagonal to upper subdiagonal + dest.diagonal(1).noalias() = dest.diagonal(-1); + + // Add the shift to the diagonal + dest.diagonal().array() += this->m_shift; + } +}; + +/// +/// @} +/// + + +} // namespace Spectra + +#endif // UPPER_HESSENBERG_QR_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseCholesky.h b/src/external/Spectra/include/Spectra/MatOp/DenseCholesky.h new file mode 100644 index 00000000..354c9508 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseCholesky.h @@ -0,0 +1,110 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_CHOLESKY_H +#define DENSE_CHOLESKY_H + +#include +#include +#include +#include "../Util/CompInfo.h" + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the operations related to Cholesky decomposition on a +/// positive definite matrix, \f$B=LL'\f$, where \f$L\f$ is a lower triangular +/// matrix. It is mainly used in the SymGEigsSolver generalized eigen solver +/// in the Cholesky decomposition mode. +/// +template +class DenseCholesky +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstMat; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + const Index m_n; + Eigen::LLT m_decomp; + int m_info; // status of the decomposition + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseCholesky(ConstGenericMatrix& mat) : + m_n(mat.rows()), m_info(NOT_COMPUTED) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("DenseCholesky: matrix must be square"); + + m_decomp.compute(mat); + m_info = (m_decomp.info() == Eigen::Success) ? + SUCCESSFUL : + NUMERICAL_ISSUE; + } + + /// + /// Returns the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Returns the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Returns the status of the computation. + /// The full list of enumeration values can be found in \ref Enumerations. + /// + int info() const { return m_info; } + + /// + /// Performs the lower triangular solving operation \f$y=L^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(L) * x_in + void lower_triangular_solve(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_decomp.matrixL().solve(x); + } + + /// + /// Performs the upper triangular solving operation \f$y=(L')^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(L') * x_in + void upper_triangular_solve(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_decomp.matrixU().solve(x); + } +}; + + +} // namespace Spectra + +#endif // DENSE_CHOLESKY_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseGenComplexShiftSolve.h b/src/external/Spectra/include/Spectra/MatOp/DenseGenComplexShiftSolve.h new file mode 100644 index 00000000..7f189067 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseGenComplexShiftSolve.h @@ -0,0 +1,104 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_GEN_COMPLEX_SHIFT_SOLVE_H +#define DENSE_GEN_COMPLEX_SHIFT_SOLVE_H + +#include +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the complex shift-solve operation on a general real matrix \f$A\f$, +/// i.e., calculating \f$y=\mathrm{Re}\{(A-\sigma I)^{-1}x\}\f$ for any complex-valued +/// \f$\sigma\f$ and real-valued vector \f$x\f$. It is mainly used in the +/// GenEigsComplexShiftSolver eigen solver. +/// +template +class DenseGenComplexShiftSolve +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + typedef std::complex Complex; + typedef Eigen::Matrix ComplexMatrix; + typedef Eigen::Matrix ComplexVector; + + typedef Eigen::PartialPivLU ComplexSolver; + + ConstGenericMatrix m_mat; + const Index m_n; + ComplexSolver m_solver; + ComplexVector m_x_cache; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseGenComplexShiftSolve(ConstGenericMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("DenseGenComplexShiftSolve: matrix must be square"); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Set the complex shift \f$\sigma\f$. + /// + /// \param sigmar Real part of \f$\sigma\f$. + /// \param sigmai Imaginary part of \f$\sigma\f$. + /// + void set_shift(Scalar sigmar, Scalar sigmai) + { + m_solver.compute(m_mat.template cast() - Complex(sigmar, sigmai) * ComplexMatrix::Identity(m_n, m_n)); + m_x_cache.resize(m_n); + m_x_cache.setZero(); + } + + /// + /// Perform the complex shift-solve operation + /// \f$y=\mathrm{Re}\{(A-\sigma I)^{-1}x\}\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = Re( inv(A - sigma * I) * x_in ) + void perform_op(const Scalar* x_in, Scalar* y_out) + { + m_x_cache.real() = MapConstVec(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_solver.solve(m_x_cache).real(); + } +}; + + +} // namespace Spectra + +#endif // DENSE_GEN_COMPLEX_SHIFT_SOLVE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseGenMatProd.h b/src/external/Spectra/include/Spectra/MatOp/DenseGenMatProd.h new file mode 100644 index 00000000..6933dac0 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseGenMatProd.h @@ -0,0 +1,82 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_GEN_MAT_PROD_H +#define DENSE_GEN_MAT_PROD_H + +#include + +namespace Spectra { + + +/// +/// \defgroup MatOp Matrix Operations +/// +/// Define matrix operations on existing matrix objects +/// + +/// +/// \ingroup MatOp +/// +/// This class defines the matrix-vector multiplication operation on a +/// general real matrix \f$A\f$, i.e., calculating \f$y=Ax\f$ for any vector +/// \f$x\f$. It is mainly used in the GenEigsSolver and +/// SymEigsSolver eigen solvers. +/// +template +class DenseGenMatProd +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseGenMatProd(ConstGenericMatrix& mat) : + m_mat(mat) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_mat.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_mat.cols(); } + + /// + /// Perform the matrix-vector multiplication operation \f$y=Ax\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_mat.cols()); + MapVec y(y_out, m_mat.rows()); + y.noalias() = m_mat * x; + } +}; + + +} // namespace Spectra + +#endif // DENSE_GEN_MAT_PROD_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseGenRealShiftSolve.h b/src/external/Spectra/include/Spectra/MatOp/DenseGenRealShiftSolve.h new file mode 100644 index 00000000..d7ba8f2a --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseGenRealShiftSolve.h @@ -0,0 +1,90 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_GEN_REAL_SHIFT_SOLVE_H +#define DENSE_GEN_REAL_SHIFT_SOLVE_H + +#include +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the shift-solve operation on a general real matrix \f$A\f$, +/// i.e., calculating \f$y=(A-\sigma I)^{-1}x\f$ for any real \f$\sigma\f$ and +/// vector \f$x\f$. It is mainly used in the GenEigsRealShiftSolver eigen solver. +/// +template +class DenseGenRealShiftSolve +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + const Index m_n; + Eigen::PartialPivLU m_solver; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseGenRealShiftSolve(ConstGenericMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("DenseGenRealShiftSolve: matrix must be square"); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Set the real shift \f$\sigma\f$. + /// + void set_shift(Scalar sigma) + { + m_solver.compute(m_mat - sigma * Matrix::Identity(m_n, m_n)); + } + + /// + /// Perform the shift-solve operation \f$y=(A-\sigma I)^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(A - sigma * I) * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_solver.solve(x); + } +}; + + +} // namespace Spectra + +#endif // DENSE_GEN_REAL_SHIFT_SOLVE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseSymMatProd.h b/src/external/Spectra/include/Spectra/MatOp/DenseSymMatProd.h new file mode 100644 index 00000000..23ca36e4 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseSymMatProd.h @@ -0,0 +1,75 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_SYM_MAT_PROD_H +#define DENSE_SYM_MAT_PROD_H + +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the matrix-vector multiplication operation on a +/// symmetric real matrix \f$A\f$, i.e., calculating \f$y=Ax\f$ for any vector +/// \f$x\f$. It is mainly used in the SymEigsSolver eigen solver. +/// +template +class DenseSymMatProd +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseSymMatProd(ConstGenericMatrix& mat) : + m_mat(mat) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_mat.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_mat.cols(); } + + /// + /// Perform the matrix-vector multiplication operation \f$y=Ax\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_mat.cols()); + MapVec y(y_out, m_mat.rows()); + y.noalias() = m_mat.template selfadjointView() * x; + } +}; + + +} // namespace Spectra + +#endif // DENSE_SYM_MAT_PROD_H diff --git a/src/external/Spectra/include/Spectra/MatOp/DenseSymShiftSolve.h b/src/external/Spectra/include/Spectra/MatOp/DenseSymShiftSolve.h new file mode 100644 index 00000000..2e2c67f6 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/DenseSymShiftSolve.h @@ -0,0 +1,94 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef DENSE_SYM_SHIFT_SOLVE_H +#define DENSE_SYM_SHIFT_SOLVE_H + +#include +#include + +#include "../LinAlg/BKLDLT.h" +#include "../Util/CompInfo.h" + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the shift-solve operation on a real symmetric matrix \f$A\f$, +/// i.e., calculating \f$y=(A-\sigma I)^{-1}x\f$ for any real \f$\sigma\f$ and +/// vector \f$x\f$. It is mainly used in the SymEigsShiftSolver eigen solver. +/// +template +class DenseSymShiftSolve +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + const int m_n; + BKLDLT m_solver; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** matrix object, whose type can be + /// `Eigen::Matrix` (e.g. `Eigen::MatrixXd` and + /// `Eigen::MatrixXf`), or its mapped version + /// (e.g. `Eigen::Map`). + /// + DenseSymShiftSolve(ConstGenericMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("DenseSymShiftSolve: matrix must be square"); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Set the real shift \f$\sigma\f$. + /// + void set_shift(Scalar sigma) + { + m_solver.compute(m_mat, Uplo, sigma); + if(m_solver.info() != SUCCESSFUL) + throw std::invalid_argument("DenseSymShiftSolve: factorization failed with the given shift"); + } + + /// + /// Perform the shift-solve operation \f$y=(A-\sigma I)^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(A - sigma * I) * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_solver.solve(x); + } +}; + + +} // namespace Spectra + +#endif // DENSE_SYM_SHIFT_SOLVE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseCholesky.h b/src/external/Spectra/include/Spectra/MatOp/SparseCholesky.h new file mode 100644 index 00000000..0788596d --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseCholesky.h @@ -0,0 +1,111 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_CHOLESKY_H +#define SPARSE_CHOLESKY_H + +#include +#include +#include +#include +#include "../Util/CompInfo.h" + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the operations related to Cholesky decomposition on a +/// sparse positive definite matrix, \f$B=LL'\f$, where \f$L\f$ is a lower triangular +/// matrix. It is mainly used in the SymGEigsSolver generalized eigen solver +/// in the Cholesky decomposition mode. +/// +template +class SparseCholesky +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + const Index m_n; + Eigen::SimplicialLLT m_decomp; + int m_info; // status of the decomposition + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseCholesky(ConstGenericSparseMatrix& mat) : + m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("SparseCholesky: matrix must be square"); + + m_decomp.compute(mat); + m_info = (m_decomp.info() == Eigen::Success) ? + SUCCESSFUL : + NUMERICAL_ISSUE; + } + + /// + /// Returns the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Returns the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Returns the status of the computation. + /// The full list of enumeration values can be found in \ref Enumerations. + /// + int info() const { return m_info; } + + /// + /// Performs the lower triangular solving operation \f$y=L^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(L) * x_in + void lower_triangular_solve(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_decomp.permutationP() * x; + m_decomp.matrixL().solveInPlace(y); + } + + /// + /// Performs the upper triangular solving operation \f$y=(L')^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(L') * x_in + void upper_triangular_solve(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_decomp.matrixU().solve(x); + y = m_decomp.permutationPinv() * y; + } +}; + + +} // namespace Spectra + +#endif // SPARSE_CHOLESKY_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseGenMatProd.h b/src/external/Spectra/include/Spectra/MatOp/SparseGenMatProd.h new file mode 100644 index 00000000..90881395 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseGenMatProd.h @@ -0,0 +1,76 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_GEN_MAT_PROD_H +#define SPARSE_GEN_MAT_PROD_H + +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the matrix-vector multiplication operation on a +/// sparse real matrix \f$A\f$, i.e., calculating \f$y=Ax\f$ for any vector +/// \f$x\f$. It is mainly used in the GenEigsSolver and SymEigsSolver +/// eigen solvers. +/// +template +class SparseGenMatProd +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + ConstGenericSparseMatrix m_mat; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseGenMatProd(ConstGenericSparseMatrix& mat) : + m_mat(mat) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_mat.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_mat.cols(); } + + /// + /// Perform the matrix-vector multiplication operation \f$y=Ax\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_mat.cols()); + MapVec y(y_out, m_mat.rows()); + y.noalias() = m_mat * x; + } +}; + + +} // namespace Spectra + +#endif // SPARSE_GEN_MAT_PROD_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseGenRealShiftSolve.h b/src/external/Spectra/include/Spectra/MatOp/SparseGenRealShiftSolve.h new file mode 100644 index 00000000..df4ec6cf --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseGenRealShiftSolve.h @@ -0,0 +1,95 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_GEN_REAL_SHIFT_SOLVE_H +#define SPARSE_GEN_REAL_SHIFT_SOLVE_H + +#include +#include +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the shift-solve operation on a sparse real matrix \f$A\f$, +/// i.e., calculating \f$y=(A-\sigma I)^{-1}x\f$ for any real \f$\sigma\f$ and +/// vector \f$x\f$. It is mainly used in the GenEigsRealShiftSolver eigen solver. +/// +template +class SparseGenRealShiftSolve +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + ConstGenericSparseMatrix m_mat; + const int m_n; + Eigen::SparseLU m_solver; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseGenRealShiftSolve(ConstGenericSparseMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("SparseGenRealShiftSolve: matrix must be square"); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Set the real shift \f$\sigma\f$. + /// + void set_shift(Scalar sigma) + { + SparseMatrix I(m_n, m_n); + I.setIdentity(); + + m_solver.compute(m_mat - sigma * I); + if(m_solver.info() != Eigen::Success) + throw std::invalid_argument("SparseGenRealShiftSolve: factorization failed with the given shift"); + } + + /// + /// Perform the shift-solve operation \f$y=(A-\sigma I)^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(A - sigma * I) * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_solver.solve(x); + } +}; + + +} // namespace Spectra + +#endif // SPARSE_GEN_REAL_SHIFT_SOLVE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseRegularInverse.h b/src/external/Spectra/include/Spectra/MatOp/SparseRegularInverse.h new file mode 100644 index 00000000..ec6614a5 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseRegularInverse.h @@ -0,0 +1,102 @@ +// Copyright (C) 2017-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_REGULAR_INVERSE_H +#define SPARSE_REGULAR_INVERSE_H + +#include +#include +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines matrix operations required by the generalized eigen solver +/// in the regular inverse mode. For a sparse and positive definite matrix \f$B\f$, +/// it implements the matrix-vector product \f$y=Bx\f$ and the linear equation +/// solving operation \f$y=B^{-1}x\f$. +/// +/// This class is intended to be used with the SymGEigsSolver generalized eigen solver +/// in the regular inverse mode. +/// +template +class SparseRegularInverse +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + ConstGenericSparseMatrix m_mat; + const int m_n; + Eigen::ConjugateGradient m_cg; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseRegularInverse(ConstGenericSparseMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("SparseRegularInverse: matrix must be square"); + + m_cg.compute(mat); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Perform the solving operation \f$y=B^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(B) * x_in + void solve(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_cg.solve(x); + } + + /// + /// Perform the matrix-vector multiplication operation \f$y=Bx\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = B * x_in + void mat_prod(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_mat.template selfadjointView() * x; + } +}; + + +} // namespace Spectra + +#endif // SPARSE_REGULAR_INVERSE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseSymMatProd.h b/src/external/Spectra/include/Spectra/MatOp/SparseSymMatProd.h new file mode 100644 index 00000000..ef8f96ee --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseSymMatProd.h @@ -0,0 +1,75 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_SYM_MAT_PROD_H +#define SPARSE_SYM_MAT_PROD_H + +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the matrix-vector multiplication operation on a +/// sparse real symmetric matrix \f$A\f$, i.e., calculating \f$y=Ax\f$ for any vector +/// \f$x\f$. It is mainly used in the SymEigsSolver eigen solver. +/// +template +class SparseSymMatProd +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + ConstGenericSparseMatrix m_mat; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseSymMatProd(ConstGenericSparseMatrix& mat) : + m_mat(mat) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_mat.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_mat.cols(); } + + /// + /// Perform the matrix-vector multiplication operation \f$y=Ax\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_mat.cols()); + MapVec y(y_out, m_mat.rows()); + y.noalias() = m_mat.template selfadjointView() * x; + } +}; + + +} // namespace Spectra + +#endif // SPARSE_SYM_MAT_PROD_H diff --git a/src/external/Spectra/include/Spectra/MatOp/SparseSymShiftSolve.h b/src/external/Spectra/include/Spectra/MatOp/SparseSymShiftSolve.h new file mode 100644 index 00000000..1821cc32 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/SparseSymShiftSolve.h @@ -0,0 +1,97 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SPARSE_SYM_SHIFT_SOLVE_H +#define SPARSE_SYM_SHIFT_SOLVE_H + +#include +#include +#include +#include + +namespace Spectra { + + +/// +/// \ingroup MatOp +/// +/// This class defines the shift-solve operation on a sparse real symmetric matrix \f$A\f$, +/// i.e., calculating \f$y=(A-\sigma I)^{-1}x\f$ for any real \f$\sigma\f$ and +/// vector \f$x\f$. It is mainly used in the SymEigsShiftSolver eigen solver. +/// +template +class SparseSymShiftSolve +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef Eigen::SparseMatrix SparseMatrix; + typedef const Eigen::Ref ConstGenericSparseMatrix; + + ConstGenericSparseMatrix m_mat; + const int m_n; + Eigen::SparseLU m_solver; + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param mat An **Eigen** sparse matrix object, whose type can be + /// `Eigen::SparseMatrix` or its mapped version + /// `Eigen::Map >`. + /// + SparseSymShiftSolve(ConstGenericSparseMatrix& mat) : + m_mat(mat), m_n(mat.rows()) + { + if(mat.rows() != mat.cols()) + throw std::invalid_argument("SparseSymShiftSolve: matrix must be square"); + } + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_n; } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_n; } + + /// + /// Set the real shift \f$\sigma\f$. + /// + void set_shift(Scalar sigma) + { + SparseMatrix mat = m_mat.template selfadjointView(); + SparseMatrix identity(m_n, m_n); + identity.setIdentity(); + mat = mat - sigma * identity; + m_solver.isSymmetric(true); + m_solver.compute(mat); + if(m_solver.info() != Eigen::Success) + throw std::invalid_argument("SparseSymShiftSolve: factorization failed with the given shift"); + } + + /// + /// Perform the shift-solve operation \f$y=(A-\sigma I)^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(A - sigma * I) * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) const + { + MapConstVec x(x_in, m_n); + MapVec y(y_out, m_n); + y.noalias() = m_solver.solve(x); + } +}; + + +} // namespace Spectra + +#endif // SPARSE_SYM_SHIFT_SOLVE_H diff --git a/src/external/Spectra/include/Spectra/MatOp/internal/ArnoldiOp.h b/src/external/Spectra/include/Spectra/MatOp/internal/ArnoldiOp.h new file mode 100644 index 00000000..b79704b5 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/internal/ArnoldiOp.h @@ -0,0 +1,155 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef ARNOLDI_OP_H +#define ARNOLDI_OP_H + +#include +#include // std::sqrt + +namespace Spectra { + + +/// +/// \ingroup Internals +/// @{ +/// + +/// +/// \defgroup Operators Operators +/// +/// Different types of operators. +/// + +/// +/// \ingroup Operators +/// +/// Operators used in the Arnoldi factorization. +/// +template +class ArnoldiOp +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + + OpType& m_op; + BOpType& m_Bop; + Vector m_cache; + +public: + ArnoldiOp(OpType* op, BOpType* Bop) : + m_op(*op), m_Bop(*Bop), m_cache(op->rows()) + {} + + inline Index rows() const { return m_op.rows(); } + + // In generalized eigenvalue problem Ax=lambda*Bx, define the inner product to be = x'By. + // For regular eigenvalue problems, it is the usual inner product = x'y + + // Compute = x'By + // x and y are two vectors + template + Scalar inner_product(const Arg1& x, const Arg2& y) + { + m_Bop.mat_prod(y.data(), m_cache.data()); + return x.dot(m_cache); + } + + // Compute res = = X'By + // X is a matrix, y is a vector, res is a vector + template + void trans_product(const Arg1& x, const Arg2& y, Eigen::Ref res) + { + m_Bop.mat_prod(y.data(), m_cache.data()); + res.noalias() = x.transpose() * m_cache; + } + + // B-norm of a vector, ||x||_B = sqrt(x'Bx) + template + Scalar norm(const Arg& x) + { + using std::sqrt; + return sqrt(inner_product(x, x)); + } + + // The "A" operator to generate the Krylov subspace + inline void perform_op(const Scalar* x_in, Scalar* y_out) + { + m_op.perform_op(x_in, y_out); + } +}; + + + +/// +/// \ingroup Operators +/// +/// Placeholder for the B-operator when \f$B = I\f$. +/// +class IdentityBOp {}; + + + +/// +/// \ingroup Operators +/// +/// Partial specialization for the case \f$B = I\f$. +/// +template +class ArnoldiOp +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + + OpType& m_op; + +public: + ArnoldiOp(OpType* op, IdentityBOp* /*Bop*/) : + m_op(*op) + {} + + inline Index rows() const { return m_op.rows(); } + + // Compute = x'y + // x and y are two vectors + template + Scalar inner_product(const Arg1& x, const Arg2& y) const + { + return x.dot(y); + } + + // Compute res = = X'y + // X is a matrix, y is a vector, res is a vector + template + void trans_product(const Arg1& x, const Arg2& y, Eigen::Ref res) const + { + res.noalias() = x.transpose() * y; + } + + // B-norm of a vector. For regular eigenvalue problems it is simply the L2 norm + template + Scalar norm(const Arg& x) + { + return x.norm(); + } + + // The "A" operator to generate the Krylov subspace + inline void perform_op(Scalar* x_in, Scalar* y_out) + { + m_op.perform_op(x_in, y_out); + } +}; + +/// +/// @} +/// + + +} // namespace Spectra + +#endif // ARNOLDI_OP_H diff --git a/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsCholeskyOp.h b/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsCholeskyOp.h new file mode 100644 index 00000000..d7acdb2a --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsCholeskyOp.h @@ -0,0 +1,77 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_GEIGS_CHOLESKY_OP_H +#define SYM_GEIGS_CHOLESKY_OP_H + +#include +#include "../DenseSymMatProd.h" +#include "../DenseCholesky.h" + +namespace Spectra { + + +/// +/// \ingroup Operators +/// +/// This class defines the matrix operation for generalized eigen solver in the +/// Cholesky decomposition mode. It calculates \f$y=L^{-1}A(L')^{-1}x\f$ for any +/// vector \f$x\f$, where \f$L\f$ is the Cholesky decomposition of \f$B\f$. +/// This class is intended for internal use. +/// +template < typename Scalar = double, + typename OpType = DenseSymMatProd, + typename BOpType = DenseCholesky > +class SymGEigsCholeskyOp +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + OpType& m_op; + BOpType& m_Bop; + Vector m_cache; // temporary working space + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param op Pointer to the \f$A\f$ matrix operation object. + /// \param Bop Pointer to the \f$B\f$ matrix operation object. + /// + SymGEigsCholeskyOp(OpType& op, BOpType& Bop) : + m_op(op), m_Bop(Bop), m_cache(op.rows()) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_Bop.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_Bop.rows(); } + + /// + /// Perform the matrix operation \f$y=L^{-1}A(L')^{-1}x\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(L) * A * inv(L') * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) + { + m_Bop.upper_triangular_solve(x_in, y_out); + m_op.perform_op(y_out, m_cache.data()); + m_Bop.lower_triangular_solve(m_cache.data(), y_out); + } +}; + + +} // namespace Spectra + +#endif // SYM_GEIGS_CHOLESKY_OP_H diff --git a/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsRegInvOp.h b/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsRegInvOp.h new file mode 100644 index 00000000..ac00dcb2 --- /dev/null +++ b/src/external/Spectra/include/Spectra/MatOp/internal/SymGEigsRegInvOp.h @@ -0,0 +1,74 @@ +// Copyright (C) 2017-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_GEIGS_REG_INV_OP_H +#define SYM_GEIGS_REG_INV_OP_H + +#include +#include "../SparseSymMatProd.h" +#include "../SparseRegularInverse.h" + +namespace Spectra { + + +/// +/// \ingroup Operators +/// +/// This class defines the matrix operation for generalized eigen solver in the +/// regular inverse mode. This class is intended for internal use. +/// +template < typename Scalar = double, + typename OpType = SparseSymMatProd, + typename BOpType = SparseRegularInverse > +class SymGEigsRegInvOp +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + OpType& m_op; + BOpType& m_Bop; + Vector m_cache; // temporary working space + +public: + /// + /// Constructor to create the matrix operation object. + /// + /// \param op Pointer to the \f$A\f$ matrix operation object. + /// \param Bop Pointer to the \f$B\f$ matrix operation object. + /// + SymGEigsRegInvOp(OpType& op, BOpType& Bop) : + m_op(op), m_Bop(Bop), m_cache(op.rows()) + {} + + /// + /// Return the number of rows of the underlying matrix. + /// + Index rows() const { return m_Bop.rows(); } + /// + /// Return the number of columns of the underlying matrix. + /// + Index cols() const { return m_Bop.rows(); } + + /// + /// Perform the matrix operation \f$y=B^{-1}Ax\f$. + /// + /// \param x_in Pointer to the \f$x\f$ vector. + /// \param y_out Pointer to the \f$y\f$ vector. + /// + // y_out = inv(B) * A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) + { + m_op.perform_op(x_in, m_cache.data()); + m_Bop.solve(m_cache.data(), y_out); + } +}; + + +} // namespace Spectra + +#endif // SYM_GEIGS_REG_INV_OP_H diff --git a/src/external/Spectra/include/Spectra/SymEigsBase.h b/src/external/Spectra/include/Spectra/SymEigsBase.h new file mode 100644 index 00000000..24d5b505 --- /dev/null +++ b/src/external/Spectra/include/Spectra/SymEigsBase.h @@ -0,0 +1,447 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_EIGS_BASE_H +#define SYM_EIGS_BASE_H + +#include +#include // std::vector +#include // std::abs, std::pow, std::sqrt +#include // std::min, std::copy +#include // std::invalid_argument + +#include "Util/TypeTraits.h" +#include "Util/SelectionRule.h" +#include "Util/CompInfo.h" +#include "Util/SimpleRandom.h" +#include "MatOp/internal/ArnoldiOp.h" +#include "LinAlg/UpperHessenbergQR.h" +#include "LinAlg/TridiagEigen.h" +#include "LinAlg/Lanczos.h" + +namespace Spectra { + + +/// +/// \defgroup EigenSolver Eigen Solvers +/// +/// Eigen solvers for different types of problems. +/// + +/// +/// \ingroup EigenSolver +/// +/// This is the base class for symmetric eigen solvers, mainly for internal use. +/// It is kept here to provide the documentation for member functions of concrete eigen solvers +/// such as SymEigsSolver and SymEigsShiftSolver. +/// +template < typename Scalar, + int SelectionRule, + typename OpType, + typename BOpType > +class SymEigsBase +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef Eigen::Array Array; + typedef Eigen::Array BoolArray; + typedef Eigen::Map MapMat; + typedef Eigen::Map MapVec; + typedef Eigen::Map MapConstVec; + + typedef ArnoldiOp ArnoldiOpType; + typedef Lanczos LanczosFac; + +protected: + OpType* m_op; // object to conduct matrix operation, + // e.g. matrix-vector product + const Index m_n; // dimension of matrix A + const Index m_nev; // number of eigenvalues requested + const Index m_ncv; // dimension of Krylov subspace in the Lanczos method + Index m_nmatop; // number of matrix operations called + Index m_niter; // number of restarting iterations + + LanczosFac m_fac; // Lanczos factorization + Vector m_ritz_val; // Ritz values + +private: + Matrix m_ritz_vec; // Ritz vectors + Vector m_ritz_est; // last row of m_ritz_vec, also called the Ritz estimates + BoolArray m_ritz_conv; // indicator of the convergence of Ritz values + int m_info; // status of the computation + + const Scalar m_near_0; // a very small value, but 1.0 / m_near_0 does not overflow + // ~= 1e-307 for the "double" type + const Scalar m_eps; // the machine precision, ~= 1e-16 for the "double" type + const Scalar m_eps23; // m_eps^(2/3), used to test the convergence + + // Implicitly restarted Lanczos factorization + void restart(Index k) + { + if(k >= m_ncv) + return; + + TridiagQR decomp(m_ncv); + Matrix Q = Matrix::Identity(m_ncv, m_ncv); + + for(Index i = k; i < m_ncv; i++) + { + // QR decomposition of H-mu*I, mu is the shift + decomp.compute(m_fac.matrix_H(), m_ritz_val[i]); + + // Q -> Q * Qi + decomp.apply_YQ(Q); + // H -> Q'HQ + // Since QR = H - mu * I, we have H = QR + mu * I + // and therefore Q'HQ = RQ + mu * I + m_fac.compress_H(decomp); + } + + m_fac.compress_V(Q); + m_fac.factorize_from(k, m_ncv, m_nmatop); + + retrieve_ritzpair(); + } + + // Calculates the number of converged Ritz values + Index num_converged(Scalar tol) + { + // thresh = tol * max(m_eps23, abs(theta)), theta for Ritz value + Array thresh = tol * m_ritz_val.head(m_nev).array().abs().max(m_eps23); + Array resid = m_ritz_est.head(m_nev).array().abs() * m_fac.f_norm(); + // Converged "wanted" Ritz values + m_ritz_conv = (resid < thresh); + + return m_ritz_conv.cast().sum(); + } + + // Returns the adjusted nev for restarting + Index nev_adjusted(Index nconv) + { + using std::abs; + + Index nev_new = m_nev; + for(Index i = m_nev; i < m_ncv; i++) + if(abs(m_ritz_est[i]) < m_near_0) nev_new++; + + // Adjust nev_new, according to dsaup2.f line 677~684 in ARPACK + nev_new += std::min(nconv, (m_ncv - nev_new) / 2); + if(nev_new == 1 && m_ncv >= 6) + nev_new = m_ncv / 2; + else if(nev_new == 1 && m_ncv > 2) + nev_new = 2; + + if(nev_new > m_ncv - 1) + nev_new = m_ncv - 1; + + return nev_new; + } + + // Retrieves and sorts Ritz values and Ritz vectors + void retrieve_ritzpair() + { + TridiagEigen decomp(m_fac.matrix_H()); + const Vector& evals = decomp.eigenvalues(); + const Matrix& evecs = decomp.eigenvectors(); + + SortEigenvalue sorting(evals.data(), evals.size()); + std::vector ind = sorting.index(); + + // For BOTH_ENDS, the eigenvalues are sorted according + // to the LARGEST_ALGE rule, so we need to move those smallest + // values to the left + // The order would be + // Largest => Smallest => 2nd largest => 2nd smallest => ... + // We keep this order since the first k values will always be + // the wanted collection, no matter k is nev_updated (used in restart()) + // or is nev (used in sort_ritzpair()) + if(SelectionRule == BOTH_ENDS) + { + std::vector ind_copy(ind); + for(Index i = 0; i < m_ncv; i++) + { + // If i is even, pick values from the left (large values) + // If i is odd, pick values from the right (small values) + if(i % 2 == 0) + ind[i] = ind_copy[i / 2]; + else + ind[i] = ind_copy[m_ncv - 1 - i / 2]; + } + } + + // Copy the Ritz values and vectors to m_ritz_val and m_ritz_vec, respectively + for(Index i = 0; i < m_ncv; i++) + { + m_ritz_val[i] = evals[ind[i]]; + m_ritz_est[i] = evecs(m_ncv - 1, ind[i]); + } + for(Index i = 0; i < m_nev; i++) + { + m_ritz_vec.col(i).noalias() = evecs.col(ind[i]); + } + } + +protected: + // Sorts the first nev Ritz pairs in the specified order + // This is used to return the final results + virtual void sort_ritzpair(int sort_rule) + { + // First make sure that we have a valid index vector + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + std::vector ind = sorting.index(); + + switch(sort_rule) + { + case LARGEST_ALGE: + break; + case LARGEST_MAGN: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case SMALLEST_ALGE: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + case SMALLEST_MAGN: + { + SortEigenvalue sorting(m_ritz_val.data(), m_nev); + ind = sorting.index(); + } + break; + default: + throw std::invalid_argument("unsupported sorting rule"); + } + + Vector new_ritz_val(m_ncv); + Matrix new_ritz_vec(m_ncv, m_nev); + BoolArray new_ritz_conv(m_nev); + + for(Index i = 0; i < m_nev; i++) + { + new_ritz_val[i] = m_ritz_val[ind[i]]; + new_ritz_vec.col(i).noalias() = m_ritz_vec.col(ind[i]); + new_ritz_conv[i] = m_ritz_conv[ind[i]]; + } + + m_ritz_val.swap(new_ritz_val); + m_ritz_vec.swap(new_ritz_vec); + m_ritz_conv.swap(new_ritz_conv); + } + +public: + /// \cond + + SymEigsBase(OpType* op, BOpType* Bop, Index nev, Index ncv) : + m_op(op), + m_n(m_op->rows()), + m_nev(nev), + m_ncv(ncv > m_n ? m_n : ncv), + m_nmatop(0), + m_niter(0), + m_fac(ArnoldiOpType(op, Bop), m_ncv), + m_info(NOT_COMPUTED), + m_near_0(TypeTraits::min() * Scalar(10)), + m_eps(Eigen::NumTraits::epsilon()), + m_eps23(Eigen::numext::pow(m_eps, Scalar(2.0) / 3)) + { + if(nev < 1 || nev > m_n - 1) + throw std::invalid_argument("nev must satisfy 1 <= nev <= n - 1, n is the size of matrix"); + + if(ncv <= nev || ncv > m_n) + throw std::invalid_argument("ncv must satisfy nev < ncv <= n, n is the size of matrix"); + } + + /// + /// Virtual destructor + /// + virtual ~SymEigsBase() {} + + /// \endcond + + /// + /// Initializes the solver by providing an initial residual vector. + /// + /// \param init_resid Pointer to the initial residual vector. + /// + /// **Spectra** (and also **ARPACK**) uses an iterative algorithm + /// to find eigenvalues. This function allows the user to provide the initial + /// residual vector. + /// + void init(const Scalar* init_resid) + { + // Reset all matrices/vectors to zero + m_ritz_val.resize(m_ncv); + m_ritz_vec.resize(m_ncv, m_nev); + m_ritz_est.resize(m_ncv); + m_ritz_conv.resize(m_nev); + + m_ritz_val.setZero(); + m_ritz_vec.setZero(); + m_ritz_est.setZero(); + m_ritz_conv.setZero(); + + m_nmatop = 0; + m_niter = 0; + + // Initialize the Lanczos factorization + MapConstVec v0(init_resid, m_n); + m_fac.init(v0, m_nmatop); + } + + /// + /// Initializes the solver by providing a random initial residual vector. + /// + /// This overloaded function generates a random initial residual vector + /// (with a fixed random seed) for the algorithm. Elements in the vector + /// follow independent Uniform(-0.5, 0.5) distribution. + /// + void init() + { + SimpleRandom rng(0); + Vector init_resid = rng.random_vec(m_n); + init(init_resid.data()); + } + + /// + /// Conducts the major computation procedure. + /// + /// \param maxit Maximum number of iterations allowed in the algorithm. + /// \param tol Precision parameter for the calculated eigenvalues. + /// \param sort_rule Rule to sort the eigenvalues and eigenvectors. + /// Supported values are + /// `Spectra::LARGEST_ALGE`, `Spectra::LARGEST_MAGN`, + /// `Spectra::SMALLEST_ALGE` and `Spectra::SMALLEST_MAGN`, + /// for example `LARGEST_ALGE` indicates that largest eigenvalues + /// come first. Note that this argument is only used to + /// **sort** the final result, and the **selection** rule + /// (e.g. selecting the largest or smallest eigenvalues in the + /// full spectrum) is specified by the template parameter + /// `SelectionRule` of SymEigsSolver. + /// + /// \return Number of converged eigenvalues. + /// + Index compute(Index maxit = 1000, Scalar tol = 1e-10, int sort_rule = LARGEST_ALGE) + { + // The m-step Lanczos factorization + m_fac.factorize_from(1, m_ncv, m_nmatop); + retrieve_ritzpair(); + // Restarting + Index i, nconv = 0, nev_adj; + for(i = 0; i < maxit; i++) + { + nconv = num_converged(tol); + if(nconv >= m_nev) + break; + + nev_adj = nev_adjusted(nconv); + restart(nev_adj); + } + // Sorting results + sort_ritzpair(sort_rule); + + m_niter += i + 1; + m_info = (nconv >= m_nev) ? SUCCESSFUL : NOT_CONVERGING; + + return std::min(m_nev, nconv); + } + + /// + /// Returns the status of the computation. + /// The full list of enumeration values can be found in \ref Enumerations. + /// + int info() const { return m_info; } + + /// + /// Returns the number of iterations used in the computation. + /// + Index num_iterations() const { return m_niter; } + + /// + /// Returns the number of matrix operations used in the computation. + /// + Index num_operations() const { return m_nmatop; } + + /// + /// Returns the converged eigenvalues. + /// + /// \return A vector containing the eigenvalues. + /// Returned vector type will be `Eigen::Vector`, depending on + /// the template parameter `Scalar` defined. + /// + Vector eigenvalues() const + { + const Index nconv = m_ritz_conv.cast().sum(); + Vector res(nconv); + + if(!nconv) + return res; + + Index j = 0; + for(Index i = 0; i < m_nev; i++) + { + if(m_ritz_conv[i]) + { + res[j] = m_ritz_val[i]; + j++; + } + } + + return res; + } + + /// + /// Returns the eigenvectors associated with the converged eigenvalues. + /// + /// \param nvec The number of eigenvectors to return. + /// + /// \return A matrix containing the eigenvectors. + /// Returned matrix type will be `Eigen::Matrix`, + /// depending on the template parameter `Scalar` defined. + /// + virtual Matrix eigenvectors(Index nvec) const + { + const Index nconv = m_ritz_conv.cast().sum(); + nvec = std::min(nvec, nconv); + Matrix res(m_n, nvec); + + if(!nvec) + return res; + + Matrix ritz_vec_conv(m_ncv, nvec); + Index j = 0; + for(Index i = 0; i < m_nev && j < nvec; i++) + { + if(m_ritz_conv[i]) + { + ritz_vec_conv.col(j).noalias() = m_ritz_vec.col(i); + j++; + } + } + + res.noalias() = m_fac.matrix_V() * ritz_vec_conv; + + return res; + } + + /// + /// Returns all converged eigenvectors. + /// + virtual Matrix eigenvectors() const + { + return eigenvectors(m_nev); + } +}; + + +} // namespace Spectra + +#endif // SYM_EIGS_BASE_H diff --git a/src/external/Spectra/include/Spectra/SymEigsShiftSolver.h b/src/external/Spectra/include/Spectra/SymEigsShiftSolver.h new file mode 100644 index 00000000..56bd44ec --- /dev/null +++ b/src/external/Spectra/include/Spectra/SymEigsShiftSolver.h @@ -0,0 +1,205 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_EIGS_SHIFT_SOLVER_H +#define SYM_EIGS_SHIFT_SOLVER_H + +#include + +#include "SymEigsBase.h" +#include "Util/SelectionRule.h" +#include "MatOp/DenseSymShiftSolve.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This class implements the eigen solver for real symmetric matrices using +/// the **shift-and-invert mode**. The background information of the symmetric +/// eigen solver is documented in the SymEigsSolver class. Here we focus on +/// explaining the shift-and-invert mode. +/// +/// The shift-and-invert mode is based on the following fact: +/// If \f$\lambda\f$ and \f$x\f$ are a pair of eigenvalue and eigenvector of +/// matrix \f$A\f$, such that \f$Ax=\lambda x\f$, then for any \f$\sigma\f$, +/// we have +/// \f[(A-\sigma I)^{-1}x=\nu x\f] +/// where +/// \f[\nu=\frac{1}{\lambda-\sigma}\f] +/// which indicates that \f$(\nu, x)\f$ is an eigenpair of the matrix +/// \f$(A-\sigma I)^{-1}\f$. +/// +/// Therefore, if we pass the matrix operation \f$(A-\sigma I)^{-1}y\f$ +/// (rather than \f$Ay\f$) to the eigen solver, then we would get the desired +/// values of \f$\nu\f$, and \f$\lambda\f$ can also be easily obtained by noting +/// that \f$\lambda=\sigma+\nu^{-1}\f$. +/// +/// The reason why we need this type of manipulation is that +/// the algorithm of **Spectra** (and also **ARPACK**) +/// is good at finding eigenvalues with large magnitude, but may fail in looking +/// for eigenvalues that are close to zero. However, if we really need them, we +/// can set \f$\sigma=0\f$, find the largest eigenvalues of \f$A^{-1}\f$, and then +/// transform back to \f$\lambda\f$, since in this case largest values of \f$\nu\f$ +/// implies smallest values of \f$\lambda\f$. +/// +/// To summarize, in the shift-and-invert mode, the selection rule will apply to +/// \f$\nu=1/(\lambda-\sigma)\f$ rather than \f$\lambda\f$. So a selection rule +/// of `LARGEST_MAGN` combined with shift \f$\sigma\f$ will find eigenvalues of +/// \f$A\f$ that are closest to \f$\sigma\f$. But note that the eigenvalues() +/// method will always return the eigenvalues in the original problem (i.e., +/// returning \f$\lambda\f$ rather than \f$\nu\f$), and eigenvectors are the +/// same for both the original problem and the shifted-and-inverted problem. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the shifted-and-inverted eigenvalues. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class. Users could either +/// use the wrapper classes such as DenseSymShiftSolve and +/// SparseSymShiftSolve, or define their +/// own that implements all the public member functions as in +/// DenseSymShiftSolve. +/// +/// Below is an example that illustrates the use of the shift-and-invert mode: +/// +/// \code{.cpp} +/// #include +/// #include +/// // is implicitly included +/// #include +/// +/// using namespace Spectra; +/// +/// int main() +/// { +/// // A size-10 diagonal matrix with elements 1, 2, ..., 10 +/// Eigen::MatrixXd M = Eigen::MatrixXd::Zero(10, 10); +/// for(int i = 0; i < M.rows(); i++) +/// M(i, i) = i + 1; +/// +/// // Construct matrix operation object using the wrapper class +/// DenseSymShiftSolve op(M); +/// +/// // Construct eigen solver object with shift 0 +/// // This will find eigenvalues that are closest to 0 +/// SymEigsShiftSolver< double, LARGEST_MAGN, +/// DenseSymShiftSolve > eigs(&op, 3, 6, 0.0); +/// +/// eigs.init(); +/// eigs.compute(); +/// if(eigs.info() == SUCCESSFUL) +/// { +/// Eigen::VectorXd evalues = eigs.eigenvalues(); +/// // Will get (3.0, 2.0, 1.0) +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// } +/// +/// return 0; +/// } +/// \endcode +/// +/// Also an example for user-supplied matrix shift-solve operation class: +/// +/// \code{.cpp} +/// #include +/// #include +/// #include +/// +/// using namespace Spectra; +/// +/// // M = diag(1, 2, ..., 10) +/// class MyDiagonalTenShiftSolve +/// { +/// private: +/// double sigma_; +/// public: +/// int rows() { return 10; } +/// int cols() { return 10; } +/// void set_shift(double sigma) { sigma_ = sigma; } +/// // y_out = inv(A - sigma * I) * x_in +/// // inv(A - sigma * I) = diag(1/(1-sigma), 1/(2-sigma), ...) +/// void perform_op(double *x_in, double *y_out) +/// { +/// for(int i = 0; i < rows(); i++) +/// { +/// y_out[i] = x_in[i] / (i + 1 - sigma_); +/// } +/// } +/// }; +/// +/// int main() +/// { +/// MyDiagonalTenShiftSolve op; +/// // Find three eigenvalues that are closest to 3.14 +/// SymEigsShiftSolver eigs(&op, 3, 6, 3.14); +/// eigs.init(); +/// eigs.compute(); +/// if(eigs.info() == SUCCESSFUL) +/// { +/// Eigen::VectorXd evalues = eigs.eigenvalues(); +/// // Will get (4.0, 3.0, 2.0) +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// } +/// +/// return 0; +/// } +/// \endcode +/// +template > +class SymEigsShiftSolver: public SymEigsBase +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Array Array; + + const Scalar m_sigma; + + // First transform back the Ritz values, and then sort + void sort_ritzpair(int sort_rule) + { + Array m_ritz_val_org = Scalar(1.0) / this->m_ritz_val.head(this->m_nev).array() + m_sigma; + this->m_ritz_val.head(this->m_nev) = m_ritz_val_org; + SymEigsBase::sort_ritzpair(sort_rule); + } + +public: + /// + /// Constructor to create a eigen solver object using the shift-and-invert mode. + /// + /// \param op Pointer to the matrix operation object, which should implement + /// the shift-solve operation of \f$A\f$: calculating + /// \f$(A-\sigma I)^{-1}v\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper class such as DenseSymShiftSolve, or + /// define their own that implements all the public member functions + /// as in DenseSymShiftSolve. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-1\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv_` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev < ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev\f$. + /// \param sigma The value of the shift. + /// + SymEigsShiftSolver(OpType* op, Index nev, Index ncv, Scalar sigma) : + SymEigsBase(op, NULL, nev, ncv), + m_sigma(sigma) + { + this->m_op->set_shift(m_sigma); + } +}; + + +} // namespace Spectra + +#endif // SYM_EIGS_SHIFT_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/SymEigsSolver.h b/src/external/Spectra/include/Spectra/SymEigsSolver.h new file mode 100644 index 00000000..8ba3e509 --- /dev/null +++ b/src/external/Spectra/include/Spectra/SymEigsSolver.h @@ -0,0 +1,174 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_EIGS_SOLVER_H +#define SYM_EIGS_SOLVER_H + +#include + +#include "SymEigsBase.h" +#include "Util/SelectionRule.h" +#include "MatOp/DenseSymMatProd.h" + +namespace Spectra { + + +/// +/// \ingroup EigenSolver +/// +/// This class implements the eigen solver for real symmetric matrices, i.e., +/// to solve \f$Ax=\lambda x\f$ where \f$A\f$ is symmetric. +/// +/// **Spectra** is designed to calculate a specified number (\f$k\f$) +/// of eigenvalues of a large square matrix (\f$A\f$). Usually \f$k\f$ is much +/// less than the size of the matrix (\f$n\f$), so that only a few eigenvalues +/// and eigenvectors are computed. +/// +/// Rather than providing the whole \f$A\f$ matrix, the algorithm only requires +/// the matrix-vector multiplication operation of \f$A\f$. Therefore, users of +/// this solver need to supply a class that computes the result of \f$Av\f$ +/// for any given vector \f$v\f$. The name of this class should be given to +/// the template parameter `OpType`, and instance of this class passed to +/// the constructor of SymEigsSolver. +/// +/// If the matrix \f$A\f$ is already stored as a matrix object in **Eigen**, +/// for example `Eigen::MatrixXd`, then there is an easy way to construct such +/// matrix operation class, by using the built-in wrapper class DenseSymMatProd +/// which wraps an existing matrix object in **Eigen**. This is also the +/// default template parameter for SymEigsSolver. For sparse matrices, the +/// wrapper class SparseSymMatProd can be used similarly. +/// +/// If the users need to define their own matrix-vector multiplication operation +/// class, it should implement all the public member functions as in DenseSymMatProd. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the requested eigenvalues, for example `LARGEST_MAGN` +/// to retrieve eigenvalues with the largest magnitude. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class. Users could either +/// use the wrapper classes such as DenseSymMatProd and +/// SparseSymMatProd, or define their +/// own that implements all the public member functions as in +/// DenseSymMatProd. +/// +/// Below is an example that demonstrates the usage of this class. +/// +/// \code{.cpp} +/// #include +/// #include +/// // is implicitly included +/// #include +/// +/// using namespace Spectra; +/// +/// int main() +/// { +/// // We are going to calculate the eigenvalues of M +/// Eigen::MatrixXd A = Eigen::MatrixXd::Random(10, 10); +/// Eigen::MatrixXd M = A + A.transpose(); +/// +/// // Construct matrix operation object using the wrapper class DenseSymMatProd +/// DenseSymMatProd op(M); +/// +/// // Construct eigen solver object, requesting the largest three eigenvalues +/// SymEigsSolver< double, LARGEST_ALGE, DenseSymMatProd > eigs(&op, 3, 6); +/// +/// // Initialize and compute +/// eigs.init(); +/// int nconv = eigs.compute(); +/// +/// // Retrieve results +/// Eigen::VectorXd evalues; +/// if(eigs.info() == SUCCESSFUL) +/// evalues = eigs.eigenvalues(); +/// +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// +/// return 0; +/// } +/// \endcode +/// +/// And here is an example for user-supplied matrix operation class. +/// +/// \code{.cpp} +/// #include +/// #include +/// #include +/// +/// using namespace Spectra; +/// +/// // M = diag(1, 2, ..., 10) +/// class MyDiagonalTen +/// { +/// public: +/// int rows() { return 10; } +/// int cols() { return 10; } +/// // y_out = M * x_in +/// void perform_op(double *x_in, double *y_out) +/// { +/// for(int i = 0; i < rows(); i++) +/// { +/// y_out[i] = x_in[i] * (i + 1); +/// } +/// } +/// }; +/// +/// int main() +/// { +/// MyDiagonalTen op; +/// SymEigsSolver eigs(&op, 3, 6); +/// eigs.init(); +/// eigs.compute(); +/// if(eigs.info() == SUCCESSFUL) +/// { +/// Eigen::VectorXd evalues = eigs.eigenvalues(); +/// // Will get (10, 9, 8) +/// std::cout << "Eigenvalues found:\n" << evalues << std::endl; +/// } +/// +/// return 0; +/// } +/// \endcode +/// +template < typename Scalar = double, + int SelectionRule = LARGEST_MAGN, + typename OpType = DenseSymMatProd > +class SymEigsSolver: public SymEigsBase +{ +private: + typedef Eigen::Index Index; + +public: + /// + /// Constructor to create a solver object. + /// + /// \param op Pointer to the matrix operation object, which should implement + /// the matrix-vector multiplication operation of \f$A\f$: + /// calculating \f$Av\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper class such as DenseSymMatProd, or + /// define their own that implements all the public member functions + /// as in DenseSymMatProd. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-1\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev < ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev\f$. + /// + SymEigsSolver(OpType* op, Index nev, Index ncv) : + SymEigsBase(op, NULL, nev, ncv) + {} + +}; + + +} // namespace Spectra + +#endif // SYM_EIGS_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/SymGEigsSolver.h b/src/external/Spectra/include/Spectra/SymGEigsSolver.h new file mode 100644 index 00000000..8e774284 --- /dev/null +++ b/src/external/Spectra/include/Spectra/SymGEigsSolver.h @@ -0,0 +1,334 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SYM_GEIGS_SOLVER_H +#define SYM_GEIGS_SOLVER_H + +#include "SymEigsBase.h" +#include "Util/GEigsMode.h" +#include "MatOp/internal/SymGEigsCholeskyOp.h" +#include "MatOp/internal/SymGEigsRegInvOp.h" + +namespace Spectra { + + +/// +/// \defgroup GEigenSolver Generalized Eigen Solvers +/// +/// Generalized eigen solvers for different types of problems. +/// + +/// +/// \ingroup GEigenSolver +/// +/// This class implements the generalized eigen solver for real symmetric +/// matrices, i.e., to solve \f$Ax=\lambda Bx\f$ where \f$A\f$ is symmetric and +/// \f$B\f$ is positive definite. +/// +/// There are two modes of this solver, specified by the template parameter +/// GEigsMode. See the pages for the specialized classes for details. +/// - The Cholesky mode assumes that \f$B\f$ can be factorized using Cholesky +/// decomposition, which is the preferred mode when the decomposition is +/// available. (This can be easily done in Eigen using the dense or sparse +/// Cholesky solver.) +/// See \ref SymGEigsSolver "SymGEigsSolver (Cholesky mode)" for more details. +/// - The regular inverse mode requires the matrix-vector product \f$Bv\f$ and the +/// linear equation solving operation \f$B^{-1}v\f$. This mode should only be +/// used when the Cholesky decomposition of \f$B\f$ is hard to implement, or +/// when computing \f$B^{-1}v\f$ is much faster than the Cholesky decomposition. +/// See \ref SymGEigsSolver "SymGEigsSolver (Regular inverse mode)" for more details. + +// Empty class template +template < typename Scalar, + int SelectionRule, + typename OpType, + typename BOpType, + int GEigsMode > +class SymGEigsSolver +{}; + + + +/// +/// \ingroup GEigenSolver +/// +/// This class implements the generalized eigen solver for real symmetric +/// matrices using Cholesky decomposition, i.e., to solve \f$Ax=\lambda Bx\f$ +/// where \f$A\f$ is symmetric and \f$B\f$ is positive definite with the Cholesky +/// decomposition \f$B=LL'\f$. +/// +/// This solver requires two matrix operation objects: one for \f$A\f$ that implements +/// the matrix multiplication \f$Av\f$, and one for \f$B\f$ that implements the lower +/// and upper triangular solving \f$L^{-1}v\f$ and \f$(L')^{-1}v\f$. +/// +/// If \f$A\f$ and \f$B\f$ are stored as Eigen matrices, then the first operation +/// can be created using the DenseSymMatProd or SparseSymMatProd classes, and +/// the second operation can be created using the DenseCholesky or SparseCholesky +/// classes. If the users need to define their own operation classes, then they +/// should implement all the public member functions as in those built-in classes. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the requested eigenvalues, for example `LARGEST_MAGN` +/// to retrieve eigenvalues with the largest magnitude. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class for \f$A\f$. Users could either +/// use the wrapper classes such as DenseSymMatProd and +/// SparseSymMatProd, or define their +/// own that implements all the public member functions as in +/// DenseSymMatProd. +/// \tparam BOpType The name of the matrix operation class for \f$B\f$. Users could either +/// use the wrapper classes such as DenseCholesky and +/// SparseCholesky, or define their +/// own that implements all the public member functions as in +/// DenseCholesky. +/// \tparam GEigsMode Mode of the generalized eigen solver. In this solver +/// it is Spectra::GEIGS_CHOLESKY. +/// +/// Below is an example that demonstrates the usage of this class. +/// +/// \code{.cpp} +/// #include +/// #include +/// #include +/// #include +/// #include +/// #include +/// #include +/// +/// using namespace Spectra; +/// +/// int main() +/// { +/// // We are going to solve the generalized eigenvalue problem A * x = lambda * B * x +/// const int n = 100; +/// +/// // Define the A matrix +/// Eigen::MatrixXd M = Eigen::MatrixXd::Random(n, n); +/// Eigen::MatrixXd A = M + M.transpose(); +/// +/// // Define the B matrix, a band matrix with 2 on the diagonal and 1 on the subdiagonals +/// Eigen::SparseMatrix B(n, n); +/// B.reserve(Eigen::VectorXi::Constant(n, 3)); +/// for(int i = 0; i < n; i++) +/// { +/// B.insert(i, i) = 2.0; +/// if(i > 0) +/// B.insert(i - 1, i) = 1.0; +/// if(i < n - 1) +/// B.insert(i + 1, i) = 1.0; +/// } +/// +/// // Construct matrix operation object using the wrapper classes +/// DenseSymMatProd op(A); +/// SparseCholesky Bop(B); +/// +/// // Construct generalized eigen solver object, requesting the largest three generalized eigenvalues +/// SymGEigsSolver, SparseCholesky, GEIGS_CHOLESKY> +/// geigs(&op, &Bop, 3, 6); +/// +/// // Initialize and compute +/// geigs.init(); +/// int nconv = geigs.compute(); +/// +/// // Retrieve results +/// Eigen::VectorXd evalues; +/// Eigen::MatrixXd evecs; +/// if(geigs.info() == SUCCESSFUL) +/// { +/// evalues = geigs.eigenvalues(); +/// evecs = geigs.eigenvectors(); +/// } +/// +/// std::cout << "Generalized eigenvalues found:\n" << evalues << std::endl; +/// std::cout << "Generalized eigenvectors found:\n" << evecs.topRows(10) << std::endl; +/// +/// // Verify results using the generalized eigen solver in Eigen +/// Eigen::MatrixXd Bdense = B; +/// Eigen::GeneralizedSelfAdjointEigenSolver es(A, Bdense); +/// +/// std::cout << "Generalized eigenvalues:\n" << es.eigenvalues().tail(3) << std::endl; +/// std::cout << "Generalized eigenvectors:\n" << es.eigenvectors().rightCols(3).topRows(10) << std::endl; +/// +/// return 0; +/// } +/// \endcode + +// Partial specialization for GEigsMode = GEIGS_CHOLESKY +template < typename Scalar, + int SelectionRule, + typename OpType, + typename BOpType > +class SymGEigsSolver: + public SymEigsBase, IdentityBOp> +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + BOpType* m_Bop; + +public: + /// + /// Constructor to create a solver object. + /// + /// \param op Pointer to the \f$A\f$ matrix operation object. It + /// should implement the matrix-vector multiplication operation of \f$A\f$: + /// calculating \f$Av\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper classes such as DenseSymMatProd, or + /// define their own that implements all the public member functions + /// as in DenseSymMatProd. + /// \param Bop Pointer to the \f$B\f$ matrix operation object. It + /// represents a Cholesky decomposition of \f$B\f$, and should + /// implement the lower and upper triangular solving operations: + /// calculating \f$L^{-1}v\f$ and \f$(L')^{-1}v\f$ for any vector + /// \f$v\f$, where \f$LL'=B\f$. Users could either + /// create the object from the wrapper classes such as DenseCholesky, or + /// define their own that implements all the public member functions + /// as in DenseCholesky. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-1\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev < ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev\f$. + /// + SymGEigsSolver(OpType* op, BOpType* Bop, Index nev, Index ncv) : + SymEigsBase, IdentityBOp>( + new SymGEigsCholeskyOp(*op, *Bop), NULL, nev, ncv + ), + m_Bop(Bop) + {} + + /// \cond + + ~SymGEigsSolver() + { + // m_op contains the constructed SymGEigsCholeskyOp object + delete this->m_op; + } + + Matrix eigenvectors(Index nvec) const + { + Matrix res = SymEigsBase, IdentityBOp>::eigenvectors(nvec); + Vector tmp(res.rows()); + const Index nconv = res.cols(); + for(Index i = 0; i < nconv; i++) + { + m_Bop->upper_triangular_solve(&res(0, i), tmp.data()); + res.col(i).noalias() = tmp; + } + + return res; + } + + Matrix eigenvectors() const + { + return SymGEigsSolver::eigenvectors(this->m_nev); + } + + /// \endcond +}; + + + +/// +/// \ingroup GEigenSolver +/// +/// This class implements the generalized eigen solver for real symmetric +/// matrices in the regular inverse mode, i.e., to solve \f$Ax=\lambda Bx\f$ +/// where \f$A\f$ is symmetric, and \f$B\f$ is positive definite with the operations +/// defined below. +/// +/// This solver requires two matrix operation objects: one for \f$A\f$ that implements +/// the matrix multiplication \f$Av\f$, and one for \f$B\f$ that implements the +/// matrix-vector product \f$Bv\f$ and the linear equation solving operation \f$B^{-1}v\f$. +/// +/// If \f$A\f$ and \f$B\f$ are stored as Eigen matrices, then the first operation +/// can be created using the DenseSymMatProd or SparseSymMatProd classes, and +/// the second operation can be created using the SparseRegularInverse class. There is no +/// wrapper class for a dense \f$B\f$ matrix since in this case the Cholesky mode +/// is always preferred. If the users need to define their own operation classes, then they +/// should implement all the public member functions as in those built-in classes. +/// +/// \tparam Scalar The element type of the matrix. +/// Currently supported types are `float`, `double` and `long double`. +/// \tparam SelectionRule An enumeration value indicating the selection rule of +/// the requested eigenvalues, for example `LARGEST_MAGN` +/// to retrieve eigenvalues with the largest magnitude. +/// The full list of enumeration values can be found in +/// \ref Enumerations. +/// \tparam OpType The name of the matrix operation class for \f$A\f$. Users could either +/// use the wrapper classes such as DenseSymMatProd and +/// SparseSymMatProd, or define their +/// own that implements all the public member functions as in +/// DenseSymMatProd. +/// \tparam BOpType The name of the matrix operation class for \f$B\f$. Users could either +/// use the wrapper class SparseRegularInverse, or define their +/// own that implements all the public member functions as in +/// SparseRegularInverse. +/// \tparam GEigsMode Mode of the generalized eigen solver. In this solver +/// it is Spectra::GEIGS_REGULAR_INVERSE. +/// + +// Partial specialization for GEigsMode = GEIGS_REGULAR_INVERSE +template < typename Scalar, + int SelectionRule, + typename OpType, + typename BOpType > +class SymGEigsSolver: + public SymEigsBase, BOpType> +{ +private: + typedef Eigen::Index Index; + +public: + /// + /// Constructor to create a solver object. + /// + /// \param op Pointer to the \f$A\f$ matrix operation object. It + /// should implement the matrix-vector multiplication operation of \f$A\f$: + /// calculating \f$Av\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper classes such as DenseSymMatProd, or + /// define their own that implements all the public member functions + /// as in DenseSymMatProd. + /// \param Bop Pointer to the \f$B\f$ matrix operation object. It should + /// implement the multiplication operation \f$Bv\f$ and the linear equation + /// solving operation \f$B^{-1}v\f$ for any vector \f$v\f$. Users could either + /// create the object from the wrapper class SparseRegularInverse, or + /// define their own that implements all the public member functions + /// as in SparseRegularInverse. + /// \param nev Number of eigenvalues requested. This should satisfy \f$1\le nev \le n-1\f$, + /// where \f$n\f$ is the size of matrix. + /// \param ncv Parameter that controls the convergence speed of the algorithm. + /// Typically a larger `ncv` means faster convergence, but it may + /// also result in greater memory use and more matrix operations + /// in each iteration. This parameter must satisfy \f$nev < ncv \le n\f$, + /// and is advised to take \f$ncv \ge 2\cdot nev\f$. + /// + SymGEigsSolver(OpType* op, BOpType* Bop, Index nev, Index ncv) : + SymEigsBase, BOpType>( + new SymGEigsRegInvOp(*op, *Bop), Bop, nev, ncv + ) + {} + + /// \cond + ~SymGEigsSolver() + { + // m_op contains the constructed SymGEigsRegInvOp object + delete this->m_op; + } + /// \endcond +}; + + +} // namespace Spectra + +#endif // SYM_GEIGS_SOLVER_H diff --git a/src/external/Spectra/include/Spectra/Util/CompInfo.h b/src/external/Spectra/include/Spectra/Util/CompInfo.h new file mode 100644 index 00000000..b8e639d6 --- /dev/null +++ b/src/external/Spectra/include/Spectra/Util/CompInfo.h @@ -0,0 +1,37 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef COMP_INFO_H +#define COMP_INFO_H + +namespace Spectra { + + +/// +/// \ingroup Enumerations +/// +/// The enumeration to report the status of computation. +/// +enum COMPUTATION_INFO +{ + SUCCESSFUL = 0, ///< Computation was successful. + + NOT_COMPUTED, ///< Used in eigen solvers, indicating that computation + ///< has not been conducted. Users should call + ///< the `compute()` member function of solvers. + + NOT_CONVERGING, ///< Used in eigen solvers, indicating that some eigenvalues + ///< did not converge. The `compute()` + ///< function returns the number of converged eigenvalues. + + NUMERICAL_ISSUE ///< Used in Cholesky decomposition, indicating that the + ///< matrix is not positive definite. +}; + + +} // namespace Spectra + +#endif // COMP_INFO_H diff --git a/src/external/Spectra/include/Spectra/Util/GEigsMode.h b/src/external/Spectra/include/Spectra/Util/GEigsMode.h new file mode 100644 index 00000000..d03f269d --- /dev/null +++ b/src/external/Spectra/include/Spectra/Util/GEigsMode.h @@ -0,0 +1,34 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef GEIGS_MODE_H +#define GEIGS_MODE_H + +namespace Spectra { + + +/// +/// \ingroup Enumerations +/// +/// The enumeration to specify the mode of generalized eigenvalue solver. +/// +enum GEIGS_MODE +{ + GEIGS_CHOLESKY = 0, ///< Using Cholesky decomposition to solve generalized eigenvalues. + + GEIGS_REGULAR_INVERSE, ///< Regular inverse mode for generalized eigenvalue solver. + + GEIGS_SHIFT_INVERT, ///< Shift-and-invert mode for generalized eigenvalue solver. + + GEIGS_BUCKLING, ///< Buckling mode for generalized eigenvalue solver. + + GEIGS_CAYLEY ///< Cayley transformation mode for generalized eigenvalue solver. +}; + + +} // namespace Spectra + +#endif // GEIGS_MODE_H diff --git a/src/external/Spectra/include/Spectra/Util/SelectionRule.h b/src/external/Spectra/include/Spectra/Util/SelectionRule.h new file mode 100644 index 00000000..19f71dcf --- /dev/null +++ b/src/external/Spectra/include/Spectra/Util/SelectionRule.h @@ -0,0 +1,277 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SELECTION_RULE_H +#define SELECTION_RULE_H + +#include // std::vector +#include // std::abs +#include // std::sort +#include // std::complex +#include // std::pair +#include // std::invalid_argument + +namespace Spectra { + + +/// +/// \defgroup Enumerations +/// +/// Enumeration types for the selection rule of eigenvalues. +/// + +/// +/// \ingroup Enumerations +/// +/// The enumeration of selection rules of desired eigenvalues. +/// +enum SELECT_EIGENVALUE +{ + LARGEST_MAGN = 0, ///< Select eigenvalues with largest magnitude. Magnitude + ///< means the absolute value for real numbers and norm for + ///< complex numbers. Applies to both symmetric and general + ///< eigen solvers. + + LARGEST_REAL, ///< Select eigenvalues with largest real part. Only for general eigen solvers. + + LARGEST_IMAG, ///< Select eigenvalues with largest imaginary part (in magnitude). Only for general eigen solvers. + + LARGEST_ALGE, ///< Select eigenvalues with largest algebraic value, considering + ///< any negative sign. Only for symmetric eigen solvers. + + SMALLEST_MAGN, ///< Select eigenvalues with smallest magnitude. Applies to both symmetric and general + ///< eigen solvers. + + SMALLEST_REAL, ///< Select eigenvalues with smallest real part. Only for general eigen solvers. + + SMALLEST_IMAG, ///< Select eigenvalues with smallest imaginary part (in magnitude). Only for general eigen solvers. + + SMALLEST_ALGE, ///< Select eigenvalues with smallest algebraic value. Only for symmetric eigen solvers. + + BOTH_ENDS ///< Select eigenvalues half from each end of the spectrum. When + ///< `nev` is odd, compute more from the high end. Only for symmetric eigen solvers. +}; + +/// +/// \ingroup Enumerations +/// +/// The enumeration of selection rules of desired eigenvalues. Alias for `SELECT_EIGENVALUE`. +/// +enum SELECT_EIGENVALUE_ALIAS +{ + WHICH_LM = 0, ///< Alias for `LARGEST_MAGN` + WHICH_LR, ///< Alias for `LARGEST_REAL` + WHICH_LI, ///< Alias for `LARGEST_IMAG` + WHICH_LA, ///< Alias for `LARGEST_ALGE` + WHICH_SM, ///< Alias for `SMALLEST_MAGN` + WHICH_SR, ///< Alias for `SMALLEST_REAL` + WHICH_SI, ///< Alias for `SMALLEST_IMAG` + WHICH_SA, ///< Alias for `SMALLEST_ALGE` + WHICH_BE ///< Alias for `BOTH_ENDS` +}; + +/// \cond + +// Get the element type of a "scalar" +// ElemType => double +// ElemType< std::complex > => double +template +class ElemType +{ +public: + typedef T type; +}; + +template +class ElemType< std::complex > +{ +public: + typedef T type; +}; + +// When comparing eigenvalues, we first calculate the "target" +// to sort. For example, if we want to choose the eigenvalues with +// largest magnitude, the target will be -abs(x). +// The minus sign is due to the fact that std::sort() sorts in ascending order. + +// Default target: throw an exception +template +class SortingTarget +{ +public: + static typename ElemType::type get(const Scalar& val) + { + using std::abs; + throw std::invalid_argument("incompatible selection rule"); + return -abs(val); + } +}; + +// Specialization for LARGEST_MAGN +// This covers [float, double, complex] x [LARGEST_MAGN] +template +class SortingTarget +{ +public: + static typename ElemType::type get(const Scalar& val) + { + using std::abs; + return -abs(val); + } +}; + +// Specialization for LARGEST_REAL +// This covers [complex] x [LARGEST_REAL] +template +class SortingTarget, LARGEST_REAL> +{ +public: + static RealType get(const std::complex& val) + { + return -val.real(); + } +}; + +// Specialization for LARGEST_IMAG +// This covers [complex] x [LARGEST_IMAG] +template +class SortingTarget, LARGEST_IMAG> +{ +public: + static RealType get(const std::complex& val) + { + using std::abs; + return -abs(val.imag()); + } +}; + +// Specialization for LARGEST_ALGE +// This covers [float, double] x [LARGEST_ALGE] +template +class SortingTarget +{ +public: + static Scalar get(const Scalar& val) + { + return -val; + } +}; + +// Here BOTH_ENDS is the same as LARGEST_ALGE, but +// we need some additional steps, which are done in +// SymEigsSolver.h => retrieve_ritzpair(). +// There we move the smallest values to the proper locations. +template +class SortingTarget +{ +public: + static Scalar get(const Scalar& val) + { + return -val; + } +}; + +// Specialization for SMALLEST_MAGN +// This covers [float, double, complex] x [SMALLEST_MAGN] +template +class SortingTarget +{ +public: + static typename ElemType::type get(const Scalar& val) + { + using std::abs; + return abs(val); + } +}; + +// Specialization for SMALLEST_REAL +// This covers [complex] x [SMALLEST_REAL] +template +class SortingTarget, SMALLEST_REAL> +{ +public: + static RealType get(const std::complex& val) + { + return val.real(); + } +}; + +// Specialization for SMALLEST_IMAG +// This covers [complex] x [SMALLEST_IMAG] +template +class SortingTarget, SMALLEST_IMAG> +{ +public: + static RealType get(const std::complex& val) + { + using std::abs; + return abs(val.imag()); + } +}; + +// Specialization for SMALLEST_ALGE +// This covers [float, double] x [SMALLEST_ALGE] +template +class SortingTarget +{ +public: + static Scalar get(const Scalar& val) + { + return val; + } +}; + +// Sort eigenvalues and return the order index +template +class PairComparator +{ +public: + bool operator() (const PairType& v1, const PairType& v2) + { + return v1.first < v2.first; + } +}; + +template +class SortEigenvalue +{ +private: + typedef typename ElemType::type TargetType; // Type of the sorting target, will be + // a floating number type, e.g. "double" + typedef std::pair PairType; // Type of the sorting pair, including + // the sorting target and the index + + std::vector pair_sort; + +public: + SortEigenvalue(const T* start, int size) : + pair_sort(size) + { + for(int i = 0; i < size; i++) + { + pair_sort[i].first = SortingTarget::get(start[i]); + pair_sort[i].second = i; + } + PairComparator comp; + std::sort(pair_sort.begin(), pair_sort.end(), comp); + } + + std::vector index() + { + std::vector ind(pair_sort.size()); + for(unsigned int i = 0; i < ind.size(); i++) + ind[i] = pair_sort[i].second; + + return ind; + } +}; + +/// \endcond + + +} // namespace Spectra + +#endif // SELECTION_RULE_H diff --git a/src/external/Spectra/include/Spectra/Util/SimpleRandom.h b/src/external/Spectra/include/Spectra/Util/SimpleRandom.h new file mode 100644 index 00000000..7b1e6162 --- /dev/null +++ b/src/external/Spectra/include/Spectra/Util/SimpleRandom.h @@ -0,0 +1,93 @@ +// Copyright (C) 2016-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef SIMPLE_RANDOM_H +#define SIMPLE_RANDOM_H + +#include + +/// \cond + +namespace Spectra { + + +// We need a simple pseudo random number generator here: +// 1. It is used to generate initial and restarted residual vector. +// 2. It is not necessary to be so "random" and advanced. All we hope +// is that the residual vector is not in the space spanned by the +// current Krylov space. This should be met almost surely. +// 3. We don't want to call RNG in C++, since we actually want the +// algorithm to be deterministic. Also, calling RNG in C/C++ is not +// allowed in R packages submitted to CRAN. +// 4. The method should be as simple as possible, so an LCG is enough. +// 5. Based on public domain code by Ray Gardner +// http://stjarnhimlen.se/snippets/rg_rand.c + + +template +class SimpleRandom +{ +private: + typedef Eigen::Index Index; + typedef Eigen::Matrix Vector; + + const unsigned int m_a; // multiplier + const unsigned long m_max; // 2^31 - 1 + long m_rand; + + inline long next_long_rand(long seed) + { + unsigned long lo, hi; + + lo = m_a * (long)(seed & 0xFFFF); + hi = m_a * (long)((unsigned long)seed >> 16); + lo += (hi & 0x7FFF) << 16; + if(lo > m_max) + { + lo &= m_max; + ++lo; + } + lo += hi >> 15; + if(lo > m_max) + { + lo &= m_max; + ++lo; + } + return (long)lo; + } +public: + SimpleRandom(unsigned long init_seed) : + m_a(16807), + m_max(2147483647L), + m_rand(init_seed ? (init_seed & m_max) : 1) + {} + + Scalar random() + { + m_rand = next_long_rand(m_rand); + return Scalar(m_rand) / Scalar(m_max) - Scalar(0.5); + } + + // Vector of random numbers of type Scalar + // Ranging from -0.5 to 0.5 + Vector random_vec(const Index len) + { + Vector res(len); + for(Index i = 0; i < len; i++) + { + m_rand = next_long_rand(m_rand); + res[i] = Scalar(m_rand) / Scalar(m_max) - Scalar(0.5); + } + return res; + } +}; + + +} // namespace Spectra + +/// \endcond + +#endif // SIMPLE_RANDOM_H diff --git a/src/external/Spectra/include/Spectra/Util/TypeTraits.h b/src/external/Spectra/include/Spectra/Util/TypeTraits.h new file mode 100644 index 00000000..a4cc05b2 --- /dev/null +++ b/src/external/Spectra/include/Spectra/Util/TypeTraits.h @@ -0,0 +1,73 @@ +// Copyright (C) 2018-2019 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef TYPE_TRAITS_H +#define TYPE_TRAITS_H + +#include +#include + +/// \cond + +namespace Spectra { + + +// For a real value type "Scalar", we want to know its smallest +// positive value, i.e., std::numeric_limits::min(). +// However, we must take non-standard value types into account, +// so we rely on Eigen::NumTraits. +// +// Eigen::NumTraits has defined epsilon() and lowest(), but +// lowest() means negative highest(), which is a very small +// negative value. +// +// Therefore, we manually define this limit, and use eplison()^3 +// to mimic it for non-standard types. + +// Generic definition +template +struct TypeTraits +{ + static inline Scalar min() + { + return Eigen::numext::pow(Eigen::NumTraits::epsilon(), Scalar(3)); + } +}; + +// Full specialization +template <> +struct TypeTraits +{ + static inline float min() + { + return std::numeric_limits::min(); + } +}; + +template <> +struct TypeTraits +{ + static inline double min() + { + return std::numeric_limits::min(); + } +}; + +template <> +struct TypeTraits +{ + static inline long double min() + { + return std::numeric_limits::min(); + } +}; + + +} // namespace Spectra + +/// \endcond + +#endif // TYPE_TRAITS_H diff --git a/src/external/Spectra/include/Spectra/contrib/LOBPCGSolver.h b/src/external/Spectra/include/Spectra/contrib/LOBPCGSolver.h new file mode 100644 index 00000000..5ca001f6 --- /dev/null +++ b/src/external/Spectra/include/Spectra/contrib/LOBPCGSolver.h @@ -0,0 +1,501 @@ +// Written by Anna Araslanova +// Modified by Yixuan Qiu +// License: MIT + +#ifndef LOBPCG_SOLVER +#define LOBPCG_SOLVER + +#include +#include + +#include +#include +#include +#include +#include + +#include "../SymGEigsSolver.h" + + +namespace Spectra { + + /// + /// \ingroup EigenSolver + /// + + /// *** METHOD + /// The class represent the LOBPCG algorithm, which was invented by Andrew Knyazev + /// Theoretical background of the procedure can be found in the articles below + /// - Knyazev, A.V., 2001. Toward the optimal preconditioned eigensolver : Locally optimal block preconditioned conjugate gradient method.SIAM journal on scientific computing, 23(2), pp.517 - 541. + /// - Knyazev, A.V., Argentati, M.E., Lashuk, I. and Ovtchinnikov, E.E., 2007. Block locally optimal preconditioned eigenvalue xolvers(BLOPEX) in HYPRE and PETSc.SIAM Journal on Scientific Computing, 29(5), pp.2224 - 2239. + /// + /// *** CONDITIONS OF USE + /// Locally Optimal Block Preconditioned Conjugate Gradient(LOBPCG) is a method for finding the M smallest eigenvalues + /// and eigenvectors of a large symmetric positive definite generalized eigenvalue problem + /// \f$Ax=\lambda Bx,\f$ + /// where \f$A_{NxN}\f$ is a symmetric matrix, \f$B\f$ is symmetric and positive - definite. \f$A and B\f$ are also assumed large and sparse + /// \f$\textit{M}\f$ should be \f$\<< textit{N}\f$ (at least \f$\textit{5M} < \textit{N} \f$) + /// + /// *** ARGUMENTS + /// Eigen::SparseMatrix A; // N*N - Ax = lambda*Bx, lrage and sparse + /// Eigen::SparseMatrix X; // N*M - initial approximations to eigenvectors (random in general case) + /// Spectra::LOBPCGSolver solver(A, X); + /// *Eigen::SparseMatrix B; // N*N - Ax = lambda*Bx, sparse, positive definite + /// solver.setConstraints(B); + /// *Eigen::SparseMatrix Y; // N*K - constraints, already found eigenvectors + /// solver.setB(B); + /// *Eigen::SparseMatrix T; // N*N - preconditioner ~ A^-1 + /// solver.setPreconditioner(T); + /// + /// *** OUTCOMES + /// solver.solve(); // compute eigenpairs // void + /// solver.info(); // state of converjance // int + /// solver.residuals(); // get residuals to evaluate biases // Eigen::Matrix + /// solver.eigenvalues(); // get eigenvalues // Eigen::Matrix + /// solver.eigenvectors(); // get eigenvectors // Eigen::Matrix + /// + /// *** EXAMPLE + /// \code{.cpp} + /// #include + /// + /// // random A + /// Matrix a; + /// a = (Matrix::Random(10, 10).array() > 0.6).cast() * Matrix::Random(10, 10).array() * 5; + /// a = Matrix((a).triangularView()) + Matrix((a).triangularView()).transpose(); + /// for (int i = 0; i < 10; i++) + /// a(i, i) = i + 0.5; + /// std::cout << a << "\n"; + /// Eigen::SparseMatrix A(a.sparseView()); + /// // random X + /// Eigen::Matrix x; + /// x = Matrix::Random(10, 2).array(); + /// Eigen::SparseMatrix X(x.sparseView()); + /// // solve Ax = lambda*x + /// Spectra::LOBPCGSolver solver(A, X); + /// solver.compute(10, 1e-4); // 10 iterations, L2_tolerance = 1e-4*N + /// std::cout << "info\n" << solver.info() << std::endl; + /// std::cout << "eigenvalues\n" << solver.eigenvalues() << std::endl; + /// std::cout << "eigenvectors\n" << solver.eigenvectors() << std::endl; + /// std::cout << "residuals\n" << solver.residuals() << std::endl; + /// \endcode + /// + + template < typename Scalar = long double> + class LOBPCGSolver { + private: + + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + + typedef std::complex Complex; + typedef Eigen::Matrix ComplexMatrix; + typedef Eigen::Matrix ComplexVector; + + typedef Eigen::SparseMatrix SparseMatrix; + typedef Eigen::SparseMatrix SparseComplexMatrix; + + const int m_n; // dimension of matrix A + const int m_nev; // number of eigenvalues requested + SparseMatrix A, X; + SparseMatrix m_Y, m_B, m_preconditioner; + bool flag_with_constraints, flag_with_B, flag_with_preconditioner; + + public: + SparseMatrix m_residuals; + Matrix m_evectors; + Vector m_evalues; + int m_info; + + private: + + // B-orthonormalize matrix M + int orthogonalizeInPlace(SparseMatrix &M, SparseMatrix &B, \ + SparseMatrix &true_BM, bool has_true_BM = false) { + + SparseMatrix BM; + + if (has_true_BM == false) { + if (flag_with_B) { BM = B * M; } + else { BM = M; } + } + else { + BM = true_BM; + } + + Eigen::SimplicialLDLT chol_MBM(M.transpose() * BM); + + if (chol_MBM.info() != SUCCESSFUL) { + // LDLT decomposition fail + m_info = chol_MBM.info(); + return chol_MBM.info(); + } + + SparseComplexMatrix Upper_MBM = chol_MBM.matrixU().template cast(); + ComplexVector D_MBM_vec = chol_MBM.vectorD().template cast(); + + D_MBM_vec = D_MBM_vec.cwiseSqrt(); + + for (int i = 0; i < D_MBM_vec.rows(); i++) { + D_MBM_vec(i) = Complex(1.0, 0.0) / D_MBM_vec(i); + } + + SparseComplexMatrix D_MBM_mat(D_MBM_vec.asDiagonal()); + + SparseComplexMatrix U_inv(Upper_MBM.rows(), Upper_MBM.cols()); + U_inv.setIdentity(); + Upper_MBM.template triangularView().solveInPlace(U_inv); + + SparseComplexMatrix right_product = U_inv * D_MBM_mat; + M = M*right_product.real(); + if (flag_with_B) { true_BM = B * M; } + else { true_BM = M; } + + return SUCCESSFUL; + } + + void applyConstraintsInPlace(SparseMatrix &X, SparseMatrix&Y, \ + SparseMatrix&B) { + SparseMatrix BY; + if (flag_with_B) { BY = B * Y; } + else { BY = Y; } + + SparseMatrix YBY = Y.transpose() * BY; + SparseMatrix BYX = BY.transpose() * X; + + SparseMatrix YBY_XYX = (Matrix(YBY).bdcSvd(Eigen::ComputeThinU | Eigen::ComputeThinV).solve(Matrix(BYX))).sparseView(); + X = X - Y * YBY_XYX; + } + + /* + return + 'AB + CD' + */ + Matrix stack_4_matricies(Matrix A, Matrix B, \ + Matrix C, Matrix D) { + Matrix result(A.rows() + C.rows(), A.cols() + B.cols()); + result.topLeftCorner(A.rows(), A.cols()) = A; + result.topRightCorner(B.rows(), B.cols()) = B; + result.bottomLeftCorner(C.rows(), C.cols()) = C; + result.bottomRightCorner(D.rows(), D.cols()) = D; + return result; + } + + Matrix stack_9_matricies(Matrix A, Matrix B, Matrix C, \ + Matrix D, Matrix E, Matrix F, \ + Matrix G, Matrix H, Matrix I) { + + Matrix result(A.rows() + D.rows() + G.rows(), A.cols() + B.cols() + C.cols()); + result.block(0, 0, A.rows(), A.cols()) = A; + result.block(0, A.cols(), B.rows(), B.cols()) = B; + result.block(0, A.cols() + B.cols(), C.rows(), C.cols()) = C; + result.block(A.rows(), 0, D.rows(), D.cols()) = D; + result.block(A.rows(), A.cols(), E.rows(), E.cols()) = E; + result.block(A.rows(), A.cols() + B.cols(), F.rows(), F.cols()) = F; + result.block(A.rows() + D.rows(), 0, G.rows(), G.cols()) = G; + result.block(A.rows() + D.rows(), A.cols(), H.rows(), H.cols()) = H; + result.block(A.rows() + D.rows(), A.cols() + B.cols(), I.rows(), I.cols()) = I; + + return result; + } + + void sort_epairs(Vector &evalues, Matrix &evectors, int SelectionRule) { + + std::function cmp; + if (SelectionRule == SMALLEST_ALGE) + cmp = std::less{}; + else + cmp = std::greater{}; + + std::map epairs(cmp); + for (int i = 0; i < m_evectors.cols(); ++i) + epairs.insert(std::make_pair(evalues(i), evectors.col(i))); + + int i = 0; + for (auto& epair : epairs) { + evectors.col(i) = epair.second; + evalues(i) = epair.first; + i++; + } + } + + void removeColumns(SparseMatrix& matrix, std::vector& colToRemove) + { + // remove columns through matrix multiplication + SparseMatrix new_matrix(matrix.cols(), matrix.cols() - int(colToRemove.size())); + int iCol = 0; + std::vector> tripletList; + tripletList.reserve(matrix.cols() - int(colToRemove.size())); + + for (int iRow = 0; iRow < matrix.cols(); iRow++) { + if (std::find(colToRemove.begin(), colToRemove.end(), iRow) == colToRemove.end()) { + tripletList.push_back(Eigen::Triplet(iRow, iCol, 1)); + iCol++; + } + } + + new_matrix.setFromTriplets(tripletList.begin(), tripletList.end()); + matrix = matrix * new_matrix; + } + + int checkConvergence_getBlocksize(SparseMatrix & m_residuals, Scalar tolerance_L2, std::vector & columnsToDelete) { + // square roots from sum of squares by column + int BlockSize = m_nev; + Scalar sum, buffer; + + for (int iCol = 0; iCol < m_nev; iCol++) { + sum = 0; + for (int iRow = 0; iRow < m_n; iRow++) { + buffer = m_residuals.coeff(iRow, iCol); + sum += buffer * buffer; + } + + if (sqrt(sum) < tolerance_L2) { + BlockSize--; + columnsToDelete.push_back(iCol); + } + } + return BlockSize; + } + + + public: + + LOBPCGSolver(const SparseMatrix& A, const SparseMatrix X) : + m_n(A.rows()), + m_nev(X.cols()), + m_info(NOT_COMPUTED), + flag_with_constraints(false), + flag_with_B(false), + flag_with_preconditioner(false), + A(A), + X(X) + { + if (A.rows() != X.rows() || A.rows() != A.cols()) + throw std::invalid_argument("Wrong size"); + + //if (m_n < 5* m_nev) + // throw std::invalid_argument("The problem size is small compared to the block size. Use standard eigensolver"); + } + + void setConstraints(const SparseMatrix& Y) { + m_Y = Y; + flag_with_constraints = true; + } + + void setB(const SparseMatrix& B) { + if (B.rows() != A.rows() || B.cols() != A.cols()) + throw std::invalid_argument("Wrong size"); + m_B = B; + flag_with_B = true; + } + + void setPreconditioner(const SparseMatrix& preconditioner) { + m_preconditioner = preconditioner; + flag_with_preconditioner = true; + } + + void compute(int maxit = 10, Scalar tol_div_n = 1e-7) { + + Scalar tolerance_L2 = tol_div_n * m_n; + int BlockSize; + int max_iter = std::min(m_n, maxit); + + SparseMatrix directions, AX, AR, BX, AD, ADD, DD, BDD, BD, XAD, RAD, DAD, XBD, RBD, BR, sparse_eVecX, sparse_eVecR, sparse_eVecD, inverse_matrix; + Matrix XAR, RAR, XBR, gramA, gramB, eVecX, eVecR, eVecD; + std::vector columnsToDelete; + + if (flag_with_constraints) { + // Apply the constraints Y to X + applyConstraintsInPlace(X, m_Y, m_B); + } + + // Make initial vectors orthonormal + // implicit BX declaration + if (orthogonalizeInPlace(X, m_B, BX) != SUCCESSFUL) { + max_iter = 0; + } + + AX = A * X; + // Solve the following NxN eigenvalue problem for all N eigenvalues and -vectors: + // first approximation via a dense problem + Eigen::EigenSolver eigs(Matrix(X.transpose() * AX)); + + if (eigs.info() != SUCCESSFUL) { + m_info = eigs.info(); + max_iter = 0; + } + else { + m_evalues = eigs.eigenvalues().real(); + m_evectors = eigs.eigenvectors().real(); + sort_epairs(m_evalues, m_evectors, SMALLEST_ALGE); + sparse_eVecX = m_evectors.sparseView(); + + X = X * sparse_eVecX; + AX = AX * sparse_eVecX; + BX = BX * sparse_eVecX; + } + + + for (int iter_num = 0; iter_num < max_iter; iter_num++) { + m_residuals.resize(m_n, m_nev); + for (int i = 0; i < m_nev; i++) { + m_residuals.col(i) = AX.col(i) - m_evalues(i) * BX.col(i); + } + BlockSize = checkConvergence_getBlocksize(m_residuals, tolerance_L2, columnsToDelete); + + if (BlockSize == 0) { + m_info = SUCCESSFUL; + break; + } + + // substitution of the original active mask + if (columnsToDelete.size() > 0) { + removeColumns(m_residuals, columnsToDelete); + if (iter_num > 0) { + removeColumns(directions, columnsToDelete); + removeColumns(AD, columnsToDelete); + removeColumns(BD, columnsToDelete); + } + columnsToDelete.clear(); // for next iteration + } + + if (flag_with_preconditioner) { + // Apply the preconditioner to the residuals + m_residuals = m_preconditioner * m_residuals; + } + + if (flag_with_constraints) { + // Apply the constraints Y to residuals + applyConstraintsInPlace(m_residuals, m_Y, m_B); + } + + if (orthogonalizeInPlace(m_residuals, m_B, BR) != SUCCESSFUL) { + break; + } + AR = A * m_residuals; + + // Orthonormalize conjugate directions + if (iter_num > 0) { + if (orthogonalizeInPlace(directions, m_B, BD, true) != SUCCESSFUL) { + break; + } + AD = A * directions; + } + + // Perform the Rayleigh Ritz Procedure + XAR = Matrix(X.transpose() * AR); + RAR = Matrix(m_residuals.transpose() * AR); + XBR = Matrix(X.transpose() * BR); + + if (iter_num > 0) { + + XAD = X.transpose() * AD; + RAD = m_residuals.transpose() * AD; + DAD = directions.transpose() * AD; + XBD = X.transpose() * BD; + RBD = m_residuals.transpose() * BD; + + gramA = stack_9_matricies(m_evalues.asDiagonal(), XAR, XAD, XAR.transpose(), RAR, RAD, XAD.transpose(), RAD.transpose(), DAD.transpose()); + gramB = stack_9_matricies(Matrix::Identity(m_nev, m_nev), XBR, XBD, XBR.transpose(), Matrix::Identity(BlockSize, BlockSize), RBD, XBD.transpose(), RBD.transpose(), Matrix::Identity(BlockSize, BlockSize)); + + } + else { + gramA = stack_4_matricies(m_evalues.asDiagonal(), XAR, XAR.transpose(), RAR); + gramB = stack_4_matricies(Matrix::Identity(m_nev, m_nev), XBR, XBR.transpose(), Matrix::Identity(BlockSize, BlockSize)); + } + + //calculate the lowest/largest m eigenpairs; Solve the generalized eigenvalue problem. + DenseSymMatProd Aop(gramA); + DenseCholesky Bop(gramB); + + SymGEigsSolver, \ + DenseCholesky, GEIGS_CHOLESKY> geigs(&Aop, &Bop, m_nev, std::min(10, int(gramA.rows()) - 1)); + + geigs.init(); + int nconv = geigs.compute(); + + //Mat evecs; + if (geigs.info() == SUCCESSFUL) { + m_evalues = geigs.eigenvalues(); + m_evectors = geigs.eigenvectors(); + sort_epairs(m_evalues, m_evectors, SMALLEST_ALGE); + } + else { + // Problem With General EgenVec + m_info = geigs.info(); + break; + } + + // Compute Ritz vectors + if (iter_num > 0) { + eVecX = m_evectors.block(0, 0, m_nev, m_nev); + eVecR = m_evectors.block(m_nev, 0, BlockSize, m_nev); + eVecD = m_evectors.block(m_nev + BlockSize, 0, BlockSize, m_nev); + + sparse_eVecX = eVecX.sparseView(); + sparse_eVecR = eVecR.sparseView(); + sparse_eVecD = eVecD.sparseView(); + + DD = m_residuals * sparse_eVecR; // new conjugate directions + ADD = AR * sparse_eVecR; + BDD = BR * sparse_eVecR; + + DD = DD + directions * sparse_eVecD; + ADD = ADD + AD * sparse_eVecD; + BDD = BDD + BD * sparse_eVecD; + } + else { + eVecX = m_evectors.block(0, 0, m_nev, m_nev); + eVecR = m_evectors.block(m_nev, 0, BlockSize, m_nev); + + sparse_eVecX = eVecX.sparseView(); + sparse_eVecR = eVecR.sparseView(); + + DD = m_residuals * sparse_eVecR; + ADD = AR * sparse_eVecR; + BDD = BR * sparse_eVecR; + } + + X = X * sparse_eVecX + DD; + AX = AX * sparse_eVecX + ADD; + BX = BX * sparse_eVecX + BDD; + + directions = DD; + AD = ADD; + BD = BDD; + + } // iteration loop + + // calculate last residuals + m_residuals.resize(m_n, m_nev); + for (int i = 0; i < m_nev; i++) { + m_residuals.col(i) = AX.col(i) - m_evalues(i) * BX.col(i); + } + BlockSize = checkConvergence_getBlocksize(m_residuals, tolerance_L2, columnsToDelete); + + if (BlockSize == 0) { + m_info = SUCCESSFUL; + } + } // compute + + Vector eigenvalues() { + return m_evalues; + } + + Matrix eigenvectors() { + return m_evectors; + } + + Matrix residuals() { + return Matrix(m_residuals); + } + + int info() { return m_info; } + + }; + + +} // namespace Spectra + +#endif // LOBPCG_SOLVER diff --git a/src/external/Spectra/include/Spectra/contrib/PartialSVDSolver.h b/src/external/Spectra/include/Spectra/contrib/PartialSVDSolver.h new file mode 100644 index 00000000..dad5b400 --- /dev/null +++ b/src/external/Spectra/include/Spectra/contrib/PartialSVDSolver.h @@ -0,0 +1,203 @@ +// Copyright (C) 2018 Yixuan Qiu +// +// This Source Code Form is subject to the terms of the Mozilla +// Public License v. 2.0. If a copy of the MPL was not distributed +// with this file, You can obtain one at https://mozilla.org/MPL/2.0/. + +#ifndef PARTIAL_SVD_SOLVER_H +#define PARTIAL_SVD_SOLVER_H + +#include +#include "../SymEigsSolver.h" + + +namespace Spectra { + + +// Abstract class for matrix operation +template +class SVDMatOp +{ +public: + virtual int rows() const = 0; + virtual int cols() const = 0; + + // y_out = A' * A * x_in or y_out = A * A' * x_in + virtual void perform_op(const Scalar* x_in, Scalar* y_out) = 0; + + virtual ~SVDMatOp() {} +}; + +// Operation of a tall matrix in SVD +// We compute the eigenvalues of A' * A +// MatrixType is either Eigen::Matrix or Eigen::SparseMatrix +template +class SVDTallMatOp: public SVDMatOp +{ +private: + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + const int m_dim; + Vector m_cache; + +public: + // Constructor + SVDTallMatOp(ConstGenericMatrix& mat) : + m_mat(mat), + m_dim(std::min(mat.rows(), mat.cols())), + m_cache(mat.rows()) + {} + + // These are the rows and columns of A' * A + int rows() const { return m_dim; } + int cols() const { return m_dim; } + + // y_out = A' * A * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) + { + MapConstVec x(x_in, m_mat.cols()); + MapVec y(y_out, m_mat.cols()); + m_cache.noalias() = m_mat * x; + y.noalias() = m_mat.transpose() * m_cache; + } +}; + +// Operation of a wide matrix in SVD +// We compute the eigenvalues of A * A' +// MatrixType is either Eigen::Matrix or Eigen::SparseMatrix +template +class SVDWideMatOp: public SVDMatOp +{ +private: + typedef Eigen::Matrix Vector; + typedef Eigen::Map MapConstVec; + typedef Eigen::Map MapVec; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + const int m_dim; + Vector m_cache; + +public: + // Constructor + SVDWideMatOp(ConstGenericMatrix& mat) : + m_mat(mat), + m_dim(std::min(mat.rows(), mat.cols())), + m_cache(mat.cols()) + {} + + // These are the rows and columns of A * A' + int rows() const { return m_dim; } + int cols() const { return m_dim; } + + // y_out = A * A' * x_in + void perform_op(const Scalar* x_in, Scalar* y_out) + { + MapConstVec x(x_in, m_mat.rows()); + MapVec y(y_out, m_mat.rows()); + m_cache.noalias() = m_mat.transpose() * x; + y.noalias() = m_mat * m_cache; + } +}; + +// Partial SVD solver +// MatrixType is either Eigen::Matrix or Eigen::SparseMatrix +template < typename Scalar = double, + typename MatrixType = Eigen::Matrix > +class PartialSVDSolver +{ +private: + typedef Eigen::Matrix Matrix; + typedef Eigen::Matrix Vector; + typedef const Eigen::Ref ConstGenericMatrix; + + ConstGenericMatrix m_mat; + const int m_m; + const int m_n; + SVDMatOp* m_op; + SymEigsSolver< Scalar, LARGEST_ALGE, SVDMatOp >* m_eigs; + int m_nconv; + Matrix m_evecs; + +public: + // Constructor + PartialSVDSolver(ConstGenericMatrix& mat, int ncomp, int ncv) : + m_mat(mat), m_m(mat.rows()), m_n(mat.cols()), m_evecs(0, 0) + { + // Determine the matrix type, tall or wide + if(m_m > m_n) + { + m_op = new SVDTallMatOp(mat); + } else { + m_op = new SVDWideMatOp(mat); + } + + // Solver object + m_eigs = new SymEigsSolver< Scalar, LARGEST_ALGE, SVDMatOp >(m_op, ncomp, ncv); + } + + // Destructor + virtual ~PartialSVDSolver() + { + delete m_eigs; + delete m_op; + } + + // Computation + int compute(int maxit = 1000, Scalar tol = 1e-10) + { + m_eigs->init(); + m_nconv = m_eigs->compute(maxit, tol); + + return m_nconv; + } + + // The converged singular values + Vector singular_values() const + { + Vector svals = m_eigs->eigenvalues().cwiseSqrt(); + + return svals; + } + + // The converged left singular vectors + Matrix matrix_U(int nu) + { + if(m_evecs.cols() < 1) + { + m_evecs = m_eigs->eigenvectors(); + } + nu = std::min(nu, m_nconv); + if(m_m <= m_n) + { + return m_evecs.leftCols(nu); + } + + return m_mat * (m_evecs.leftCols(nu).array().rowwise() / m_eigs->eigenvalues().head(nu).transpose().array().sqrt()).matrix(); + } + + // The converged right singular vectors + Matrix matrix_V(int nv) + { + if(m_evecs.cols() < 1) + { + m_evecs = m_eigs->eigenvectors(); + } + nv = std::min(nv, m_nconv); + if(m_m > m_n) + { + return m_evecs.leftCols(nv); + } + + return m_mat.transpose() * (m_evecs.leftCols(nv).array().rowwise() / m_eigs->eigenvalues().head(nv).transpose().array().sqrt()).matrix(); + } +}; + + +} // namespace Spectra + +#endif // PARTIAL_SVD_SOLVER_H diff --git a/src/external/minimum_ellipsoid/bnmin_main.h b/src/external/minimum_ellipsoid/bnmin_main.h new file mode 100644 index 00000000..a9a22a93 --- /dev/null +++ b/src/external/minimum_ellipsoid/bnmin_main.h @@ -0,0 +1,87 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// This file is converted from BNMin1 (https://www.mrao.cam.ac.uk/~bn204/oof/bnmin1.html) by Apostolos Chalkis + +// Original copyright notice: + +/** + Bojan Nikolic + Initial version 2008 + + This file is part of BNMin1 and is licensed under GNU General + Public License version 2. + + \file bnmin_main.cxx + +*/ +#ifndef BNMIN_MAIN_H +#define BNMIN_MAIN_H + +#include +#include + +#include + +//#include "bnmin_main1.h" +//#include "config.h" + +//namespace Minim { + + inline const char * version(void) + { + //return PACKAGE_VERSION; + return "11"; + } + + class BaseErr: + public std::runtime_error + { + public: + BaseErr(const std::string &s): + std::runtime_error(s) + { + } + + }; + + class NParsErr: + public BaseErr + { + public: + NParsErr(const std::string &fname, + size_t expected, + size_t received): + BaseErr( (boost::format("In function %s expected %i but received %i pars ") + % fname + % expected + % received).str()) + { + } + + + }; + + /*BaseErr::BaseErr(const std::string &s): + std::runtime_error(s) + { + } + + NParsErr::NParsErr(const std::string &fname, + size_t expected, + size_t received): + BaseErr( (boost::format("In function %s expected %i but received %i pars ") + % fname + % expected + % received).str()) + { + }*/ + + +#endif + +//} + + diff --git a/src/external/minimum_ellipsoid/khach.h b/src/external/minimum_ellipsoid/khach.h new file mode 100644 index 00000000..88492b8e --- /dev/null +++ b/src/external/minimum_ellipsoid/khach.h @@ -0,0 +1,220 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// This file is converted from BNMin1 (https://www.mrao.cam.ac.uk/~bn204/oof/bnmin1.html) by Apostolos Chalkis + +// Original copyright notice: + +/** + Bojan Nikolic + Initial version 2010 + + This file is part of BNMin1 and is licensed under GNU General + Public License version 2 + + \file ellipsoids.cxx + + Computation and use of ellipsoids releated to sets of points +*/ +#ifndef KHACH_H +#define KHACH_H + +#include +#include +#include + +//#include "khach1.h" +//#include "mcpoint1.h" +#include "mcpoint.h" +//#include "bnmin_main1.h" +//#include "bnmin_main2.h" + +//#include "../bnmin_main.hxx" + +//namespace Minim { + + template + using MTT = Eigen::Matrix; + + template + using VTT = Eigen::Matrix; + + struct KhachiyanEllipsoid + { + Eigen::Matrix Q; + Eigen::Matrix c; + }; + + template + inline bool is_nan(const Eigen::MatrixBase& x) + { + return ((x.array() == x.array())).all(); + } + + template + bool InvertMatrix(const MTT &input, + MTT &inverse) + { + inverse = input.inverse(); + return !is_nan(inverse); + } + + + inline void InvertLP(const MTT &Lambdap, + MTT &LpInv) + { + bool res = InvertMatrix(Lambdap, LpInv); + if (not res) + { + // throw an error of your choice here! + // throw MatrixErr("Could not invert matrix: ", + // Lambdap); + } + } + + inline void Lift(const MTT &A, MTT &Ap) + { + Ap.resize(A.rows()+1, A.cols()); + Ap.topLeftCorner(A.rows(), A.cols()) = A; + Ap.row(Ap.rows()-1).setConstant(1.0); + } + + inline void genDiag(const VTT &p, MTT &res) + { + res.setZero(p.size(), p.size()); + + for(size_t i=0; i &Ap, + const VTT &p, + MTT &Lambdap) + { + + MTT dp(p.size(), p.size()); + genDiag(p, dp); + + dp = dp * Ap.transpose(); + Lambdap.noalias() = Ap * dp; + } + + inline double KhachiyanIter(const MTT &Ap, VTT &p) + { + /// Dimensionality of the problem + const size_t d = Ap.rows()-1; + + MTT Lp; + MTT M; + KaLambda(Ap, p, Lp); + MTT ILp(Lp.rows(), Lp.cols()); + InvertLP(Lp, ILp); + M.noalias() = ILp * Ap; + M = Ap.transpose() * M; + + double maxval=0; + size_t maxi=0; + for(size_t i=0; i maxval) + { + maxval=M(i,i); + maxi=i; + } + } + const double step_size=(maxval -d - 1)/((d+1)*(maxval-1)); + VTT newp = p*(1-step_size); + newp(maxi) += step_size; + + const double err= (newp-p).norm(); + p = newp; + return err; + + } + + inline void KaInvertDual(const MTT &A, + const VTT &p, + MTT &Q, + VTT &c) + { + const size_t d = A.rows(); + MTT dp(p.size(), p.size()); + genDiag(p, dp); + + MTT PN; + PN.noalias() = dp * A.transpose(); + PN = A * PN; + + VTT M2; + M2.noalias() = A * p; + + MTT M3; + M3.noalias() = M2 * M2.transpose(); + + MTT invert(PN.rows(), PN.cols()); + InvertLP(PN- M3, invert); + Q.noalias() = (invert/d); + c.noalias() = A * p; + + } + + inline double KhachiyanAlgo(const MTT &A, + double eps, + size_t maxiter, + MTT &Q, + VTT &c) + { + VTT p(A.cols()); + p.setConstant(1.0/A.cols()); + + MTT Ap; + Lift(A, Ap); + + double ceps=eps*2; + for (size_t i=0; ieps; ++i) + { + ceps=KhachiyanIter(Ap, p); + } + + KaInvertDual(A, p, Q, c); + + return ceps; + + + } + + inline double KhachiyanAlgo(const std::set &ss, + double eps, + size_t maxiter, + KhachiyanEllipsoid &res) + { + const size_t d=ss.begin()->p.size(); + MTT A(d, ss.size()); + + size_t j=0; + for (std::set::const_iterator i=ss.begin(); + i != ss.end(); + ++i) + { + for(size_t k=0; k p[k]; + ++j; + } + + MTT Q(d,d); + VTT c(d); + + const double ceps=KhachiyanAlgo(A, eps, maxiter, + Q, c); + res.Q=Q; + res.c=c; + return ceps; + } + +#endif + +//} diff --git a/src/external/minimum_ellipsoid/mcpoint.h b/src/external/minimum_ellipsoid/mcpoint.h new file mode 100644 index 00000000..65176143 --- /dev/null +++ b/src/external/minimum_ellipsoid/mcpoint.h @@ -0,0 +1,477 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// This file is converted from BNMin1 (https://www.mrao.cam.ac.uk/~bn204/oof/bnmin1.html) by Apostolos Chalkis + +// Original copyright notice: + +/** + Bojan Nikolic + Initial version 2009 + + This file is part of BNMin1 and is licensed under GNU General + Public License version 2 + + \file mcpoint.cxx +*/ +#ifndef MCPOINT_H +#define MCPOINT_H + +#include +#include +#include + +#include +#include + +//exclude gsl library Apostolos Chalkis +//#include +//#include + +//#include "mcpoint1.h" +//#include "mcpoint2.h" +#include "bnmin_main.h" +//#include "bnmin_main2.h" + +//namespace Minim { + struct MCPoint + { + /// The actual parameters + std::vector p; + /// Log-likelihood of this point + double ll; + /// A vector to store derived quantities at sample of the + /// distribution + std::vector fval; + + /// Default constructor allowed, fill in the data later + MCPoint(void): + p(0), + ll(-9999), + fval(0) + { + } + + /** \Construct with supplied position vector + */ + MCPoint(const std::vector &p): + p(p), + ll(-9999), + fval(0) + { + } + + /** \brief The parameter vector has n values + */ + MCPoint(size_t np): + p(np), + ll(-9999), + fval(0) + { + } + + MCPoint(const MCPoint &other): + p(other.p), + ll(other.ll), + fval(other.fval) + { + } + + MCPoint & operator=(const MCPoint &other) + { + p=other.p; + ll=other.ll; + fval=other.fval; + return *this; + } + + + }; + + inline bool operator< (const MCPoint &a, const MCPoint &b) + { + return a.ll < b.ll; + } + + struct WPPoint: + public MCPoint + { + /** \brief Weight factor + */ + double w; + + WPPoint(void): + w(0.0) + { + } + + WPPoint(const std::vector &p, + double w): + MCPoint(p), + w(w) + { + } + + /** \brief Construct from MCPoint and a supplied weight + */ + WPPoint(const MCPoint &mp, + double w): + MCPoint(mp), + w(w) + { + } + + }; + + /* + MCPoint::MCPoint(void): + p(0), + ll(-9999), + fval(0) + { + } + + MCPoint::MCPoint(const std::vector &p): + p(p), + ll(-9999), + fval(0) + { + } + + MCPoint::MCPoint(size_t np): + p(np), + ll(-9999), + fval(0) + { + } + + MCPoint::MCPoint(const MCPoint &other): + p(other.p), + ll(other.ll), + fval(other.fval) + { + } + + MCPoint &MCPoint::operator=(const MCPoint &other) + { + p=other.p; + ll=other.ll; + fval=other.fval; + return *this; + }*/ + + + inline void moment1(const std::list &l, + std::vector &res) + { + const size_t n=l.begin()->p.size(); + res=std::vector(n, 0.0); + for(std::list::const_iterator i=l.begin(); + i!= l.end(); + ++i) + { + for (size_t j=0; jp[j] * i->w * exp(- i->ll)); + } + } + } + + inline void moment1(const std::list &l, + double Z, + std::vector &res) + { + moment1(l,res); + for(size_t j=0; j &l, + const std::vector &m1, + std::vector &res) + { + const size_t n=m1.size(); + res=std::vector(n, 0.0); + for(std::list::const_iterator i=l.begin(); + i!= l.end(); + ++i) + { + for (size_t j=0; jp[j]-m1[j],2.0) * i->w * exp(- i->ll)); + } + } + } + + inline void moment2(const std::list &l, + const std::vector &m1, + double Z, + std::vector &res) + { + moment2(l, m1, res); + for(size_t j=0; j &s, + std::vector &res) + { + const size_t n=s.begin()->p.size(); + res=std::vector(n, 0.0); + + size_t N=0; + for(std::set::const_iterator i=s.begin(); + i!= s.end(); + ++i) + { + if(i->p.size() != n) + { + throw NParsErr("moment1", n, i->p.size()); + } + for (size_t j=0; jp[j]); + } + ++N; + } + + for(size_t j=0; j &s, + const std::vector &m1, + std::vector &res) + { + const size_t n=m1.size(); + res=std::vector(n, 0.0); + + size_t N=0; + for(std::set::const_iterator i=s.begin(); + i!= s.end(); + ++i) + { + for (size_t j=0; jp[j]-m1[j], 2); + } + ++N; + } + + for(size_t j=0; j &s, + const std::vector &m1, + std::vector &res) + { + const size_t n=m1.size(); + res=std::vector(n*n, 0.0); + + size_t N=0; + for(std::set::const_iterator i=s.begin(); + i!= s.end(); + ++i) + { + for (size_t j=0; jp[j]-m1[j])*(i->p[k]-m1[k]); + } + } + ++N; + } + + for(size_t j=0; j &s, + std::vector &res) + { + std::vector m1; + moment1(s, m1); + omoment2(s, m1, res); + } + + + inline void StdDev(const std::set &s, + std::vector &res) + { + std::vector m1, m2; + moment1(s, m1); + moment2(s, m1, m2); + res.resize(m2.size()); + for(size_t j=0; j &cv, + std::vector &eigvals, + std::vector &eigvects) + { + const size_t n=sqrt(cv.size()); + gsl_matrix_view m + = gsl_matrix_view_array (const_cast(&cv[0]), n, n); + + gsl_vector *eval = gsl_vector_alloc (n); + gsl_matrix *evec = gsl_matrix_alloc (n, n); + + gsl_eigen_symmv_workspace * w = + gsl_eigen_symmv_alloc (n); + + gsl_eigen_symmv (&m.matrix, + eval, + evec, + w); + + gsl_eigen_symmv_free (w); + + gsl_eigen_symmv_sort (eval, + evec, + GSL_EIGEN_SORT_ABS_ASC); + + eigvals.resize(n); + eigvects.resize(n*n); + for(size_t j=0; j &l, + double Z, + const std::vector &low, + const std::vector &high, + size_t nbins, + std::vector &res) + { + const size_t ndim=low.size(); + + //res.resize(pow(nbins, static_cast(ndim))); + res.resize( static_cast( pow(static_cast(nbins), static_cast(ndim)) ) ); + std::fill(res.begin(), res.end(), 0.0); + + + std::vector deltas(ndim); + for(size_t i=0; i::const_iterator i=l.begin(); + i!= l.end(); + ++i) + { + bool inside=true; + size_t k=0; + for (size_t j=0; jp[j]-low[j])/deltas[j]); + if (dimi >= 0 and dimi < (int)nbins) + { + k+= dimi * static_cast( pow(static_cast(nbins), static_cast(ndim-j-1)) ); + } + else + { + inside=false; + } + } + if (inside) + { + res[k]+= i->w * exp(- i->ll); + } + } + } + + + inline void marginHist(const std::list &l, + size_t pi, + double Z, + double low, + double high, + size_t nbins, + std::vector &res) + { + res.resize(nbins); + std::fill(res.begin(), res.end(), + 0.0); + + const double d=(high-low)/nbins; + for(std::list::const_iterator i=l.begin(); + i!= l.end(); + ++i) + { + int k=int((i->p[pi]-low)/d); + if (k > 0 and k < (int)nbins) + { + res[k]+= i->w * exp(- i->ll); + } + } + + for(size_t i=0; i &l, + double Z, + size_t i, + double ilow, + double ihigh, + size_t j, + double jlow, + double jhigh, + size_t nbins, + std::vector &res) + { + // Two dimensions only + res.resize( static_cast( pow(static_cast(nbins), static_cast(2)) ) ); + std::fill(res.begin(), res.end(), + 0.0); + const double idelta=(ihigh-ilow)/nbins; + const double jdelta=(jhigh-jlow)/nbins; + + for(std::list::const_iterator p=l.begin(); + p!= l.end(); + ++p) + { + + int dimi = int((p->p[i]-ilow)/idelta); + int dimj = int((p->p[j]-jlow)/jdelta); + + if (dimi >= 0 and dimi<((int)nbins) and dimj >= 0 and dimj < ((int)nbins)) + { + const size_t k= dimi*nbins + dimj; + res[k]+= p->w * exp(- p->ll); + } + + } + } +//} + +#endif diff --git a/src/volesti b/src/volesti deleted file mode 160000 index 5bf9188f..00000000 --- a/src/volesti +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 5bf9188fb11fddab17de98fabc505e993216ca5f diff --git a/src/volesti/include/SDPAFormatManager.h b/src/volesti/include/SDPAFormatManager.h new file mode 100644 index 00000000..c38e2d59 --- /dev/null +++ b/src/volesti/include/SDPAFormatManager.h @@ -0,0 +1,271 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_SDPA_FORMAT_MANAGER_H +#define VOLESTI_SDPA_FORMAT_MANAGER_H + + +#include "convex_bodies/spectrahedra/spectrahedron.h" + +#include +#include + + +/// Reads/writes files according to the SDPA format for sdps. +/// Currently supported Format: +/// +/// +/// +/// +/// +/// For example: +/// 2 +/// 3 +/// 1 1 +/// +/// +/// \tparam NT Numerical Type +template +class SdpaFormatManager { +private: + typedef std::string::iterator string_it; + typedef std::list listVector; + + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + /// Return the first non white space/tab character and advance the iterator one position + /// @param[in, out] it Current position + /// @param[in] end End of string + /// @return First non white space/tab character + char consumeSymbol(string_it &at, string_it & end) { + while (at != end) { + if (*at != ' ' && *at != '\t') { + char c = *at; + at++; + return c; + } + + at++; + } + + return '\0'; + } + + + /// Determine if current line is a comment + /// @param[in] line The current line + /// @return true if line is a comment, false otherwise + bool isCommentLine(std::string & line) { + string_it at = line.begin(); + string_it end = line.end(); + char c = consumeSymbol(at, end); + return c == '"' || c == '*'; + } + + + /// Get an integer from the string + /// \param[in] string + /// \return an integer + int fetchNumber(std::string &string) { + std::stringstream stream(string); + int num; + stream >> num; + return num; + } + + + /// Read a vector of the form {val1, val2, ..., valn} + /// @param string Contains the vector + /// @return a list with the n numbers + listVector readVector(std::string &string) { + std::stringstream stream(string); + listVector vector; + NT value; + + while (stream >> value) { + vector.push_back(value); + } + + return vector; + } + +public: + + /// Reads an SDPA format file + /// \param[in] is An open stram pointing to the file + /// \param[out] matrices the matrices A0, A1, A2, ..., An + /// \param[out] objectiveFunction The objective function of the sdp + void loadSDPAFormatFile(std::ifstream &is, std::vector &matrices, VT &objectiveFunction) { + std::string line; + std::string::size_type sz; + + std::getline(is, line, '\n'); + + //skip comments + while (isCommentLine(line)) { + std::getline(is, line, '\n'); + } + + //read variables number + int variablesNum = fetchNumber(line); + + if (std::getline(is, line, '\n').eof()) + throw std::runtime_error("Unexpected end of file"); + + //read number of blocks + int blocksNum = fetchNumber(line); + + if (std::getline(is, line, '\n').eof()) + throw std::runtime_error("Unexpected end of file"); + + //read block structure vector + listVector blockStructure = readVector(line); + + if (blockStructure.size() != blocksNum) + throw std::runtime_error("Wrong number of blocks"); + + if (std::getline(is, line, '\n').eof()) + throw std::runtime_error("Unexpected end of file"); + + //read constant vector + listVector constantVector = readVector(line); + + while (constantVector.size() < variablesNum) { + if (std::getline(is, line, '\n').eof()) + throw std::runtime_error("Unexpected end of file"); + + listVector t = readVector(line); + constantVector.insert(std::end(constantVector), std::begin(t), std::end(t)); + } + + matrices = std::vector(variablesNum + 1); + int matrixDim = 0; + for (auto x : blockStructure) + matrixDim += std::abs((int) x); + + //read constraint matrices + for (int atMatrix = 0; atMatrix < matrices.size(); atMatrix++) { + MT matrix; + matrix.setZero(matrixDim, matrixDim); + + int offset = 0; + + for (auto blockSize : blockStructure) { + + if (blockSize > 0) { //read a block blockSize x blockSize + int at = 0; + int i = 0, j = 0; + + while (at < blockSize * blockSize) { + if (std::getline(is, line, '\n').eof()) + throw 1; + + listVector vec = readVector(line); + + for (double value : vec) { + matrix(offset + i, offset + j) = value; + at++; + if (at % (int) blockSize == 0) { // new row + i++; + j = 0; + } else { //new column + j++; + } + } + } /* while (at0, I want it <0 + if (atMatrix == 0) //F0 has - before it in SDPA format, the rest have + + matrices[atMatrix] = matrix; + else + matrices[atMatrix] = -1 * matrix; + } + + // return lmi and objective function + objectiveFunction.setZero(variablesNum); + int at = 0; + + for (auto value : constantVector) + objectiveFunction(at++) = value; + } + + /// Create a SDPA format file + /// \param[in] os Open stream to file + /// \param[in] matrices The matrices A0, ..., An + /// \param[in] objectiveFunction The objective function of the sdp + void writeSDPAFormatFile(std::ostream &os, std::vector const & matrices, VT const & objectiveFunction) { + int dim = matrices.size() - 1; + MT A0 = matrices[0]; + + os << dim << "\n"; + os << 1 << "\n"; + os << A0.rows() << "\n"; + + os << objectiveFunction.transpose() << "\n"; + + for (int i = 0; i < A0.rows(); i++) + os << A0.row(i) << "\n"; + + for (int at=1 ; at + void loadSDPAFormatFile(std::ifstream &is, Spectrahedron &spectrahedron, Point &objectiveFunction) { + std::vector matrices; + VT coeffs; + loadSDPAFormatFile(is, matrices, coeffs); + LMI lmi(matrices); + spectrahedron = Spectrahedron(lmi); + objectiveFunction = Point(coeffs); + } + + + /// Write a spectrahedron and a vector (objective function) to a SDPA format output file + /// \tparam Point + /// \param[in] is opened stream to output file + /// \param[in] spectrahedron + /// \param[in] objectiveFunction + template + void writeSDPAFormatFile(std::ostream &os, Spectrahedron const & spectrahedron, Point const & objectiveFunction) { + writeSDPAFormatFile(os, spectrahedron.getLMI().getMatrices(), objectiveFunction.getCoefficients()); + } +}; + + +#endif //VOLESTI_SDPA_FORMAT_MANAGER_H + diff --git a/src/volesti/include/cartesian_geom/cartesian_kernel.h b/src/volesti/include/cartesian_geom/cartesian_kernel.h new file mode 100644 index 00000000..3e1049b2 --- /dev/null +++ b/src/volesti/include/cartesian_geom/cartesian_kernel.h @@ -0,0 +1,39 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// VolEsti is free software: you can redistribute it and/or modify it +// under the terms of the GNU Lesser General Public License as published by +// the Free Software Foundation, either version 3 of the License, or (at +// your option) any later version. +// +// VolEsti is distributed in the hope that it will be useful, but WITHOUT +// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +// for more details. +// +// See the file COPYING.LESSER for the text of the GNU Lesser General +// Public License. If you did not receive this file along with HeaDDaCHe, +// see . + + +#ifndef CARTESIAN_KERNEL_H +#define CARTESIAN_KERNEL_H + +#include "point.h" + +/// This class represents a cartesian kernel parameterized by a numerical type e.g. double +/// \tparam K Numerical Type +template +class Cartesian +{ +public: + typedef Cartesian Self; + typedef K FT; + typedef point Point; + +}; + +#endif diff --git a/src/volesti/include/cartesian_geom/point.h b/src/volesti/include/cartesian_geom/point.h new file mode 100644 index 00000000..b8c8e32b --- /dev/null +++ b/src/volesti/include/cartesian_geom/point.h @@ -0,0 +1,238 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018-2020 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef POINT_H +#define POINT_H + +#include +#include + +/// This class manipulates a point parameterized by a number type e.g. double +/// \tparam K Numerical Type +template +class point +{ +private: + unsigned int d; + + Eigen::Matrix coeffs; + typedef typename std::vector::iterator iter; +public: + typedef Eigen::Matrix Coeff; + typedef typename K::FT FT; + + point() {} + + point(const unsigned int dim) + { + d = dim; + coeffs.setZero(d); + } + + point(const unsigned int dim, iter begin, iter endit) + { + d = dim; + coeffs.resize(d); + int i = 0; + + for (iter it=begin ; it != endit ; it++) + coeffs(i++) = *it; + } + + point(const Coeff& coeffs) + { + d = coeffs.rows(); + this->coeffs = coeffs; + } + + point(const unsigned int dim, std::vector cofs) + { + d = dim; + coeffs.resize(d); + iter it = cofs.begin(); + int i=0; + + for (; it != cofs.end(); it++,i++) + coeffs(i) = *it; + + } + + void add(const Coeff& coeffs) + { + this->coeffs += coeffs; + } + + const Coeff& getCoefficients() const + { + return coeffs; + } + + int dimension() const + { + return d; + } + + void set_dimension(const unsigned int dim) + { + d = dim; + coeffs.setZero(d); + } + + void set_coord(const unsigned int i, FT coord) + { + coeffs(i) = coord; + } + + void set_coeffs (const Coeff& coeffs2) { + d = coeffs2.rows(); + coeffs = coeffs2; + } + + void set_to_origin() { + coeffs.setZero(d); + } + + FT operator[] (const unsigned int i) const + { + return coeffs(i); + } + + FT* pointerToData() + { + return coeffs.data(); + } + + FT sum() const { + return coeffs.sum(); + } + + void operator+= (const point& p) + { + coeffs += p.getCoefficients(); + } + + void operator+= (const Coeff& coeffs) + { + this->coeffs += coeffs; + } + + void operator-= (const point& p) + { + coeffs -= p.getCoefficients(); + } + + void operator-= (const Coeff& coeffs) + { + this->coeffs -= coeffs; + } + + void operator= (const Coeff& coeffs) + { + this->coeffs = coeffs; + d = coeffs.rows(); + } + + //TODO: avoid point construction in operators +,-,* + point operator+ (const point& p) const + { + point temp; + temp.d = d; + temp.coeffs = coeffs + p.getCoefficients(); + return temp; + } + + point operator- (const point& p) const + { + point temp; + temp.d = d; + temp.coeffs = coeffs - p.getCoefficients(); + return temp; + } + + point operator* (const FT k) const + { + point temp; + temp.d = d; + temp.coeffs = coeffs * k; + return temp; + } + + void operator*= (const FT k) + { + coeffs *= k; + } + + void operator/= (const FT k) + { + coeffs /= k; + } + + bool operator== (point& p) const + { + int i=0; + const Coeff & coeffs = p.getCoefficients(); + + /* degree of approximation in + "The art of computer programming" (vol II), p. 234, Donald. E. Knuth. */ + FT e = 0.00000000001; + for (i=0 ; icoeffs(i) - coeffs(i)) > e *std::abs(this->coeffs(i)) || + std::abs(this->coeffs(i) - coeffs(i)) > e *std::abs(coeffs(i))) + return false; + } + + return true; + } + + FT distance(point const & p) { + return (this->coeffs - p.coeffs).norm(); + } + + FT dot(const point& p) const + { + return coeffs.dot(p.getCoefficients()); + } + + FT dot(const Coeff& coeffs) const + { + return this->coeffs.dot(coeffs); + } + + FT squared_length() const { + FT lsq = length(); + return lsq * lsq; + } + + FT length() const { + return coeffs.norm(); + } + + void print() const + { + for(unsigned int i=0; i +point operator* (const typename K::FT& k, point const& p) +{ + return p * k; +} + +#endif diff --git a/src/volesti/include/convex_bodies/ball.h b/src/volesti/include/convex_bodies/ball.h new file mode 100644 index 00000000..51dc2647 --- /dev/null +++ b/src/volesti/include/convex_bodies/ball.h @@ -0,0 +1,176 @@ +// volesti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef BALL_H +#define BALL_H + +#include + +/// This class represents a ball parameterized by a point type +/// \tparam Point Point Type +template +class Ball{ +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef typename std::vector::iterator viterator; + typedef Eigen::Matrix VT; + typedef Eigen::Matrix MT; + + Ball() {} + + Ball(Point cc, NT RR) : c(cc), R(RR) {} + + std::pair InnerBall() const + { + return std::pair(c, R); + } + + Point center() const + { + return c; + } + + NT squared_radius() const + { + return R; + } + + NT radius() const + { + return std::sqrt(R); + } + + int dimension() const + { + return c.dimension(); + } + + int is_in(Point const& p) const + { + if (p.squared_length() <= R) + return -1; + else return 0; + } + + std::pair line_intersect(Point const& r, Point const& v) const + { + + NT vrc(0), v2(0), rc2(0); + + vrc = v.dot(r); + v2 = v.dot(v); + rc2 = r.dot(r); + + NT disc_sqrt = std::sqrt(std::pow(vrc,2) - v2 * (rc2 - R)); + return std::pair ((NT(-1)*vrc + disc_sqrt)/v2, + (NT(-1)*vrc - disc_sqrt)/v2); + } + + std::pair line_intersect(Point const& r, + Point const& v, + const VT &Ar, + const VT &Av) const + { + return line_intersect(r, v); + } + + + std::pair line_intersect(Point const& r, + Point const& v, + const VT &Ar, + const VT &Av, + NT &lambda_prev) const + { + return line_intersect(r, v); + } + + std::pair line_positive_intersect(Point const& r, + Point const& v) const + { + return std::pair(line_intersect(r, v).first, 0); + } + + std::pair line_positive_intersect(Point const& r, + Point const& v, + const VT &Ar, + const VT &Av) const + { + return line_positive_intersect(r, v); + } + + std::pair line_positive_intersect(Point const& r, + Point const& v, + const VT &Ar, + const VT &Av, + NT &lambda_prev) const + { + return line_positive_intersect(r, v); + } + + std::pair line_intersect_coord(Point const& r, + unsigned int const& rand_coord) const + { + + NT vrc = r[rand_coord]; + NT rc2(R); + rc2 -= r.dot(r); + + + NT disc_sqrt = std::sqrt(std::pow(vrc,2) + rc2); + return std::pair (NT(-1)*vrc + disc_sqrt, NT(-1)*vrc - disc_sqrt); + + } + + std::pair line_intersect_coord(Point const& r, + unsigned int const& rand_coord, + const VT &lamdas) const + { + return line_intersect_coord(r, rand_coord); + } + + std::pair line_intersect_coord(Point const& r, + Point const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + const VT &lamdas) const + { + return line_intersect_coord(r, rand_coord); + } + + int num_of_hyperplanes() const + { + return 0; + } + + void compute_reflection (Point& v, Point const& p) const + { + Point s = p; + s *= (1.0 / std::sqrt(s.squared_length())); + s *= (-2.0 * v.dot(s)); + v += s; + } + + template + void compute_reflection (Point &v, Point const& p, update_parameters ¶ms) const { + + params.ball_inner_norm = p.length(); + params.inner_vi_ak = v.dot(p) / params.ball_inner_norm; + v += (p * (-2.0 * params.inner_vi_ak * (1.0 / params.ball_inner_norm))); + } + +private: + Point c; //center + NT R; //SQUARED radius !!! +}; + + +#endif diff --git a/src/volesti/include/convex_bodies/ballintersectconvex.h b/src/volesti/include/convex_bodies/ballintersectconvex.h new file mode 100644 index 00000000..b61ae532 --- /dev/null +++ b/src/volesti/include/convex_bodies/ballintersectconvex.h @@ -0,0 +1,401 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef BALLINTERSECTCONVEX_H +#define BALLINTERSECTCONVEX_H + +/// This class represents a polytope intersected with a ball +/// \tparam Polytope Polytope Type +/// \tparam CBall Ball Type +template +class BallIntersectPolytope { +private: + Polytope P; + CBall B; +public: + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename CBall::NT NT; + typedef typename CBall::PointType PointType; + + BallIntersectPolytope() {} + + BallIntersectPolytope(Polytope& PP, CBall &BB) : P(PP), B(BB) {}; + + Polytope first() const { return P; } + CBall second() const { return B; } + + std::pair InnerBall() const + { + return P.InnerBall(); + } + + MT get_mat() const { + return P.get_mat(); + } + + MT get_T() const { + return P.get_mat(); + } + + MT get_vec() const { + return P.get_vec(); + } + + int is_in(PointType const& p) const + { + if (B.is_in(p)==-1) + return P.is_in(p); + return 0; + } + + int num_of_hyperplanes() const { + return P.num_of_hyperplanes(); + } + + unsigned int dimension() const { + return P.dimension(); + } + + NT radius() const { + return B.radius(); + } + + std::pair line_intersect(PointType const& r, PointType const& v) const + { + + std::pair polypair = P.line_intersect(r, v); + std::pair ballpair = B.line_intersect(r, v); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT &Ar, + VT &Av) const + { + std::pair polypair = P.line_intersect(r, v, Ar, Av); + std::pair ballpair = B.line_intersect(r, v); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT &Ar, + VT &Av, + NT &lambda_prev) const + { + std::pair polypair = P.line_intersect(r, v, Ar, Av, lambda_prev); + std::pair ballpair = B.line_intersect(r, v); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT &Ar, + VT &Av) + { + std::pair polypair = P.line_positive_intersect(r, v, Ar, Av); + std::pair ball_lambda = B.line_positive_intersect(r, v); + int facet = (polypair.first < ball_lambda.first) ? polypair.second : P.num_of_hyperplanes(); + + return std::pair(std::min(polypair.first, ball_lambda.first), facet); + } + + + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT &Ar, + VT &Av, + NT &lambda_prev) + { + std::pair polypair = P.line_positive_intersect(r, v, Ar, Av, lambda_prev); + std::pair ball_lambda = B.line_positive_intersect(r, v); + int facet = (polypair.first < ball_lambda.first) ? polypair.second : P.num_of_hyperplanes(); + + return std::pair(std::min(polypair.first, ball_lambda.first), facet); + } + + //---------------------accelerated billiard---------------------// + template + std::pair line_first_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + update_parameters& params) + { + std::pair polypair = P.line_first_positive_intersect(r, v, Ar, Av, params); + std::pair ball_lambda = B.line_positive_intersect(r, v); + + params.hit_ball = (polypair.first < ball_lambda.first) ? false : true; + int facet = params.hit_ball ? P.num_of_hyperplanes() : polypair.second; + params.facet_prev = polypair.second; + + return std::pair(std::min(polypair.first, ball_lambda.first), facet); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters& params) + { + std::pair polypair = P.line_positive_intersect(r, v, Ar, Av, lambda_prev, AA, params); + std::pair ball_lambda = B.line_positive_intersect(r, v); + + params.hit_ball = (polypair.first < ball_lambda.first) ? false : true; + int facet = params.hit_ball ? P.num_of_hyperplanes() : polypair.second; + params.facet_prev = polypair.second; + + return std::pair(std::min(polypair.first, ball_lambda.first), facet); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters& params) + { + std::pair polypair = P.line_positive_intersect(r, v, Ar, Av, lambda_prev, params); + std::pair ball_lambda = B.line_positive_intersect(r, v); + + params.hit_ball = (polypair.first < ball_lambda.first) ? false : true; + int facet = params.hit_ball ? P.num_of_hyperplanes() : polypair.second; + params.facet_prev = polypair.second; + + return std::pair(std::min(polypair.first, ball_lambda.first), facet); + } +//-------------------------------------------------------------------------// + + //First coordinate ray shooting intersecting convex body + std::pair line_intersect_coord(PointType const& r, + unsigned int const& rand_coord, + VT &lamdas) const + { + + std::pair polypair = P.line_intersect_coord(r, rand_coord, lamdas); + std::pair ballpair = B.line_intersect_coord(r, rand_coord); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + //Not the first coordinate ray shooting intersecting convex body + std::pair line_intersect_coord(PointType const& r, + PointType const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + VT &lamdas) const + { + std::pair polypair = P.line_intersect_coord(r, r_prev, rand_coord, + rand_coord_prev, lamdas); + std::pair ballpair = B.line_intersect_coord(r, rand_coord); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + std::pair query_dual(PointType const& p, + unsigned int const& rand_coord) + { + std::pair polypair = P.query_dual(p, rand_coord); + std::pair ballpair = B.line_intersect_coord(p, rand_coord); + return std::pair(std::min(polypair.first, ballpair.first), + std::max(polypair.second, ballpair.second)); + } + + void compute_reflection (PointType& v, PointType const& p, int &facet) + { + + if (facet == P.num_of_hyperplanes()) { + B.compute_reflection(v, p); + } else { + P.compute_reflection(v, p, facet); + } + + } + + template + void compute_reflection (PointType &v, PointType const& p, update_parameters ¶ms) + { + if (params.hit_ball) { + B.compute_reflection(v, p, params); + } else { + P.compute_reflection(v, p, params); + } + } + +}; + + +/* EXPERIMENTAL +template +class PolytopeIntersectEllipsoid { +private: + T1 P; + T2 E; + typedef typename T2::K K; +public: + PolytopeIntersectEllipsoid(T1 &Pin, T2 &Ein) : P(Pin), E(Ein) {}; + + T1 first() { return P; } + T2 second() { return E; } + + int is_in(Point p){ + //std::cout << "calling is in"< line_intersect(Point r, + Point v){ + + std::pair polypair = P.line_intersect(r,v); + std::pair returnpair; + std::pair ellpair; + bool ellinter=false; + + //check the first intersection point if it is inside ball + if(E.is_in(polypair.first)){ + returnpair.first = polypair.first; + }else{ + ellinter=true; + //compute the intersection with ball + ellpair = E.line_intersect(r,v); + returnpair.first = ellpair.first; + } + //check the second intersection point + if(E.is_in(polypair.second)){ + returnpair.second = polypair.second; + }else{ + if(ellinter) //if the intersection with ball is already computed + returnpair.second = ellpair.second; + else returnpair.second = (E.line_intersect(r,v)).second; + } + return returnpair; + } + + std::pair line_intersect_coord(Point &r, + Point &r_prev, + int rand_coord, + int rand_coord_prev, + std::vector &lamdas, + bool init + ){ + + std::pair polypair = P.line_intersect_coord(r,r_prev,rand_coord,rand_coord_prev,lamdas,init); + std::pair ellpair = E.line_intersect_coord(r,rand_coord); + return std::pair (std::min(polypair.first,ellpair.first), + std::max(polypair.second,ellpair.second)); + } + +}; + + +template +class BallPolyIntersectEll { +private: + T1 BP; + T2 E; + typedef typename T2::K K; +public: + BallPolyIntersectEll(T1 &BPin, T2 &Ein) : BP(BPin), E(Ein) {}; + + T1 first() { return BP; } + T2 second() { return E; } + + int is_in(Point p){ + //std::cout << "calling is in"< line_intersect(Point r, + Point v){ + + std::pair Bpolypair = BP.line_intersect(r,v); + std::pair returnpair; + std::pair ellpair; + bool ellinter=false; + + //check the first intersection point if it is inside ball + if(E.is_in(Bpolypair.first)){ + //std::cout<<"inside ball 1, radius:"<<_B.radius()< line_intersect_coord(Point &r, + Point &r_prev, + int rand_coord, + int rand_coord_prev, + std::vector &lamdas, + bool init + ){ + + std::pair Bpolypair = BP.line_intersect_coord(r,r_prev,rand_coord,rand_coord_prev,lamdas,init); + std::pair ellpair = E.line_intersect_coord(r,rand_coord); + return std::pair (std::min(Bpolypair.first,ellpair.first), + std::max(Bpolypair.second,ellpair.second)); + } + + + +};*/ + +#endif diff --git a/src/volesti/include/convex_bodies/barriers.h b/src/volesti/include/convex_bodies/barriers.h new file mode 100644 index 00000000..977e173f --- /dev/null +++ b/src/volesti/include/convex_bodies/barriers.h @@ -0,0 +1,52 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +struct LogarithmicBarrierAugmenter { + + template + struct LogarithmicBarrierObjective { + typedef HPolytope Hpolytope; + typedef typename Point::FT NT; + + func f; + Hpolytope &P; + bool enable; + + LogarithmicBarrierObjective(func f_, Hpolytope P_, bool enable_) : + f(f_), P(P_), enable(enable_) {} + + NT operator() (Point &x) { + if (enable) return f(x) + P.log_barrier(x); + else return f(x); + } + + }; + + template + struct LogarithmicBarrierGradient { + typedef HPolytope Hpolytope; + typedef typename Point::FT NT; + + func f; + Hpolytope &P; + bool enable; + + LogarithmicBarrierGradient(func f_, Hpolytope P_, bool enable_) : + f(f_), P(P_), enable(enable_) {} + + Point operator() (Point &x) { + if (enable) return f(x) + P.grad_log_barrier(x); + else return f(x); + } + + }; + +}; diff --git a/src/volesti/include/convex_bodies/convex_body.h b/src/volesti/include/convex_bodies/convex_body.h new file mode 100644 index 00000000..e70e9e66 --- /dev/null +++ b/src/volesti/include/convex_bodies/convex_body.h @@ -0,0 +1,109 @@ +#ifndef CONVEX_BODY_H +#define CONVEX_BODY_H + +#include +#include +#include +#include +#include + +/// This class represents a general convex body parameterized by a point type +/// \tparam Point Point type +template +class ConvexBody { +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef typename std::vector::iterator viterator; + //using RowMatrixXd = Eigen::Matrix; + //typedef RowMatrixXd MT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + typedef std::function func; + typedef std::function grad; +private: + unsigned int dim; //dimension + std::vector gs; // convex functions defining the convex body + std::vector grad_gs; // convex function gradients + unsigned int m; + NT tol = NT(1e-4); + + +public: + + ConvexBody() : m(0) {} + + ConvexBody(std::vector gs_, std::vector grad_gs_, unsigned int dim_) : + gs(gs_), grad_gs(grad_gs_), dim(dim_) + { + m = gs.size(); + } + + unsigned int dimension() { + return dim; + } + + // Compute positive line intersection (in [0, 1]) using Binary search + // x: starting point + // v: direction (ray) + std::pair line_positive_intersect(Point const& x, Point const &v) const { + NT t_min = NT(1); + int constraint = -1; + NT t; + for (unsigned int i = 0; i < m; i++) { + t = binary_search(x, v, gs[i]); + if (t < t_min) { + t_min = t; + constraint = i; + } + } + + return std::make_pair(t_min, constraint); + } + + NT binary_search(Point const &x, Point const &v, func const& f) const { + NT t_min = NT(0); + NT t_max = NT(1); + NT t; + NT value; + + while (t_max - t_min > tol) { + t = (t_max + t_min) / 2; + value = f(x + t * v); + if (value >= -tol && value <= 0) { + return t; + } else if (value < -tol) { + t_min = t; + } else { + t_max = t; + } + + } + + return t; + } + + + // Computes unit normal at point p of the boundary + Point unit_normal(Point const& p, int const& constraint) const { + Point n = grad_gs[constraint](p); + return (1 / n.length()) * n; + } + + // Computes reflection of v about point p on the boundary + void compute_reflection(Point &v, Point const& p, int const& constraint) const { + Point n = unit_normal(p, constraint); + v += -2 * v.dot(n) * n; + } + + // Check if point is in K + int is_in(Point const& p, NT tol=NT(0)) { + for (func g : gs) { + if (g(p) > NT(-tol)) return 0; + } + return -1; + } + +}; + +#endif diff --git a/src/volesti/include/convex_bodies/correlation_matrices/corre_matrix.hpp b/src/volesti/include/convex_bodies/correlation_matrices/corre_matrix.hpp new file mode 100755 index 00000000..23f4ecbe --- /dev/null +++ b/src/volesti/include/convex_bodies/correlation_matrices/corre_matrix.hpp @@ -0,0 +1,161 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Contributed by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_CORRE_MATRIX_HPP +#define VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_CORRE_MATRIX_HPP + +/// This class handles the PointType used by CorreSpectra_MT class. +/// Every point is a correlation matrix and only the lower triangular part is stored. +/// @tparam NT Number Type +template +class CorreMatrix{ + public: + + /// The numeric/matrix/vector types we use + typedef NT FT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + typedef Eigen::Matrix Coeff; + + MT mat; + + CorreMatrix(){} + + CorreMatrix(unsigned int n){ + mat = MT::Identity(n,n); + } + + CorreMatrix(MT const& mat){ + this->mat = mat; + } + + CorreMatrix(VT const& coeffs){ + unsigned int n = ceil(sqrt(2*coeffs.rows())); + this->mat = MT::Identity(n,n); + int ind = 0; + for(int i = 0; i < n; ++i){ + for(int j = 0; j < i; ++j){ + this->mat(i,j) = coeffs(ind); + ++ind; + } + } + } + + int dimension() const { + int n = this->mat.rows(); + return n*(n-1)/2; + } + + void operator+= (const CorreMatrix & p){ + this->mat += p.mat; + } + + void operator-= (const CorreMatrix & p){ + this->mat -= p.mat; + } + + void operator= (const CorreMatrix & p){ + this->mat = p.mat; + } + + CorreMatrix operator+ (const CorreMatrix& p) const { + CorreMatrix temp; + temp.mat = this->mat + p.mat; + return temp; + } + + + CorreMatrix operator- () const { + CorreMatrix temp; + temp.mat = - this->mat; + return temp; + } + + void operator*= (const FT k){ + this->mat *= k; + } + + CorreMatrix operator* (const FT k) const { + MT M = this->mat; + M *= k; + return CorreMatrix(M); + } + + void operator/= (const FT k){ + this->mat /= k; + } + + NT dot(MT grad){ + int i, j, n = this->mat.rows(); + NT ret = NT(0); + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + ret += this->mat(i,j) * grad(i,j); + } + } + return ret; + } + + NT dot(CorreMatrix c){ + int i, j, n = this->mat.rows(); + NT ret = NT(0); + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + ret += this->mat(i,j) * c.mat(i,j); + } + } + return ret; + } + + NT squared_length() const { + int i, j, n = this->mat.rows(); + NT ret = NT(0); + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + ret += this->mat(i,j) * this->mat(i,j); + } + } + return ret; + } + + void print() const { + int n = this->mat.rows(), i, j; + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + std::cout<< this->mat(i,j) <<" "; + } + } + std::cout<<"\n"; + } + + VT getCoefficients() const { + int n = this->mat.rows(), ind = 0, i, j; + VT coeff(n*(n-1)/2); + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + coeff(ind) = this->mat(i,j); + ++ind; + } + } + return coeff; + } +}; + +template +CorreMatrix operator* (const NT k, CorreMatrix p){ + return p * k; +} + +template +std::ostream& operator<<(std::ostream& os, const CorreMatrix& p){ + os << p.mat; + return os; +} + +#endif //VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_CORRE_MATRIX_HPP diff --git a/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron.hpp b/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron.hpp new file mode 100755 index 00000000..a615b33c --- /dev/null +++ b/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron.hpp @@ -0,0 +1,266 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Contributed by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_HPP +#define VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_HPP + +template +struct Precompute{ + + /// These flags indicate whether the corresponding matrices are computed + bool computed_A = false; + bool computed_B = false; + + /// The matrices the method positiveIntersection receives from its previous call + /// if the flag first_positive_intersection is true. + /// Matrix A is also used in coordinateIntersection + MT A, B; + + /// In method positive_intersect, the distance we are computing corresponds + /// to the minimum positive eigenvalue of a quadratic eigenvalue problem. + /// This will hold the eigenvector for that eigenvalue + VT eigenvector; + + /// Sets all flags to false + void resetFlags(){ + computed_A = computed_B = false; + } + + void set_mat_size(int const& n){ + A = -MT::Identity(n,n); + B.setZero(n, n); + eigenvector.setZero(n); + } +}; + +/// This class handles the spectrahedra of correlation matrices +/// The PointType here is stored as vector. +/// For the matrix PointType class, refer to CorrelationSpectrahedron_MT +/// @tparam Point Point Type +template +class CorrelationSpectrahedron : public Spectrahedron{ + public: + + /// The numeric/matrix/vector types we use + typedef Point PointType; + typedef typename Point::FT NT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + typedef Precompute PrecomputationOfValues; + + /// The size of the matrix + unsigned int n; + + PrecomputationOfValues _precomputedValues; + + /// Constructor of correlation matrix spectrahedra + + CorrelationSpectrahedron(unsigned int n){ + int i,j; + this->n = n; + this->d = n*(n-1)/2; + this->_inner_ball.first = PointType(this->d); + this->_inner_ball.second = 1/std::sqrt(this->d); + _precomputedValues.set_mat_size(n); + } + + /// \returns The size of the matrix + unsigned int matrixSize() const { + return n; + } + + std::pair getInnerBall() const { + return this->_inner_ball; + } + + /// Build a correlation matrix from a vector of entries + /// \param[in] vector of coefficients + /// \param[in] the matrix to be assigned + void buildMatrix(VT const &pvector, unsigned int const n, MT &mat) const { + NT coeff; + int i, j, ind = 0; + for(i = 0; i < n ; ++i){ + mat(i,i) = -1; + } + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + coeff = -pvector[ind]; + mat(i,j) = mat(j,i) = coeff; + ++ind; + } + } + } + + /// Computes the reflected direction at a point on the boundary of the spectrahedron. + /// \param[in] r A point on the boundary of the spectrahedron + /// \param[in] v The direction of the trajectory as it hits the boundary + /// \param[out] reflectedDirection The reflected direction + template + void compute_reflection(PointType &v, PointType const &r, update_parameters&) const { + VT grad(this->d); + VT e = _precomputedValues.eigenvector; + int i, j, ind = 0; + NT sum_sq = NT(0); + + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + grad(ind) = e[i]*e[j]; + sum_sq += grad(ind)*grad(ind); + ++ind; + } + } + NT dot = v.dot(grad); + dot = 2 * dot / sum_sq; + v -= dot * PointType(grad); + } + + /// Construct the generalized eigenvalue problem \[Bt - A \] for positive_intersect. + /// \param[in] p Input vector + /// \param[in] v Input vector + /// \param[in, out] _precomputedValues Holds matrices B = I - A(v), A = A(p) + void createMatricesForPositiveLinearIntersection(VT const &p, VT const &v){ + if (true) { + VT pvector = p, vvector = v; + NT coeff; + int i, j, ind =0; + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + coeff = -pvector[ind]; + _precomputedValues.A(i,j) = _precomputedValues.A(j,i) = coeff; + coeff = -vvector[ind]; + _precomputedValues.B(i,j) = _precomputedValues.B(j,i) = coeff; + ++ind; + } + } + _precomputedValues.computed_B = true; + } + } + + NT positiveLinearIntersection(VT const &p, VT const &v){ + createMatricesForPositiveLinearIntersection(p, v); + return this->EigenvaluesProblem.minPosLinearEigenvalue_EigenSymSolver(-_precomputedValues.A, _precomputedValues.B, _precomputedValues.eigenvector); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const &r, PointType const &v){ + NT pos_inter = positiveLinearIntersection(r.getCoefficients(), v.getCoefficients()); + return std::pair (pos_inter, -1); + } + + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT& , + NT const&){ + return line_positive_intersect(r, v); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT& , + NT const&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + NT const&, + MT const&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_first_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + // compute intersection point of ray starting from r and pointing to v + std::pair line_intersect(PointType const &r, PointType const &v){ + createMatricesForPositiveLinearIntersection(r.getCoefficients(), v.getCoefficients()); + return this->EigenvaluesProblem.symGeneralizedProblem(_precomputedValues.A, _precomputedValues.B); + } + + + std::pair line_intersect(PointType const &r, + PointType const &v, + VT&, + VT&){ + return line_intersect(r, v); + } + + std::pair line_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + NT&){ + return line_intersect(r, v); + } + + /// Compute the gradient of the determinant of the LMI at p + /// \param[in] p Input parameter + /// \param[in] Input vector: the eigenvector A(p)*e = 0 + /// \param[out] ret The unit normal vector at p + void unit_normal(VT p, VT const &e, VT &ret) const { + int i, j, ind = 0; + NT sum_sqqrt_sq = NT(0); + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + ret(ind) = e[i]*e[j]; + sum_sqqrt_sq += ret(ind)*ret(ind); + ++ind; + } + } + ret /= std::sqrt(sum_sqqrt_sq); //normalize + } + + /// Test if a point p is in the spectrahedron + /// \param p is the current point + /// \return true if position is outside the spectrahedron + int is_in(PointType const &p, NT tol=NT(0)) const { + if(isExterior(p.getCoefficients())) return 0; + return -1; + } + + bool isExterior(VT const &pos) const { + MT mat = MT(n, n); + buildMatrix(pos, n, mat); + return isExterior(mat); + } + + bool isExterior(MT const &mat) const { + return !this->EigenvaluesProblem.isPositiveSemidefinite(-mat); + } + + MT get_mat() const { + return MT::Identity(this->d, this->d); + } +}; + +#endif //VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_HPP \ No newline at end of file diff --git a/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron_MT.hpp b/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron_MT.hpp new file mode 100755 index 00000000..d3f754a5 --- /dev/null +++ b/src/volesti/include/convex_bodies/correlation_matrices/correlation_spectrahedron_MT.hpp @@ -0,0 +1,180 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Contributed by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_MT_HPP +#define VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_MT_HPP + +#include "convex_bodies/correlation_matrices/corre_matrix.hpp" + +/// This class handles the spectrahedra of correlation matrices +/// @tparam CorreMatrix The Correlation Matrix +template +class CorrelationSpectrahedron_MT : public Spectrahedron{ + public: + + /// The numeric/matrix/vector types we use + typedef CorreMatrix PointType; + typedef typename PointType::FT NT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + /// The size of the matrix + unsigned int n; + + VT eigenvector; + + /// Constructor of correlation matrix spectrahedra + + CorrelationSpectrahedron_MT(unsigned int n){ + int i,j; + this->n = n; + this->d = n*(n-1)/2; + this->_inner_ball.first = PointType(this->d); + this->_inner_ball.second = 1/std::sqrt(this->d); + this->eigenvector.setZero(n); + } + + /// \returns The size of the matrix + unsigned int matrixSize() const { + return n; + } + + std::pair getInnerBall() const { + return this->_inner_ball; + } + + /// Computes the reflected direction at a point on the boundary of the spectrahedron. + /// \param[in] r A point on the boundary of the spectrahedron + /// \param[in] v The direction of the trajectory as it hits the boundary + /// \param[out] reflectedDirection The reflected direction + template + void compute_reflection(PointType &v, PointType const &r, update_parameters&) const { + MT grad = MT::Zero(this->n, this->n); + int i, j; + NT sum_sq = NT(0), dot = NT(0); + + for(i = 0; i < n ; ++i){ + for(j = 0; j < i; ++j){ + grad(i,j) = eigenvector[i]*eigenvector[j]; + sum_sq += grad(i,j)*grad(i,j); + dot += grad(i,j) * v.mat(i,j); + } + } + dot = 2 * dot / sum_sq; + grad = dot*grad; + v -= PointType(grad); + } + + /// Computes the minimal positive t s.t. r+t*v intersects the boundary of the spectrahedron + /// \param[in] r + /// \param[in] v + /// \param[out] a NT value t + NT positiveLinearIntersection(PointType const &r, PointType const &v){ + + // minPosLinearEigenvalue_EigenSymSolver(A,B) computes the minimal positive eigenvalue of A-t*B + + return this->EigenvaluesProblem.minPosLinearEigenvalue_EigenSymSolver(r.mat, (-v).mat, eigenvector); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const &r, + PointType const &v) { + NT pos_inter = positiveLinearIntersection(r, v); + return std::pair (pos_inter, -1); + } + + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT& , + NT const&){ + return line_positive_intersect(r, v); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT& , + NT const&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + NT const&, + MT const&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + template + std::pair line_first_positive_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + update_parameters&){ + return line_positive_intersect(r, v); + } + + // compute intersection point of ray starting from r and pointing to v + std::pair line_intersect(PointType const &r, PointType const &v) const { + return this->EigenvaluesProblem.symGeneralizedProblem(-r.mat, -v.mat); + } + + std::pair line_intersect(PointType const &r, + PointType const &v, + VT&, + VT&) const { + return line_intersect(r, v); + } + + std::pair line_intersect(PointType const &r, + PointType const &v, + VT&, + VT&, + NT&) const { + return line_intersect(r, v); + } + + + /// Test if a point p is in the spectrahedron + /// \param p is the current point + /// \return true if position is outside the spectrahedron + int is_in(PointType const &p, NT tol=NT(0)) const { + if(this->EigenvaluesProblem.isPositiveSemidefinite(p.mat)){ + return -1; + } + return 0; + } + + bool isExterior(MT const &mat) const { + return !this->EigenvaluesProblem.isPositiveSemidefinite(mat); + } + + MT get_mat() const { + return MT::Identity(this->d, this->d); + } +}; + +#endif //VOLESTI_CONVEX_BODIES_CORRELATION_MATRICES_VOLESTI_CORRELATION_SPECTRAHEDRON_MT_HPP diff --git a/src/volesti/include/convex_bodies/ellipsoid.h b/src/volesti/include/convex_bodies/ellipsoid.h new file mode 100644 index 00000000..18c7570c --- /dev/null +++ b/src/volesti/include/convex_bodies/ellipsoid.h @@ -0,0 +1,281 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef ELLIPSOIDS_H +#define ELLIPSOIDS_H + +#include +#include +#include "volume/math_helpers.hpp" + +/// This class represents an ellipsoid parameterized by a point type +/// \tparam Point Point type +template +class Ellipsoid{ +public: +typedef Point PointType; + typedef typename Point::FT NT; + typedef typename std::vector::iterator viterator; + typedef Eigen::Matrix VT; + typedef Eigen::Matrix MT; + +private: + // representation is x'A x <= 1, i.e center is always assumed to be origin. + MT A; + + unsigned int _dim; + MT _L_cov; // LL' = inv(A) for sampling procedures + + // eigen vectors and values + VT _eigen_values; + VT _eigen_values_inv; + VT _eigen_values_inv_sqrt; + MT _Eigen_Vectors; + +public: + + Ellipsoid() {} + + // TODO(vaithak): Add a flag for telling whether the matrix passed is already inverse + Ellipsoid(MT& Ain) : A(Ain) { + Eigen::SelfAdjointEigenSolver eigensolver(A); + if (eigensolver.info() != Eigen::Success) { + throw std::runtime_error("Eigen solver returned error!"); + } + + _eigen_values = eigensolver.eigenvalues(); + _Eigen_Vectors = eigensolver.eigenvectors(); + + _eigen_values_inv = _eigen_values.array().inverse().matrix(); + _eigen_values_inv_sqrt = _eigen_values_inv.array().sqrt().matrix(); + + _dim = A.rows(); + + Eigen::LLT lltOfA(A.inverse()); // compute the Cholesky decomposition of inv(A) + if (lltOfA.info() != Eigen::Success) { + throw std::runtime_error("Cholesky decomposition failed!"); + } + _L_cov = lltOfA.matrixL(); + } + + + // Constructor for copula ellipsoid only + Ellipsoid(std::vector >& Ain) { + _dim = Ain.size(); + A.resize(_dim, _dim); + for (unsigned int i = 0; i < Ain.size(); i++) { + for (unsigned int j = 0; j < Ain.size(); j++) { + A(i,j) = Ain[i][j]; + } + } + } + + + NT radius() const { + return _eigen_values_inv_sqrt(0); + } + + + VT eigenvalues() const { + return _eigen_values; + } + + + VT eigenvalues_inv() const { + return _eigen_values_inv; + } + + + VT eigenvalues_inv_sqrt() const { + return _eigen_values_inv_sqrt; + } + + + MT eigenvectors() const { + return _Eigen_Vectors; + } + + + unsigned int dimensions() const { + return _dim; + } + + + MT Lcov() const { + return _L_cov; + } + + + // return L_cov * x + VT mult_Lcov(VT const& x) const { + return _L_cov.template triangularView() * x; + } + + + void print() const { + std::cout << "Ellipse is in the form: x' A x <= 1, (center is assumed to be origin always) \n"; + std::cout << "A = \n" << A; + } + + + + NT mat_mult(Point const& p) const { + VT x = p.getCoefficients(); + return x.transpose() * A.template selfadjointView() * x; + } + + + VT vec_mult(VT const& b) const { + return A.template selfadjointView()*b; + } + + + NT log_volume() const { + NT ball_log_vol = (NT(_dim)/NT(2) * std::log(M_PI)) - log_gamma_function(NT(_dim) / NT(2) + 1); + NT det_factor = std::log( _eigen_values_inv_sqrt.prod() ); + + return det_factor + ball_log_vol; + } + + + void scale(NT scale_factor) { + assert (scale_factor > 0); + + NT scale_factor_sq = scale_factor * scale_factor; + NT inv_scale_factor = (NT(1.0) / scale_factor); + NT inv_scale_factor_sq = (NT(1.0) / scale_factor_sq); + + _eigen_values = inv_scale_factor_sq * _eigen_values; + _eigen_values_inv = scale_factor_sq * _eigen_values_inv; + _eigen_values_inv_sqrt = scale_factor * _eigen_values_inv_sqrt; + _L_cov = scale_factor * _L_cov; + + A = inv_scale_factor_sq * A; // as volume depends on square root of it's determinant + } + + + int is_in(Point const& p) const { + return mat_mult(p) > 1 ? 0 : -1; + } + + + // compute intersection point of ray starting from r and pointing to v + std::pair line_intersect(Point const& r, Point const& v) const { + // constants of a quadratic equation + NT a_q = mat_mult(v); + NT b_q = 2 * r.getCoefficients().dot(vec_mult(v.getCoefficients())); + NT c_q = mat_mult(r) - 1; + + NT D = std::pow(b_q, 2) - 4*a_q*c_q; + return std::pair ( (-b_q + std::sqrt(D))/(2*a_q) , (-b_q - std::sqrt(D))/(2*a_q) ); + } + + + std::pair line_intersect(Point const& r, + Point const& v, + const VT &Ar, + const VT &Av) const + { + return line_intersect(r, v); + } + + + std::pair line_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT &lambda_prev) const + { + return line_intersect(r, v); + } + + + std::pair line_positive_intersect(Point const& r, + Point const& v) const + { + NT res = line_intersect(r, v).first; + return std::pair(res, 0); + } + + + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av) const + { + return line_positive_intersect(r, v); + } + + + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev) const + { + return line_positive_intersect(r, v); + } + + + // Compute the intersection of a coordinate ray + std::pair line_intersect_coord(Point const& r, const unsigned int rand_coord) const { + NT a_q = A(rand_coord, rand_coord); + NT b_q = 2 * r.getCoefficients().dot(A.col(rand_coord)); + NT c_q = mat_mult(r) - 1; + + NT D = std::pow(b_q, 2) - 4*a_q*c_q; + return std::pair ( (-b_q + std::sqrt(D))/(2*a_q) , (-b_q - std::sqrt(D))/(2*a_q) ); + } + + + std::pair line_intersect_coord(Point const& r, + unsigned int const& rand_coord, + VT& lamdas) const + { + return line_intersect_coord(r, rand_coord); + } + + + std::pair line_intersect_coord(Point const& r, + Point const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + VT& lamdas) const + { + return line_intersect_coord(r, rand_coord); + } + + + void compute_reflection (Point& v, Point const& p) const + { + // normal vector is Ap + Point s(vec_mult(p.getCoefficients())); + s *= (1.0 / s.length()); + s *= (-2.0 * v.dot(s)); + v += s; + } + + + template + void compute_reflection (Point& v, Point const& p, update_parameters ¶ms) const + { + // normal vector is Ap + Point s(vec_mult(p.getCoefficients())); + params.ball_inner_norm = s.length(); + + params.inner_vi_ak = v.dot(s) / params.ball_inner_norm; + v += (s * (-2.0 * params.inner_vi_ak * (1.0 / params.ball_inner_norm))); + } +}; + +#endif \ No newline at end of file diff --git a/src/volesti/include/convex_bodies/hpolytope.h b/src/volesti/include/convex_bodies/hpolytope.h new file mode 100644 index 00000000..49d94282 --- /dev/null +++ b/src/volesti/include/convex_bodies/hpolytope.h @@ -0,0 +1,927 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef HPOLYTOPE_H +#define HPOLYTOPE_H + +#include +#include +#include +#include "preprocess/max_inscribed_ball.hpp" +#include "root_finders/quadratic_polynomial_solvers.hpp" +#ifndef DISABLE_LPSOLVE + #include "lp_oracles/solve_lp.h" +#endif + + + +// check if an Eigen vector contains NaN or infinite values +template +bool is_inner_point_nan_inf(VT const& p) +{ + typedef Eigen::Array VTint; + VTint a = p.array().isNaN(); + for (int i = 0; i < p.rows(); i++) { + if (a(i) || std::isinf(p(i))){ + return true; + } + } + return false; +} + +/// This class describes a polytope in H-representation or an H-polytope +/// i.e. a polytope defined by a set of linear inequalities +/// \tparam Point Point type +template +class HPolytope { +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef typename std::vector::iterator viterator; + //using RowMatrixXd = Eigen::Matrix; + //typedef RowMatrixXd MT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + +private: + unsigned int _d; //dimension + MT A; //matrix A + VT b; // vector b, s.t.: Ax<=b + std::pair _inner_ball; + +public: + //TODO: the default implementation of the Big3 should be ok. Recheck. + HPolytope() {} + + HPolytope(unsigned d_, MT const& A_, VT const& b_) : + _d{d_}, A{A_}, b{b_} + { + } + + // Copy constructor + HPolytope(HPolytope const& p) : + _d{p._d}, A{p.A}, b{p.b}, _inner_ball{p._inner_ball} + { + } + + //define matrix A and vector b, s.t. Ax<=b, + // from a matrix that contains both A and b, i.e., [A | b ] + HPolytope(std::vector> const& Pin) + { + _d = Pin[0][1] - 1; + A.resize(Pin.size() - 1, _d); + b.resize(Pin.size() - 1); + for (unsigned int i = 1; i < Pin.size(); i++) { + b(i - 1) = Pin[i][0]; + for (unsigned int j = 1; j < _d + 1; j++) { + A(i - 1, j - 1) = -Pin[i][j]; + } + } + _inner_ball.second = -1; + //_inner_ball = ComputeChebychevBall(A, b); + } + + + std::pair InnerBall() const + { + return _inner_ball; + } + + void set_InnerBall(std::pair const& innerball) //const + { + _inner_ball = innerball; + } + + void set_interior_point(Point const& r) + { + _inner_ball.first = r; + } + + //Compute Chebyshev ball of H-polytope P:= Ax<=b + //Use LpSolve library + std::pair ComputeInnerBall() + { + normalize(); + #ifndef DISABLE_LPSOLVE + _inner_ball = ComputeChebychevBall(A, b); // use lpsolve library + #else + + if (_inner_ball.second <= NT(0)) { + + NT const tol = 0.00000001; + std::tuple inner_ball = max_inscribed_ball(A, b, 150, tol); + + // check if the solution is feasible + if (is_in(Point(std::get<0>(inner_ball))) == 0 || std::get<1>(inner_ball) < NT(0) || + std::isnan(std::get<1>(inner_ball)) || std::isinf(std::get<1>(inner_ball)) || + !std::get<2>(inner_ball) || is_inner_point_nan_inf(std::get<0>(inner_ball))) + { + _inner_ball.second = -1.0; + } else + { + _inner_ball.first = Point(std::get<0>(inner_ball)); + _inner_ball.second = std::get<1>(inner_ball); + } + } + #endif + + return _inner_ball; + } + + // return dimension + unsigned int dimension() const + { + return _d; + } + + + // return the number of facets + int num_of_hyperplanes() const + { + return A.rows(); + } + + int num_of_generators() const + { + return 0; + } + + + // return the matrix A + MT get_mat() const + { + return A; + } + + + MT get_AA() const { + return A * A.transpose(); + } + + // return the vector b + VT get_vec() const + { + return b; + } + + + // change the matrix A + void set_mat(MT const& A2) + { + A = A2; + } + + + // change the vector b + void set_vec(VT const& b2) + { + b = b2; + } + + Point get_mean_of_vertices() const + { + return Point(_d); + } + + NT get_max_vert_norm() const + { + return 0.0; + } + + + // print polytope in input format + void print() { + std::cout << " " << A.rows() << " " << _d << " double" << std::endl; + for (unsigned int i = 0; i < A.rows(); i++) { + for (unsigned int j = 0; j < _d; j++) { + std::cout << A(i, j) << " "; + } + std::cout << "<= " << b(i) << std::endl; + } + } + + + // Compute the reduced row echelon form + // used to transofm {Ax=b,x>=0} to {A'x'<=b'} + // e.g. Birkhoff polytopes + /* + // Todo: change the implementation in order to use eigen matrix and vector. + int rref(){ + to_reduced_row_echelon_form(_A); + std::vector zeros(_d+1,0); + std::vector ones(_d+1,0); + std::vector zerorow(_A.size(),0); + for (int i = 0; i < _A.size(); ++i) + { + for (int j = 0; j < _d+1; ++j){ + if ( _A[i][j] == double(0)){ + ++zeros[j]; + ++zerorow[i]; + } + if ( _A[i][j] == double(1)){ + ++ones[j]; + } + } + } + for(typename stdMatrix::iterator mit=_A.begin(); mit<_A.end(); ++mit){ + int j =0; + for(typename stdCoeffs::iterator lit=mit->begin(); litend() ; ){ + if(zeros[j]==_A.size()-1 && ones[j]==1) + (*mit).erase(lit); + else{ //reverse sign in all but the first column + if(lit!=mit->end()-1) *lit = (-1)*(*lit); + ++lit; + } + ++j; + } + } + //swap last and first columns + for(typename stdMatrix::iterator mit=_A.begin(); mit<_A.end(); ++mit){ + double temp=*(mit->begin()); + *(mit->begin())=*(mit->end()-1); + *(mit->end()-1)=temp; + } + //delete zero rows + for (typename stdMatrix::iterator mit=_A.begin(); mit<_A.end(); ){ + int zero=0; + for(typename stdCoeffs::iterator lit=mit->begin(); litend() ; ++lit){ + if(*lit==double(0)) ++zero; + } + if(zero==(*mit).size()) + _A.erase(mit); + else + ++mit; + } + //update _d + _d=(_A[0]).size(); + // add unit vectors + for(int i=1;i<_d;++i){ + std::vector e(_d,0); + e[i]=1; + _A.push_back(e); + } + // _d should equals the dimension + _d=_d-1; + return 1; + }*/ + + + //Check if Point p is in H-polytope P:= Ax<=b + int is_in(Point const& p, NT tol=NT(0)) const + { + int m = A.rows(); + const NT* b_data = b.data(); + + for (int i = 0; i < m; i++) { + //Check if corresponding hyperplane is violated + if (*b_data - A.row(i) * p.getCoefficients() < NT(-tol)) + return 0; + + b_data++; + } + return -1; + } + + // compute intersection point of ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_intersect(Point const& r, Point const& v) const + { + + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom, sum_denom; + //unsigned int i, j; + unsigned int j; + int m = num_of_hyperplanes(); + + + sum_nom.noalias() = b - A * r.getCoefficients(); + sum_denom.noalias() = A * v.getCoefficients(); + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = sum_denom.data(); + + for (int i = 0; i < m; i++) { + + if (*sum_denom_data == NT(0)) { + //std::cout<<"div0"< 0) min_plus = lamda; + if (lamda > max_minus && lamda < 0) max_minus = lamda; + } + + sum_nom_data++; + sum_denom_data++; + } + return std::make_pair(min_plus, max_minus); + } + + // compute intersection points of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + bool pos = false) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom; + int m = num_of_hyperplanes(), facet; + + Ar.noalias() = A * r.getCoefficients(); + sum_nom = b - Ar; + Av.noalias() = A * v.getCoefficients();; + + + NT* Av_data = Av.data(); + NT* sum_nom_data = sum_nom.data(); + + for (int i = 0; i < m; i++) { + if (*Av_data == NT(0)) { + //std::cout<<"div0"< 0) { + min_plus = lamda; + if (pos) facet = i; + }else if (lamda > max_minus && lamda < 0) max_minus = lamda; + } + + Av_data++; + sum_nom_data++; + } + if (pos) return std::make_pair(min_plus, facet); + return std::make_pair(min_plus, max_minus); + } + + std::pair line_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + bool pos = false) const + { + + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom; + NT mult; + //unsigned int i, j; + unsigned int j; + int m = num_of_hyperplanes(), facet; + + Ar.noalias() += lambda_prev*Av; + sum_nom = b - Ar; + Av.noalias() = A * v.getCoefficients(); + + NT* sum_nom_data = sum_nom.data(); + NT* Av_data = Av.data(); + + for (int i = 0; i < m; i++) { + if (*Av_data == NT(0)) { + //std::cout<<"div0"< 0) { + min_plus = lamda; + if (pos) facet = i; + }else if (lamda > max_minus && lamda < 0) max_minus = lamda; + } + Av_data++; + sum_nom_data++; + } + if (pos) return std::make_pair(min_plus, facet); + return std::make_pair(min_plus, max_minus); + } + + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av) const + { + return line_intersect(r, v, Ar, Av, true); + } + + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev) const + { + return line_intersect(r, v, Ar, Av, lambda_prev, true); + } + + + //---------------------------accelarated billiard---------------------------------- + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + template + std::pair line_first_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + update_parameters& params) const + { + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + + NT lamda = 0; + VT sum_nom; + int m = num_of_hyperplanes(), facet; + + Ar.noalias() = A * r.getCoefficients(); + sum_nom.noalias() = b - Ar; + Av.noalias() = A * v.getCoefficients(); + + NT* Av_data = Av.data(); + NT* sum_nom_data = sum_nom.data(); + + for (int i = 0; i < m; i++) { + if (*Av_data == NT(0)) { + //std::cout<<"div0"< 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *Av_data; + } + } + + Av_data++; + sum_nom_data++; + } + params.facet_prev = facet; + return std::pair(min_plus, facet); + } + + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters& params) const + { + + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + + NT lamda = 0; + NT inner_prev = params.inner_vi_ak; + VT sum_nom; + int m = num_of_hyperplanes(), facet; + + Ar.noalias() += lambda_prev*Av; + if(params.hit_ball) { + Av.noalias() += (-2.0 * inner_prev) * (Ar / params.ball_inner_norm); + } else { + Av.noalias() += (-2.0 * inner_prev) * AA.col(params.facet_prev); + } + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* Av_data = Av.data(); + + for (int i = 0; i < m; i++) { + if (*Av_data == NT(0)) { + //std::cout<<"div0"< 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *Av_data; + } + } + Av_data++; + sum_nom_data++; + } + params.facet_prev = facet; + return std::pair(min_plus, facet); + } + + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters& params) const + { + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + + NT lamda = 0; + VT sum_nom; + unsigned int j; + int m = num_of_hyperplanes(), facet; + + Ar.noalias() += lambda_prev*Av; + sum_nom.noalias() = b - Ar; + Av.noalias() = A * v.getCoefficients(); + + NT* sum_nom_data = sum_nom.data(); + NT* Av_data = Av.data(); + + for (int i = 0; i < m; i++) { + if (*Av_data == NT(0)) { + //std::cout<<"div0"< 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *Av_data; + } + } + Av_data++; + sum_nom_data++; + } + params.facet_prev = facet; + return std::pair(min_plus, facet); + } + + //-----------------------------------------------------------------------------------// + + + //First coordinate ray intersecting convex polytope + std::pair line_intersect_coord(Point const& r, + unsigned int const& rand_coord, + VT& lamdas) const + { + + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_denom; + + int m = num_of_hyperplanes(); + + sum_denom = A.col(rand_coord); + lamdas.noalias() = b - A * r.getCoefficients(); + + NT* lamda_data = lamdas.data(); + NT* sum_denom_data = sum_denom.data(); + + for (int i = 0; i < m; i++) { + + if (*sum_denom_data == NT(0)) { + //std::cout<<"div0"< 0) min_plus = lamda; + if (lamda > max_minus && lamda < 0) max_minus = lamda; + + } + lamda_data++; + sum_denom_data++; + } + return std::make_pair(min_plus, max_minus); + } + + + //Not the first coordinate ray intersecting convex + std::pair line_intersect_coord(Point const& r, + Point const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + VT& lamdas) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + + int m = num_of_hyperplanes(); + + lamdas.noalias() += A.col(rand_coord_prev) + * (r_prev[rand_coord_prev] - r[rand_coord_prev]); + NT* data = lamdas.data(); + + for (int i = 0; i < m; i++) { + NT a = A(i, rand_coord); + + if (a == NT(0)) { + //std::cout<<"div0"< 0) min_plus = lamda; + if (lamda > max_minus && lamda < 0) max_minus = lamda; + + } + data++; + } + return std::make_pair(min_plus, max_minus); + } + + + //------------------------------oracles for exponential sampling---------------////// + + std::pair get_positive_quadratic_root(Point const& r, //current poistion + Point const& v, // current velocity + VT& Ac, // the product Ac where c is the bias vector of the exponential distribution + NT const& T, // the variance of the exponential distribution + VT& Ar, // the product Ar + VT& Av, // the product Av + int& facet_prev) const //the facet that the trajectory hit in the previous reflection + { + NT lamda = 0; + NT lamda2 =0; + NT lamda1 =0; + NT alpha; + NT min_plus = std::numeric_limits::max(); + VT sum_nom; + int m = num_of_hyperplanes(); + int facet = -1; + + sum_nom = Ar - b; + Av.noalias() = A * v.getCoefficients();; + + NT* Av_data = Av.data(); + NT* sum_nom_data = sum_nom.data(); + NT* Ac_data = Ac.data(); + + for (int i = 0; i < m; i++) + { + alpha = -((*Ac_data) / (2.0 * T)); + if (solve_quadratic_polynomial(alpha, (*Av_data), (*sum_nom_data), lamda1, lamda2)) + { + lamda = pick_first_intersection_time_with_boundary(lamda1, lamda2, i, facet_prev); + if (lamda < min_plus && lamda > 0) + { + min_plus = lamda; + facet = i; + } + } + Av_data++; + sum_nom_data++; + Ac_data++; + } + facet_prev = facet; + return std::make_pair(min_plus, facet); + } + + + // compute intersection points of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair quadratic_positive_intersect(Point const& r, //current poistion + Point const& v, // current velocity + VT& Ac, // the product Ac where c is the bias vector of the exponential distribution + NT const& T, // the variance of the exponential distribution + VT& Ar, // the product Ar + VT& Av, // the product Av + int& facet_prev) const //the facet that the trajectory hit in the previous reflection + { + Ar.noalias() = A * r.getCoefficients(); + return get_positive_quadratic_root(r, v, Ac, T, Ar, Av, facet_prev); + } + + std::pair quadratic_positive_intersect(Point const& r, //current poistion + Point const& v, // current velocity + VT& Ac, // the product Ac where c is the bias vector of the exponential distribution + NT const& T, // the variance of the exponential distribution + VT& Ar, // the product Ar + VT& Av, // the product Av + NT const& lambda_prev, // the intersection time of the previous reflection + int& facet_prev) const //the facet that the trajectory hit in the previous reflection + { + Ar.noalias() += ((lambda_prev * lambda_prev) / (-2.0*T)) * Ac + lambda_prev * Av; + return get_positive_quadratic_root(r, v, Ac, T, Ar, Av, facet_prev); + } + + NT pick_first_intersection_time_with_boundary(NT const& lamda1, NT const& lamda2, int const& current_facet, int const& previous_facet) const + { + if (lamda1 == lamda2) + { + return lamda1; + } + NT lamda; + const double tol = 1e-10; + std::pair minmax_values = std::minmax(lamda1, lamda2); + + lamda = (previous_facet == current_facet) + ? minmax_values.second < NT(tol) ? minmax_values.first : minmax_values.second + : minmax_values.second; + + if (lamda1 * lamda2 < NT(0)) + { + lamda = (previous_facet == current_facet) + ? (minmax_values.second < NT(tol)) ? minmax_values.first : minmax_values.second + : minmax_values.second; + } + else + { + lamda = (previous_facet == current_facet) + ? (minmax_values.first >= NT(0) && minmax_values.first < NT(tol)) + ? minmax_values.second : minmax_values.first + : minmax_values.first; + } + return lamda; + } + + + //------------oracle for exact hmc spherical gaussian sampling---------------// + + // compute intersection point of ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair trigonometric_positive_intersect(Point const& r, Point const& v, + NT const& omega, int &facet_prev) const + { + + NT lamda = 0, C, Phi, t1, t2, tmin; + NT min_plus = std::numeric_limits::max(), t = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom, sum_denom; + unsigned int j; + int m = num_of_hyperplanes(), facet = -1; + + + sum_nom.noalias() = A * r.getCoefficients(); + sum_denom.noalias() = A * v.getCoefficients(); + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = sum_denom.data(); + const NT* b_data = b.data(); + + for (int i = 0; i < m; i++) { + + C = std::sqrt((*sum_nom_data) * (*sum_nom_data) + ((*sum_denom_data) * (*sum_denom_data)) / (omega * omega)); + Phi = std::atan((-(*sum_denom_data)) / ((*sum_nom_data) * omega)); + if ((*sum_denom_data) < 0.0 && Phi < 0.0) { + Phi += M_PI; + } else if ((*sum_denom_data) > 0.0 && Phi > 0.0) { + Phi -= M_PI; + } + + if (C > (*b_data)) { + NT acos_b = std::acos((*b_data) / C); + t1 = (acos_b - Phi) / omega; + if (facet_prev == i && std::abs(t1) < 1e-10){ + t1 = (2.0 * M_PI) / omega; + } + + t2 = (-acos_b - Phi) / omega; + if (facet_prev == i && std::abs(t2) < 1e-10){ + t2 = (2.0 * M_PI) / omega; + } + + t1 += (t1 < NT(0)) ? (2.0 * M_PI) / omega : NT(0); + t2 += (t2 < NT(0)) ? (2.0 * M_PI) / omega : NT(0); + + tmin = std::min(t1, t2); + + if (tmin < t && tmin > NT(0)) { + facet = i; + t = tmin; + } + } + + sum_nom_data++; + sum_denom_data++; + b_data++; + } + facet_prev = facet; + return std::make_pair(t, facet); + } + + + // Apply linear transformation, of square matrix T^{-1}, in H-polytope P:= Ax<=b + void linear_transformIt(MT const& T) + { + A = A * T; + } + + + // shift polytope by a point c + + void shift(const VT &c) + { + b -= A*c; + } + + + // return for each facet the distance from the origin + std::vector get_dists(NT const& radius) const + { + unsigned int i=0; + std::vector dists(num_of_hyperplanes(), NT(0)); + typename std::vector::iterator disit = dists.begin(); + for ( ; disit!=dists.end(); disit++, i++) + *disit = b(i) / A.row(i).norm(); + + return dists; + } + + // no points given for the rounding, you have to sample from the polytope + template + bool get_points_for_rounding (T const& /*randPoints*/) + { + return false; + } + + MT get_T() const + { + return A; + } + + void normalize() + { + NT row_norm; + for (int i = 0; i < num_of_hyperplanes(); ++i) + { + row_norm = A.row(i).norm(); + A.row(i) = A.row(i) / row_norm; + b(i) = b(i) / row_norm; + } + } + + void compute_reflection(Point& v, Point const&, int const& facet) const + { + v += -2 * v.dot(A.row(facet)) * A.row(facet); + } + + void resetFlags() {} + + NT log_barrier(Point &x, NT t = NT(100)) const { + int m = num_of_hyperplanes(); + NT total = NT(0); + NT slack; + + for (int i = 0; i < m; i++) { + slack = b(i) - x.dot(A.row(i)); + total += log(slack); + } + + return total / t; + } + + Point grad_log_barrier(Point &x, NT t = NT(100)) { + int m = num_of_hyperplanes(); + NT slack; + + Point total(x.dimension()); + + for (int i = 0; i < m; i++) { + slack = b(i) - x.dot(A.row(i)); + total = total + (1 / slack) * A.row(i); + } + total = (1.0 / t) * total; + return total; + } + + template + void compute_reflection(Point &v, const Point &, update_parameters const& params) const { + + Point a((-2.0 * params.inner_vi_ak) * A.row(params.facet_prev)); + v += a; + } + + void update_position_internal(NT&){} + + template + std::tuple curve_intersect( + NT t_prev, + NT t0, + NT eta, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + NonLinearOracle &intersection_oracle, + int ignore_facet=-1) + { + return intersection_oracle.apply(t_prev, t0, eta, A, b, *this, + coeffs, phi, grad_phi, ignore_facet); + } +}; + +#endif diff --git a/src/volesti/include/convex_bodies/orderpolytope.h b/src/volesti/include/convex_bodies/orderpolytope.h new file mode 100644 index 00000000..2df5c661 --- /dev/null +++ b/src/volesti/include/convex_bodies/orderpolytope.h @@ -0,0 +1,755 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2021 Vaibhav Thakkar + +// Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ORDER_POLYTOPE_H +#define ORDER_POLYTOPE_H + +#include +#include "misc/poset.h" +#include +#include "preprocess/max_inscribed_ball.hpp" +#ifndef DISABLE_LPSOLVE + #include "lp_oracles/solve_lp.h" +#endif + +/// This class represents an order polytope parameterized by a point type +/// \tparam Point Point type +template +class OrderPolytope { +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef Eigen::Matrix VT; + typedef Eigen::Matrix MT; + +private: + Poset _poset; + unsigned int _d; // dimension + + VT b; + VT _row_norms; + MT _A; // representing as Ax <= b for ComputeInnerBall and printing + + unsigned int _num_hyperplanes; + bool _normalized; + +public: + OrderPolytope(Poset const& poset) : _poset(poset) + { + _d = _poset.num_elem(); + _num_hyperplanes = 2*_d + _poset.num_relations(); // 2*d are for >=0 and <=1 constraints + b = Eigen::MatrixXd::Zero(_num_hyperplanes, 1); + _A = Eigen::MatrixXd::Zero(_num_hyperplanes, _d); + _row_norms = Eigen::MatrixXd::Constant(_num_hyperplanes, 1, 1.0); + + // first add (ai >= 0) or (-ai <= 0) rows + _A.topLeftCorner(_d, _d) = -Eigen::MatrixXd::Identity(_d, _d); + + // next add (ai <= 1) rows + _A.block(_d, 0, _d, _d) = Eigen::MatrixXd::Identity(_d, _d); + b.block(_d, 0, _d, 1) = Eigen::MatrixXd::Constant(_d, 1, 1.0); + + // next add the relations + unsigned int num_relations = _poset.num_relations(); + for(int idx=0; idx curr_relation = _poset.get_relation(idx); + _A(2*_d + idx, curr_relation.first) = 1; + _A(2*_d + idx, curr_relation.second) = -1; + } + _row_norms.block(2*_d, 0, num_relations, 1) = Eigen::MatrixXd::Constant(num_relations, 1, sqrt(2)); + + _normalized = false; + } + + + // return dimension + unsigned int dimension() const + { + return _d; + } + + + // return number of hyperplanes + unsigned int num_of_hyperplanes() const + { + return _num_hyperplanes; + } + + + // get ith column of A + VT get_col (unsigned int i) const { + return _A.col(i); + } + + + Eigen::SparseMatrix get_mat() const + { + return _A.sparseView(); + } + + + VT get_vec() const + { + return b; + } + + + // print polytope in Ax <= b format + void print() const + { + std::cout << " " << _A.rows() << " " << _d << " double" << std::endl; + for (unsigned int i = 0; i < _A.rows(); i++) { + for (unsigned int j = 0; j < _d; j++) { + std::cout << _A(i, j) << " "; + } + std::cout << "<= " << b(i) << std::endl; + } + } + + + /** multiply the sparse matrix A of the order polytope by a vector x + * if transpose = false : return Ax + * else if transpose = true : return (A^T)x + */ + VT vec_mult(VT const& x, bool transpose=false) const + { + unsigned int rows = num_of_hyperplanes(); + unsigned int i = 0; + VT res; + if (!transpose) res = Eigen::MatrixXd::Zero(rows, 1); + else res = Eigen::MatrixXd::Zero(_d, 1); + + // ------- no effect of normalize on first 2*_d rows, norm = 1 --------- + // first _d rows of >=0 constraints + for(; i < _d; ++i) { + res(i) = (-1.0) * x(i); + } + + // next _d rows of <=1 constraints + for(; i < 2*_d; ++i) { + if (!transpose) { + res(i) = (1.0) * x(i - _d); + } + else { + res(i - _d) += (1.0) * x(i); + } + } + // ----------------------------------------------------------------- + + // next rows are for order relations + for(; i < rows; ++i) { + std::pair curr_relation = _poset.get_relation(i - 2*_d); + + if (!transpose) { + if (! _normalized) + res(i) = x(curr_relation.first) - x(curr_relation.second); + else + res(i) = (x(curr_relation.first) - x(curr_relation.second)) / _row_norms(i); + } + else { + if (! _normalized) { + res(curr_relation.first) += x(i); + res(curr_relation.second) -= x(i); + } + else { + res(curr_relation.first) += x(i) / _row_norms(i); + res(curr_relation.second) -= x(i) / _row_norms(i); + } + } + } + + return res; + } + + + //Check if Point p is in the order-polytope + int is_in(Point const& p, NT tol=NT(0)) const + { + assert(p.dimension() == _d); + + VT pt_coeffs = p.getCoefficients(); + NT diff; + + for (int i = 0; i < _d; i++) { + // DON'T JUST check violation of point between 0 and 1 + // as b will change for shifted polytope. + diff = -pt_coeffs(i) - b(i); + if (diff > NT(tol)) return 0; + + diff = pt_coeffs(i) - b(i + _d); + if (diff > NT(tol)) return 0; + } + + // check violations of order relations + // again note that b can be arbitrary because of shifting + unsigned int num_relations = _poset.num_relations(); + for(int idx=0; idx curr_relation = _poset.get_relation(idx); + diff = (pt_coeffs(curr_relation.first) - pt_coeffs(curr_relation.second)); + + if (_normalized) diff /= _row_norms(idx + 2*_d); + if((diff - b(idx + 2*_d)) > NT(tol)) + return 0; + } + + return -1; + } + + + //Compute Chebyshev ball of the polytope P:= Ax<=b + //Use LpSolve library + std::pair ComputeInnerBall() + { + normalize(); + std::pair inner_ball; + #ifndef DISABLE_LPSOLVE + inner_ball = ComputeChebychevBall(_A, b); // use lpsolve library + #else + + if (inner_ball.second <= NT(0)) { + + NT const tol = 0.00000001; + std::tuple inner_ball = max_inscribed_ball(_A, b, 150, tol); + + // check if the solution is feasible + if (is_in(Point(std::get<0>(inner_ball))) == 0 || std::get<1>(inner_ball) < NT(0) || + std::isnan(std::get<1>(inner_ball)) || std::isinf(std::get<1>(inner_ball)) || + !std::get<2>(inner_ball) || is_inner_point_nan_inf(std::get<0>(inner_ball))) + { + inner_ball.second = -1.0; + } else + { + inner_ball.first = Point(std::get<0>(inner_ball)); + inner_ball.second = std::get<1>(inner_ball); + } + } + #endif + + return inner_ball; + + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // compute intersection point of ray starting from r and pointing to v + // with the order polytope + std::pair line_intersect(Point const& r, Point const& v, bool pos = false) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom, sum_denom; + + int rows = num_of_hyperplanes(), facet; + + sum_nom.noalias() = b - vec_mult(r.getCoefficients()); + sum_denom.noalias() = vec_mult(v.getCoefficients()); + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = sum_denom.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + if (pos) facet = i; + } + else if (lamda > max_minus && lamda < 0) { + max_minus = lamda; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + if (pos) + return std::make_pair(min_plus, facet); + + return std::make_pair(min_plus, max_minus); + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // compute intersection point of ray starting from r and pointing to v + // with the order-polytope + std::pair line_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + bool pos = false) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom; + + int rows = num_of_hyperplanes(), facet; + + Ar.noalias() = vec_mult(r.getCoefficients()); + Av.noalias() = vec_mult(v.getCoefficients()); + + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = Av.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + if (pos) facet = i; + } + else if (lamda > max_minus && lamda < 0) { + max_minus = lamda; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + if (pos) + return std::make_pair(min_plus, facet); + + return std::make_pair(min_plus, max_minus); + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // compute intersection point of ray starting from r and pointing to v + // with the order-polytope + std::pair line_intersect(Point const& r, + Point const& v, + VT &Ar, + VT &Av, + NT const& lambda_prev, + bool pos = false) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_nom; + + int rows = num_of_hyperplanes(), facet; + + Ar.noalias() += lambda_prev*Av; + Av.noalias() = vec_mult(v.getCoefficients()); + + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = Av.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + if (pos) facet = i; + } + else if (lamda > max_minus && lamda < 0) { + max_minus = lamda; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + if (pos) + return std::make_pair(min_plus, facet); + + return std::make_pair(min_plus, max_minus); + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // compute intersection point of a ray starting from r and pointing to v + // with the order-polytope + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av) const + { + return line_intersect(r, v, Ar, Av, true); + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // compute intersection point of a ray starting from r and pointing to v + // with the order-polytope + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev) const + { + return line_intersect(r, v, Ar, Av, lambda_prev, true); + } + + + //-------------------------accelerated billiard--------------------------------// + // compute intersection point of a ray starting from r and pointing to v + // with the order-polytope + template + std::pair line_first_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + update_parameters ¶ms) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + VT sum_nom; + + int rows = num_of_hyperplanes(), facet; + + Ar.noalias() = vec_mult(r.getCoefficients()); + Av.noalias() = vec_mult(v.getCoefficients()); + + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = Av.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *sum_denom_data; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + params.facet_prev = facet; + return std::make_pair(min_plus, facet); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters ¶ms) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + VT sum_nom; + + int rows = num_of_hyperplanes(), facet; + NT inner_prev = params.inner_vi_ak; + + Ar.noalias() += lambda_prev*Av; + if(params.hit_ball) { + Av.noalias() += (-2.0 * inner_prev) * (Ar / params.ball_inner_norm); + } else { + Av.noalias() += (-2.0 * inner_prev) * AA.col(params.facet_prev); + } + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = Av.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *sum_denom_data; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + params.facet_prev = facet; + return std::make_pair(min_plus, facet); + } + + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters ¶ms) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + VT sum_nom; + + int rows = num_of_hyperplanes(), facet; + + Ar.noalias() += lambda_prev*Av; + Av.noalias() = vec_mult(v.getCoefficients()); + + sum_nom.noalias() = b - Ar; + + NT* sum_nom_data = sum_nom.data(); + NT* sum_denom_data = Av.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + facet = i; + params.inner_vi_ak = *sum_denom_data; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + params.facet_prev = facet; + return std::make_pair(min_plus, facet); + } + //------------------------------------------------------------------------------// + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + // Compute the intersection of a coordinate ray + // with the order polytope + std::pair line_intersect_coord(Point const& r, + unsigned int const& rand_coord, + VT& lamdas) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + VT sum_denom; + + int rows = num_of_hyperplanes(); + + sum_denom = get_col(rand_coord); + lamdas = b - vec_mult(r.getCoefficients()); + + NT* sum_nom_data = lamdas.data(); + NT* sum_denom_data = sum_denom.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + } + else if (lamda > max_minus && lamda < 0) { + max_minus = lamda; + } + } + + sum_nom_data++; + sum_denom_data++; + } + + return std::make_pair(min_plus, max_minus); + } + + + // TODO: This can be removed as only modified Accelerated Billiard Walk will be used with Order Polytope + std::pair line_intersect_coord(Point const& r, + Point const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + VT& lamdas) const + { + NT lamda = 0; + NT min_plus = std::numeric_limits::max(); + NT max_minus = std::numeric_limits::lowest(); + + int rows = num_of_hyperplanes(); + + lamdas.noalias() += get_col(rand_coord_prev) + * (r_prev[rand_coord_prev] - r[rand_coord_prev]); + NT* sum_nom_data = lamdas.data(); + + // iterate over all hyperplanes + for(unsigned int i = 0; i 0) { + min_plus = lamda; + } + else if (lamda > max_minus && lamda < 0) { + max_minus = lamda; + } + } + + sum_nom_data++; + } + + return std::make_pair(min_plus, max_minus); + } + + + // no points given for the rounding, you have to sample from the polytope + template + bool get_points_for_rounding (T const& /*randPoints*/) + { + return false; + } + + + // shift polytope by a point c + void shift(VT const& c) + { + b -= vec_mult(c); + } + + + // get a point inside the order polytope, + // NOTE: the current implementation only works for non shifted order polytope + VT inner_point() + { + // get topologically sorted list of indices + std::vector sorted_list = _poset.topologically_sorted_list(); + + // vector to hold n linearly spaced values between 0-1 + std::vector lin_space_values(_d); + NT start = 0.05, end = 0.95; + NT h = (end - start)/static_cast(_d-1); + NT val = start; + for(auto x=lin_space_values.begin(); x!=lin_space_values.end(); ++x) { + *x = val; + val += h; + } + + // final result vector + VT res(_d); + for(int i=0; i<_d; ++i) { + unsigned int curr_idx = sorted_list[i]; + res(curr_idx) = lin_space_values[i]; + } + + return res; + } + + + void normalize() + { + if (_normalized == true) + return; + + // for b and _A, first 2*_d rows are already normalized, for + _normalized = true; // -> will be used to make normalization idempotent + for (unsigned int i = 0; i < _num_hyperplanes; ++i) + { + _A.row(i) /= _row_norms(i); + b(i) /= _row_norms(i); + } + } + + + // return for each facet the distance from the origin + std::vector get_dists(NT const& radius) const + { + std::vector dists(_num_hyperplanes, NT(0)); + + for (unsigned int i = 0; i < _num_hyperplanes; ++i) { + if (_normalized) + dists[i] = b(i); + else + dists[i] = b(i) / _row_norms(i); + } + + return dists; + } + + + // compute reflection given dot product and facet + void compute_reflection(Point& v, NT dot_prod, unsigned int facet) const + { + // calculating -> v += -2 * dot_prod * A.row(facet); + if (facet < _d) { + v.set_coord(facet, v[facet] - 2 * dot_prod * (-1.0)); + } + else if (facet < 2*_d) { + v.set_coord(facet-_d, v[facet-_d] - 2 * dot_prod * (1.0)); + } + else { + std::pair curr_relation = _poset.get_relation(facet - 2*_d); + v.set_coord(curr_relation.first, v[curr_relation.first] - 2 * dot_prod * (1.0 / _row_norms(facet))); + v.set_coord(curr_relation.second, v[curr_relation.second] - 2 * dot_prod * (-1.0 / _row_norms(facet))); + } + } + + + // compute reflection in O(1) time for order polytope + void compute_reflection(Point& v, Point const&, unsigned int facet) const + { + NT dot_prod; + if (facet < _d) { + dot_prod = -v[facet]; + } + else if (facet < 2*_d) { + dot_prod = v[facet - _d]; + } + else { + std::pair curr_relation = _poset.get_relation(facet - 2*_d); + dot_prod = v[curr_relation.first] - v[curr_relation.second]; + dot_prod = dot_prod / _row_norms(facet); + } + + compute_reflection(v, dot_prod, facet); + } + + + template + void compute_reflection(Point &v, Point const&, update_parameters const& params) const + { + NT dot_prod = params.inner_vi_ak; + unsigned int facet = params.facet_prev; + compute_reflection(v, dot_prod, facet); + } +}; + +#endif diff --git a/src/volesti/include/convex_bodies/spectrahedra/LMI.h b/src/volesti/include/convex_bodies/spectrahedra/LMI.h new file mode 100644 index 00000000..5ad879fc --- /dev/null +++ b/src/volesti/include/convex_bodies/spectrahedra/LMI.h @@ -0,0 +1,251 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_LMI_H +#define VOLESTI_LMI_H + +#include "matrix_operations/EigenvaluesProblems.h" + + +template +struct evaluate_lmi { + +}; + +template +struct evaluate_lmi > { +public: + typedef Eigen::Matrix MT; + /// The type for Eigen vector + typedef Eigen::Matrix VT; + + MT vectorMatrix; + + int _m, _d; + + /// Create the vectorMatrix, which has at each column the distinct elements of each A_i, i=1,...,d + void setVectorMatrix(int const& m, int const& d, std::vector &matrices) + { + _m = m; + _d = d; + int newM = m * (m + 1) / 2; + + // allocate memory + vectorMatrix.setZero(newM, d); + //a.setZero(m); + + // initialze iterator and skip A_0 + typename std::vector::iterator iter = matrices.begin(); + iter++; + + // copy elements + int atMatrix = 0; + + for (; iter != matrices.end(); iter++, atMatrix++) { + int i = 0; + + for (int at_row = 0; at_row < m; at_row++) + for (int at_col = at_row; at_col < m; at_col++) { + vectorMatrix(i++, atMatrix) = (*iter)(at_row, at_col); + } + + } + } + + /// Compute \[x_1*A_1 + ... + x_n A_n] + /// \param[in] x Input vector + /// \param[out] res Output matrix + void evaluateWithoutA0(const VT &x, MT& res, bool complete_mat = false) const { + //#define EVALUATE_WITHOUT_A0_NAIVE + #if defined(EVALUATE_WITHOUT_A0_NAIVE) + res.setZero(_m, _m); + typename std::vector::iterator it; + + int i = 0; + it = matrices.begin(); + ++it; // skip A0 + for (; it != matrices.end(); it++, i++) + res.noalias() += x(i) * (*it); + #else + + VT a = vectorMatrix * x; + + double *data = res.data(); + double *v = a.data(); + + int at = 0; + + // copy lower triangular + for (int at_col = 0; at_col < _m; at_col++) { + int col_offset = at_col * _m; + double *target = data + col_offset + at_col; + + for (int at_row = at_col; at_row < _m; at_row++) { + *(target++) = *(v++); + } + } + + if(complete_mat) { + v = a.data(); + + // copy upper triangular + for (int at_row = 0; at_row < _m; at_row++) { + double *target = data + at_row + at_row * _m; + + for (int at_col = at_row; at_col < _m; at_col++) { + *target = *(v++); + target = target + _m; + } + } + } + #endif + //return true; + } + +}; + + +/// This class handles a linear matrix inequality of the form \[A_0 + \sum x_i A_i\] +/// A template specialization for dense Eigen matrices and vectors +/// @tparam NT Numeric Type +/// @tparam MT Matrix Type +/// @tparam VT Vector Type +template +class LMI { + public: + + evaluate_lmi lmi_evaluator; + + /// The matrices A_0, A_i + std::vector matrices; + + /// The dimension of the vector x + unsigned int d; + + /// The size of the matrices A_i + unsigned int m; + + /// At each column keep the m*(m+1)/2 distinct elements of each matrix A_i, i=1,...,d + MT vectorMatrix; + + LMI(){} + + /// Creates A LMI object + /// \param[in] matrices The matrices A_0, A_i + LMI(std::vector& matrices) { + typename std::vector::iterator it = matrices.begin(); + + while (it!=matrices.end()) { + this->matrices.push_back(*it); + it++; + } + + d = matrices.size() - 1; + m = matrices[0].rows(); + + lmi_evaluator.setVectorMatrix(m, d, matrices); + } + + /// \returns The dimension of vector x + unsigned int dimension() const { + return d; + } + + /// \return The matrices A0, A1, ..., Ad + std::vector getMatrices() const { + return matrices; + } + + /// \returns The size of the matrices + unsigned int sizeOfMatrices() const { + return m; + } + + /// Evaluate A_0 + \[A_0 + \sum x_i A_i \] + /// \param[in] x The input vector + /// \param[out] ret The output matrix + void evaluate(VT const & x, MT& ret, bool complete_mat = false) const { + lmi_evaluator.evaluateWithoutA0(x, ret, complete_mat); + + // add A0 + ret += matrices[0]; + } + + /// Compute \[x_1*A_1 + ... + x_n A_n] + /// \param[in] x Input vector + /// \param[out] res Output matrix + void evaluateWithoutA0(const VT& x, MT& res, bool complete_mat = false) const { + lmi_evaluator.evaluateWithoutA0(x, res, complete_mat); + } + + /// Compute the gradient of the determinant of the LMI at p + /// \param[in] p Input parameter + /// \param[in] Input vector: lmi(p)*e = 0, e != 0 + /// \param[out] ret The normalized gradient of the determinant of the LMI at p + void normalizedDeterminantGradient(VT r, VT const& e, VT &ret) const { + NT* ret_data = ret.data(); + NT sum_sqqrt_sq = NT(0); + for (int i = 0; i < d; i++) { + // todo, use iterators + *ret_data = e.dot(matrices[i+1].template selfadjointView< Eigen::Lower >() * e); + sum_sqqrt_sq += (*ret_data) * (*ret_data); + ret_data++; + } + + //normalize + ret /= std::sqrt(sum_sqqrt_sq); + } + + /// \param i An indicator to a matrix + /// \return Pointer to A_i + MT* const getMatrix(const int i) { + return &(matrices[i]); + } + + MT get_A0() { + return matrices[0]; + } + + void set_A0(MT const& A0) { + matrices[0] = A0; + } + + /// Prints the matrices A0, ..., An + void print() const { + int i = 0; + + for (auto iter = matrices.begin(); iter != matrices.end(); iter++, i++) { + std::cout << "A" << i << "\n"; + std::cout << *iter << "\n\n"; + } + } + + /// check if the matrix is negative semidefinite + /// \param matrix a matrix + /// \return Pointer to A_i + bool isNegativeSemidefinite(MT const & matrix ) const { + EigenvaluesProblems eigs; + NT eival = eigs.findSymEigenvalue(matrix); + return eival <= 0; + } + + /// evaluate LMI(pos) and check if its negative semidefinite + /// \param pos a vector of our current position + /// \return true is LMI(pos) is negative semidefinite + bool isNegativeSemidefinite(VT const & pos) const { + MT mat; + mat.setZero(m, m); + + evaluate(pos, mat, true); + return isNegativeSemidefinite(mat); + } + +}; + +#endif //VOLESTI_LMI_H diff --git a/src/volesti/include/convex_bodies/spectrahedra/spectrahedron.h b/src/volesti/include/convex_bodies/spectrahedra/spectrahedron.h new file mode 100644 index 00000000..e8f34138 --- /dev/null +++ b/src/volesti/include/convex_bodies/spectrahedra/spectrahedron.h @@ -0,0 +1,492 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +// Modified by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_SPECTRAHEDRON_H +#define VOLESTI_SPECTRAHEDRON_H + +#include "LMI.h" +#include "chrono" + + +/// Among successive calls of this class methods, we may need to pass data +/// from one call to the next, to avoid repeating computations, or to efficiently update values +/// Warning: this struct assists in many methods; perhaps for different methods use different instances +template +struct PrecomputationOfValues { + + /// These flags indicate whether the corresponding matrices are computed + /// if yes, we can use them and not compute them fro scratch + // TODO: avoid the use of flags + bool computed_A = false; + bool computed_C = false; + bool computed_XY = false; + + /// The matrices the method positiveIntersection receives from its previous call + /// if the flag first_positive_intersection is true. + /// Matrix A is also used in coordinateIntersection + MT A, B, C, X, Y; + + /// In method positive_intersect, the distance we are computing corresponds + /// to the minimum positive eigenvalue of a quadratic eigenvalue problem. + /// This will hold the eigenvector for that eigenvalue + VT eigenvector; + + /// Sets all flags to false + void resetFlags() { + computed_XY = computed_C = computed_A = false; + } + + void set_mat_size(int const& m) + { + A.setZero(m, m); + B.setZero(m, m); + C.setZero(m, m); + + X.setZero(2*m, 2*m); + Y.setZero(2*m, 2*m); + + eigenvector.setZero(m); + } +}; + + +/// This class manipulates a spectrahedron, described by a Linear Matrix Inequality i.e. LMI +/// \tparam Point Point Type +template +class Spectrahedron { +public: + + /// The numeric/matrix/vector types we use + typedef Point PointType; + typedef typename Point::FT NT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + double maxDouble = std::numeric_limits::max(); + + /// The type of a pair of NT + typedef std::pair pairNT; + + typedef PrecomputationOfValues _PrecomputationOfValues; + + _PrecomputationOfValues precomputedValues; + + EigenvaluesProblems EigenvaluesProblem; + + /// The dimension of the spectrahedron + unsigned int d; + VT grad; + std::pair _inner_ball; + + /// The linear matrix inequality that describes the spectrahedron + LMI lmi; + + Spectrahedron() {} + + /// Creates a spectrahedron + /// \param[in] lmi The linear matrix inequality that describes the spectrahedron + Spectrahedron(const LMI& lmi) : lmi(lmi) { + d = lmi.dimension(); + //grad.setZero(d); + precomputedValues.resetFlags(); + precomputedValues.set_mat_size(lmi.sizeOfMatrices()); + } + + void set_interior_point(PointType const& r) + { + _inner_ball.first = r; + } + + std::pair ComputeInnerBall() { + NT radius = maxDouble; + + for (unsigned int i = 0; i < dimension(); ++i) { + + std::pair min_max = coordinateIntersection(_inner_ball.first.getCoefficients(), i+1); + + if (min_max.first < radius) radius = min_max.first; + if (-min_max.second < radius) radius = -min_max.second; + } + + radius = radius / std::sqrt(NT(dimension())); + _inner_ball.second = radius; + + return std::pair(_inner_ball.first, radius); + } + + std::pair InnerBall() const + { + return _inner_ball; + } + + /// Construct the quadratic eigenvalue problem \[At^2 + Bt + C \] for positive_intersect. + /// A = lmi(c) - A0, B = lmi(b) - A0 and C = lmi(c). + /// \param[in] a Input vector + /// \param[in] b Input vector + /// \param[in] c Input vector + /// \param[in, out] precomputedValues Holds matrices A, C + void createMatricesForPositiveQuadIntersection(const VT& a, const VT& b, const VT& c) { + + // check if matrices A, C are ready + // if not compute them + if (!precomputedValues.computed_A) { + lmi.evaluateWithoutA0(a, precomputedValues.A, true); + } + + if (!precomputedValues.computed_C) { + lmi.evaluate(c, precomputedValues.C, true); + } + + // compute Matrix B + lmi.evaluateWithoutA0(b, precomputedValues.B, true); + } + + /// Construct the generalized eigenvalue problem \[Bt + C \] for positive_intersect. + /// \param[in] p Input vector + /// \param[in] v Input vector + /// \param[in, out] precomputedValues Holds matrices A, C + void createMatricesForPositiveLinearIntersection(const VT& p, const VT& v) { + // check if matrices A, C are ready + // if not compute them + if (!precomputedValues.computed_C) { + lmi.evaluate(p, precomputedValues.C); + } + + // compute Matrix B + lmi.evaluateWithoutA0(v, precomputedValues.B); + } + + + void createMatricesForPositiveIntersection(const VT& p, const VT& v) { + + // check if matrices B, C are ready if not compute them + if (!precomputedValues.computed_C) + { + lmi.evaluate(p, precomputedValues.C); + } + + lmi.evaluateWithoutA0(v, precomputedValues.B); + } + + /// Computes the distance d we must travel on the parametrized polynomial curve \[at^2 + bt + c \], + /// assuming we start at t=0, and we start increasing t. + /// We construct the quadratic eigenvalue problem \[At^2 + Bt + C \], + /// where A = lmi(c) - A0, B = lmi(b) - A0 and C = lmi(c). + /// Then we do a linearization and transform it to the generalized problem X+lY, + /// which we pass to an external library. + /// \param[in] a Input vector, the coefficient of t \[t^2\] + /// \param[in] b Input vector, the coefficient of t + /// \param[in] c Input Vector, the constant term + /// \returns The distance d + NT positiveQuadIntersection(VT const & a, VT const & b, VT const & c) { + unsigned int matrixDim = lmi.sizeOfMatrices(); + + // create matrices A, B, C + createMatricesForPositiveQuadIntersection(a, b, c); + + // get the minimum positive eigenvalue of At^2 + Bt + C + NT distance = EigenvaluesProblem.minPosQuadraticEigenvalue(precomputedValues.A, precomputedValues.B, + precomputedValues.C, precomputedValues.X, + precomputedValues.Y, + precomputedValues.eigenvector, + precomputedValues.computed_XY); + return distance; + } + + + NT positiveLinearIntersection(VT const & p, VT const & v) + { + createMatricesForPositiveLinearIntersection(p, v); + NT distance = EigenvaluesProblem.minPosLinearEigenvalue(precomputedValues.C, precomputedValues.B, + precomputedValues.eigenvector); + return distance; + } + + /// Computes the distance d one must travel on the line a + tb, + /// assuming we start at t=0 and that b has zero everywhere and 1 in its i-th coordinate. + /// We must solve the generalized eigenvalue problem A+tB, where A = lmi(a) and B=(lmi) - A0 = A_i + /// If the flag precomputedValues,computed_A is true, the matrix A is not computed. + /// \param[in] a Input vector + /// \param[in] coordinate Indicator of the i-th coordinate, 1 <= coordinate <= dimension + /// \return The pair (positive t, negative t) for which we reach the boundary + pairNT coordinateIntersection(VT const & a, int const coordinate) { + + // prepare the generalized eigenvalue problem A+lB + // we may not have to compute A! + if (!precomputedValues.computed_A) + lmi.evaluate(a, precomputedValues.A); + + return EigenvaluesProblem.symGeneralizedProblem(precomputedValues.A, *(lmi.getMatrix(coordinate))); + } + + //First coordinate ray intersecting convex polytope + std::pair line_intersect_coord(Point &r, + unsigned int const& rand_coord, + VT&) + { + return coordinateIntersection(r.getCoefficients(), rand_coord); + } + + //Not the first coordinate ray intersecting convex + std::pair line_intersect_coord(PointType &r, + PointType&, + unsigned int const& rand_coord, + unsigned int&, + VT&) + { + return coordinateIntersection(r.getCoefficients(), rand_coord); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const& r, + PointType const& v) + { + NT pos_inter = positiveLinearIntersection(r.getCoefficients(), v.getCoefficients()); + return std::pair (pos_inter, -1); + } + + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT&, + VT& , + NT const&) { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT&, + VT& , + NT const&, + update_parameters&) + { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT&, + VT&, + NT const&, + MT const&, + update_parameters& ) + { + return line_positive_intersect(r, v); + } + + template + std::pair line_first_positive_intersect(PointType const& r, + PointType const& v, + VT&, + VT&, + update_parameters&) + { + return line_positive_intersect(r, v); + } + + // compute intersection point of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT&, + VT&) + { + return line_positive_intersect(r, v); + } + + // compute intersection point of ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair line_intersect(PointType const& r, PointType const& v) + { + NT pos_inter = positiveLinearIntersection(r.getCoefficients(), v.getCoefficients()); + NT neg_inter = -positiveLinearIntersection(r.getCoefficients(), NT(-1)*v.getCoefficients()); + + return std::make_pair(pos_inter, neg_inter); + } + + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT&, + VT&) + { + return line_intersect(r, v); + } + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT&, + VT&, + NT&) + { + return line_intersect(r, v); + } + + void update_position_internal(NT &t){ + precomputedValues.C += t * precomputedValues.B; + precomputedValues.computed_C = true; + } + + MT get_mat() const + { + return MT::Identity(lmi.dimension(), lmi.dimension()); + } + + void resetFlags() + { + precomputedValues.resetFlags(); + } + + void set_flags(bool bool_flag) + { + precomputedValues.computed_A = bool_flag; + precomputedValues.computed_C = bool_flag; + precomputedValues.computed_XY = bool_flag; + } + + MT get_C() const + { + return precomputedValues.C; + } + + void update_C(NT const& lambda) + { + precomputedValues.C += (lambda * lambda) * precomputedValues.A + lambda * precomputedValues.B; + } + + // return the number of facets + int num_of_hyperplanes() const + { + return 0; + } + + void shift(VT e) { + MT A0 = getLMI().get_A0(); + std::vector matrices = getLMI().getMatrices(); + + int d = matrices.size(); + + for (int i = 1; i < d; ++i) { + A0 = A0 + e(i-1)*matrices[i]; + } + + lmi.set_A0(A0); + + _inner_ball.first = PointType(dimension()); + } + + /// Computes the reflected direction at a point on the boundary of the spectrahedron. + /// \param[in] r A point on the boundary of the spectrahedron + /// \param[in] v The direction of the trajectory as it hits the boundary + /// \param[out] reflectedDirection The reflected direction + template + void compute_reflection(PointType &v, PointType const& r, update_parameters& ) const + { + VT grad(d); + lmi.normalizedDeterminantGradient(r.getCoefficients(), precomputedValues.eigenvector, grad); + + // compute reflected direction + // if v is original direction and s the surface normal, + // reflected direction = v - 2 *s + + NT dot = 2 * v.dot(grad); + v += -dot * PointType(grad); + } + + + /// \return The dimension of the spectrahedron + unsigned int dimension() const { + return d; + } + + /// \return The LMI describing this spectrahedron + LMI getLMI() const { + return lmi; + } + + /// Estimates the diameter of the spectrahedron. It samples points uniformly with coordinate directions + /// hit and run, and returns the maximum distance between them. + /// \tparam Point + /// \param[in] numPoints The number of points to sample for the estimation + /// \return An estimation of the diameter of the spectrahedron + template + NT estimateDiameter(int const numPoints, PointType const & interiorPoint, RNGType &rng) { + + std::list randPoints; + + precomputedValues.computed_A = false; + VT p = interiorPoint.getCoefficients(); + + // sample points with walk length set to 1 + for (int samplingNo=0 ; samplingNo distances = this->coordinateIntersection(p, coordinate); + + // uniformly set the new point on the segment + // defined by the intersection points + NT lambda = rng.sample_urdist(); + NT diff = distances.first + lambda * (distances.second - distances.first); + + p(coordinate - 1) = p(coordinate - 1) + diff; + + // update the precomputedValues, so we can skip + // computations in the next call + precomputedValues.computed_A = true; + precomputedValues.A += diff * *(this->getLMI().getMatrix(coordinate)); + randPoints.push_back(Point(p)); + } + + // find maximum distance among points; + NT maxDistance = 0; + typename std::list::iterator itInner, itOuter = randPoints.begin(); + + for (; itOuter!=randPoints.end() ; ++itOuter) + for (itInner=itOuter ; itInner!=randPoints.end() ; ++itInner) { + NT current = itOuter->distance(*itInner); + if (current > maxDistance) + maxDistance = current; + } + + return maxDistance; + } + + int is_in(PointType const& p, NT tol=NT(0)) const + { + if (isExterior(p.getCoefficients())) { + return 0; + } + return -1; + } + + /// Find out is lmi(current position) = mat is in the exterior of the spectrahedron + /// \param mat a matrix where mat = lmi(current position) + /// \return true if position is outside the spectrahedron + bool isExterior(MT const & mat) const { + return !lmi.isNegativeSemidefinite(mat); + } + + /// Find out is pos is in the exterior of the spectrahedron + /// \param pos a vector + /// \return true if pos is outside the spectrahedron + bool isExterior(VT const & pos) const { + return !lmi.isNegativeSemidefinite(pos); + } +}; + +#endif //VOLESTI_SPECTRAHEDRON_H diff --git a/src/volesti/include/convex_bodies/vpolyintersectvpoly.h b/src/volesti/include/convex_bodies/vpolyintersectvpoly.h new file mode 100644 index 00000000..a6dd97ca --- /dev/null +++ b/src/volesti/include/convex_bodies/vpolyintersectvpoly.h @@ -0,0 +1,429 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VPOLYINTERSECTVPOLY_H +#define VPOLYINTERSECTVPOLY_H + +#include +#include +#include +#include "sampling/sphere.hpp" + +/// This class represents the intersection of two V-polytopes +/// \tparam VPolytope VPolytope Type +/// \tparam RNGType RNGType Type +template +class IntersectionOfVpoly { +public: + typedef typename VPolytope::NT NT; + typedef typename VPolytope::PointType PointType; + typedef PointType Point; + typedef typename VPolytope::MT MT; + typedef typename VPolytope::VT VT; + unsigned seed; + std::pair _inner_ball; + NT rad; + VPolytope P1; + VPolytope P2; + + IntersectionOfVpoly(): P1(), P2() {} + + IntersectionOfVpoly(VPolytope P, VPolytope Q) : P1(P), P2(Q) { + seed = std::chrono::system_clock::now().time_since_epoch().count(); + } + + IntersectionOfVpoly(VPolytope P, VPolytope Q, unsigned _seed) : P1(P), P2(Q) { + seed = _seed; + } + + VPolytope first() { return P1; } + VPolytope second() { return P2; } + + std::pair InnerBall() const + { + return _inner_ball; + } + + int is_in(const Point &p, NT tol=NT(0)) const { + if(P1.is_in(p)==-1) + return P2.is_in(p); + return 0; + } + + + int num_of_hyperplanes() const { + return 0; + } + + unsigned int dimension() const { + return P1.dimension(); + } + + int num_of_vertices() const { + return P1.num_of_vertices() + P2.num_of_vertices(); + } + + unsigned int upper_bound_of_hyperplanes() const { + return dimension() + 1; + //return 4; + } + + int num_of_generators() const { + return 0; + } + + NT getRad() const { + return rad; + } + + MT get_mat() const { + return P1.get_mat(); + } + + VT get_vec() const { + return P1.get_vec(); + } + + MT get_T() const { + return P1.get_mat(); + } + + MT get_mat2() const { + return P2.get_mat(); + } + + Point get_mean_of_vertices() const { + return Point(P1.dimension()); + } + + + NT get_max_vert_norm() const { + return 0.0; + } + + void print() { + P1.print(); + P2.print(); + } + + bool is_feasible() { + bool empty; + int k = P1.get_mat().rows() + P2.get_mat().rows(); + RNGType rng(k); + rng.set_seed(seed); + PointInIntersection(P1.get_mat(), P2.get_mat(), + GetDirection::apply(k, rng), empty); + return !empty; + } + + std::pair ComputeInnerBall() { + + unsigned int num = 0, d = P1.dimension(); + MT V1 = P1.get_mat(), V2 = P2.get_mat(); + int k1 = V1.rows(), k2 = V2.rows(); + int k = k1 + k2; + RNGType rng(k); + rng.set_seed(seed); + Point direction(k), p(d); + std::vector vertices; + typename std::vector::iterator rvert; + bool same; + + while(num::apply(k, rng); + p = PointInIntersection(V1, V2, direction, same); + + same = false; + rvert = vertices.begin(); + for ( ; rvert!=vertices.end(); ++rvert) { + if (p==(*rvert)) { + same = true; + break; + } + } + if (same) continue; + vertices.push_back(p); + num++; + + } + + _inner_ball = P1.get_center_radius_inscribed_simplex(vertices.begin(), vertices.end()); + return _inner_ball; + + } + + void set_InnerBall(std::pair const& innerball) const + { + _inner_ball = innerball; + } + + void set_interior_point(Point const& r) + { + _inner_ball.first = r; + } + +/* + unsigned int num_of_v = 0; + unsigned int d = dimension(); + MT V(0, d); + MT V1 = P1.get_mat(); + MT V2 = P2.get_mat(); + VT itervec(d); + std::vector temp_p(d, 0.0); + typename std::vector::iterator pit; + int j; + for (int i = 0; i < V1.rows(); ++i) { + pit = temp_p.begin(); + j = 0; + for (; pit!=temp_p.end(); ++pit, ++j) { + *pit = V1(i,j); + itervec(j) = V1(i,j); + } + Point p(d, temp_p.begin(), temp_p.end()); + if (P2.is_in(p) == -1) { + V.conservativeResize(V.rows() + 1, V.cols()); + V.row(V.rows()-1) = itervec; + num_of_v++; + } + } + + for (int i = 0; i < V2.rows(); ++i) { + pit = temp_p.begin(); + j = 0; + for (; pit!=temp_p.end(); ++pit, ++j) { + *pit = V2(i,j); + itervec(j) = V2(i,j); + } + Point p(d, temp_p.begin(), temp_p.end()); + if (P1.is_in(p) == -1) { + V.conservativeResize(V.rows() + 1, V.cols()); + V.row(V.rows()-1) = itervec; + num_of_v++; + } + } + if (num_of_v <= d) { + std::cout<<"no simplex"< res; + res.second = -1.0; + return res; + } + + VPolytope Q; + Q.init(d, V, itervec); + return Q.ComputeInnerBall(); + }*/ + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v) const { + + std::pair P1pair = P1.line_intersect(r, v); + std::pair P2pair = P2.line_intersect(r, v); + return std::pair(std::min(P1pair.first, P2pair.first), + std::max(P1pair.second, P2pair.second)); + + } + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av) const { + return line_intersect(r, v); + } + + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av, const NT &lambda) const { + return line_intersect(r, v); + } + + std::pair line_positive_intersect(const Point &r, const Point &v) const { + + std::pair P1pair = P1.line_positive_intersect(r, v); + std::pair P2pair = P2.line_positive_intersect(r, v); + + if(P1pair.first < P2pair.first) { + return std::pair(P1pair.first, 1); + } + return std::pair(P2pair.first, 2); + } + + std::pair line_positive_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av) const { + return line_positive_intersect(r, v); + } + + + std::pair line_positive_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av, const NT &lambda_prev) const { + return line_positive_intersect(r, v);//, Ar, Av); + } + + //------------------------------accelarated billiard------------------------------// + template + std::pair line_first_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + update_parameters& params) const + { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters& params) const + { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters& params) const + { + return line_positive_intersect(r, v); + } + //------------------------------------------------------------------------------// + + + // Compute the intersection of a coordinate ray + // with the V-polytope + std::pair line_intersect_coord(const Point &r, + const unsigned int &rand_coord, + const VT &lamdas) const { + std::pair P1pair = P1.line_intersect_coord(r, rand_coord, lamdas); + std::pair P2pair = P2.line_intersect_coord(r, rand_coord, lamdas); + return std::pair(std::min(P1pair.first, P2pair.first), + std::max(P1pair.second, P2pair.second)); + } + + + // Compute the intersection of a coordinate ray + // with the V-polytope + std::pair line_intersect_coord(const Point &r, + const Point &r_prev, + const unsigned int &rand_coord, + const unsigned int &rand_coord_prev, + const VT &lamdas) const { + return line_intersect_coord(r, rand_coord, lamdas); + } + + + //------------------------------oracles for exponential sampling---------------////// + + // compute intersection points of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + NT const& lambda_prev, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + //------------oracle for exact hmc spherical gaussian sampling---------------// + std::pair trigonometric_positive_intersect(Point const& r, Point const& v, + NT const& omega, int &facet_prev) const + { + return std::make_pair(0, 0); + } + + // shift polytope by a point c + void shift(const VT &c) { + P1.shift(c); + P2.shift(c); + } + + + // apply linear transformation, of square matrix T, to the V-Polytope + void linear_transformIt(const MT &T) { + P1.linear_transformIt(T); + P2.linear_transformIt(T); + } + + std::vector get_dists(const NT &radius) const { + std::vector res(upper_bound_of_hyperplanes(), radius); + return res; + } + + template + bool get_points_for_rounding (PointList &randPoints) { + if (num_of_vertices()>40*dimension()) { + return false; + } + if(!P1.get_points_for_rounding(randPoints)) { + return false; + } + if(!P2.get_points_for_rounding(randPoints)) { + return false; + } + + return true; + } + + void normalize() {} + + void resetFlags() {} + + void update_position_internal(NT&){} + + void compute_reflection (Point &v, const Point &p, const int &facet) const { + + if (facet == 1) { + P1.compute_reflection (v, p, facet); + } else { + P1.compute_reflection (v, p, facet); + } + + } + + template + void compute_reflection (Point &v, const Point &p, update_parameters const& params) const { + + if (params.facet_prev == 1) { + P1.compute_reflection (v, p, params); + } else { + P2.compute_reflection (v, p, params); + } + + } + +}; + + +#endif diff --git a/src/volesti/include/convex_bodies/vpolytope.h b/src/volesti/include/convex_bodies/vpolytope.h new file mode 100644 index 00000000..82ca6689 --- /dev/null +++ b/src/volesti/include/convex_bodies/vpolytope.h @@ -0,0 +1,648 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VPOLYTOPE_H +#define VPOLYTOPE_H + +#include +#include +#include + +#include "lp_oracles/vpolyoracles.h" +#include + + +/// This class describes a polytope in V-representation or an V-polytope +/// i.e. a polytope defined as a convex combination of points +/// \tparam Point Point type +template +class VPolytope { +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + +private: + unsigned int _d; //dimension + MT V; //matrix V. Each row contains a vertex + VT b; // vector b that contains first column of ine file + std::pair _inner_ball; + + // TODO: Why don't we use std::vector and std::vector for these pointers? + REAL *conv_comb, *conv_comb2, *conv_mem, *row; + int *colno, *colno_mem; + +public: + VPolytope() {} + + VPolytope(const unsigned int &dim, const MT &_V, const VT &_b): + _d{dim}, V{_V}, b{_b}, + conv_comb{new REAL[V.rows() + 1]}, + conv_comb2{new REAL[V.rows() + 1]}, + conv_mem{new REAL[V.rows()]}, + row{new REAL[V.rows() + 1]}, + colno{new int[V.rows() + 1]}, + colno_mem{new int[V.rows()]} + { + } + + // Construct matrix V which contains the vertices row-wise + // TODO: change rows; + VPolytope(std::vector> const& Pin) + { + _d = Pin[0][1] - 1; + V.resize(Pin.size() - 1, _d); + b.resize(Pin.size() - 1); + for (unsigned int i = 1; i < Pin.size(); i++) { + b(i - 1) = Pin[i][0]; + unsigned int j; + for (j = 1; j < _d + 1; j++) { + V(i - 1, j - 1) = Pin[i][j]; + } + } + conv_comb = new REAL[Pin.size()]; + conv_comb2 = new REAL[Pin.size()]; + conv_mem = new REAL[V.rows()]; + row = new REAL[V.rows() + 1]; + colno = new int[V.rows() + 1]; + colno_mem = new int[V.rows()]; + } + + template + void copy_array(T* source, T* result, size_t count) + { + T* tarray; + tarray = new T[count]; + std::copy_n(source, count, tarray); + delete [] result; + result = tarray; + } + + VPolytope& operator=(const VPolytope& other) + { + if (this != &other) { // protect against invalid self-assignment + _d = other._d; + V = other.V; + b = other.b; + + copy_array(other.conv_comb, conv_comb, V.rows() + 1); + copy_array(other.conv_comb2, conv_comb2, V.rows() + 1); + copy_array(other.conv_mem, conv_mem, V.rows()); + copy_array(other.row, row, V.rows() + 1); + copy_array(other.colno, colno, V.rows() + 1); + copy_array(other.colno_mem, colno_mem, V.rows()); + } + return *this; + } + + VPolytope& operator=(VPolytope&& other) + { + if (this != &other) { // protect against invalid self-assignment + _d = other._d; + V = other.V; + b = other.b; + + conv_comb = other.conv_comb; other.conv_comb = nullptr; + conv_comb2 = other.conv_comb2; other.conv_comb2 = nullptr; + conv_mem = other.conv_mem; other.conv_mem = nullptr; + row = other.row; other.row = nullptr; + colno = other.colno; colno = nullptr; + colno_mem = other.colno_mem; colno_mem = nullptr; + } + return *this; + } + + + VPolytope(const VPolytope& other) : + _d{other._d}, V{other.V}, b{other.b}, + conv_comb{new REAL[V.rows() + 1]}, + conv_comb2{new REAL[V.rows() + 1]}, + conv_mem{new REAL[V.rows()]}, + row{new REAL[V.rows() + 1]}, + colno{new int[V.rows() + 1]}, + colno_mem{new int[V.rows()]} + { + std::copy_n(other.conv_comb, V.rows() + 1, conv_comb); + std::copy_n(other.conv_comb2, V.rows() + 1, conv_comb2); + std::copy_n(other.conv_mem, V.rows(), conv_mem); + std::copy_n(other.row, V.rows() + 1, row); + std::copy_n(other.colno, V.rows() + 1, colno); + std::copy_n(other.colno_mem, V.rows(), colno_mem); + } + + VPolytope(VPolytope&& other) : + _d{other._d}, V{other.V}, b{other.b}, + conv_comb{nullptr}, conv_comb2{nullptr}, conv_mem{nullptr}, row{nullptr}, + colno{nullptr}, colno_mem{nullptr} + { + conv_comb = other.conv_comb; other.conv_comb = nullptr; + conv_comb2 = other.conv_comb2; other.conv_comb2 = nullptr; + conv_mem = other.conv_mem; other.conv_mem = nullptr; + row = other.row; other.row = nullptr; + colno = other.colno; colno = nullptr; + colno_mem = other.colno_mem; colno_mem = nullptr; + } + + ~VPolytope() { + delete [] conv_comb; + delete [] conv_comb2; + delete [] colno; + delete [] colno_mem; + delete [] row; + delete [] conv_mem; + } + + std::pair InnerBall() const + { + return _inner_ball; + } + + void set_InnerBall(std::pair const& innerball) //const + { + _inner_ball = innerball; + } + + void set_interior_point(Point const& r) + { + _inner_ball.first = r; + } + + // return dimension + unsigned int dimension() const { + return _d; + } + + + // this function returns 0. The main sampler requests this function to set the length of lambdas vector + int num_of_hyperplanes() const { + return 0; + } + + int num_of_generators() const { + return 0; + } + + // compute the number of facets of the cyclic polytope in dimension _d with the same number of vertices + // this is an upper bound for the number of the facets from McMullen's Upper Bound Theorem + unsigned int upper_bound_of_hyperplanes() const { + return 2 * _d; + } + + + // return the number of vertices + int num_of_vertices() const { + return V.rows(); + } + + // return the matrix V + MT get_mat() const { + return V; + } + + // return the vector b + VT get_vec() const { + return b; + } + + // change the matrix V + void set_mat(const MT &V2) { + V = V2; + } + + // change the vector b + void set_vec(const VT &b2) { + b = b2; + } + + MT get_T() const { + return V; + } + + + // print polytope in input format + void print() { + std::cout << " " << V.rows() << " " << _d << " float" << std::endl; + for (unsigned int i = 0; i < V.rows(); i++) { + for (unsigned int j = 0; j < _d; j++) { + std::cout << V(i, j) << " "; + } + std::cout<<"\n"; + } + } + + Point get_mean_of_vertices() { + Point xc(_d); + for (int i = 0; i < num_of_vertices(); ++i) { + xc.add(V.row(i)); + } + xc *= (1.0/NT(num_of_vertices())); + + return xc; + } + + + NT get_max_vert_norm() { + NT rad =0.0; + NT rad_iter; + for (int i = 0; i < num_of_vertices(); ++i) { + rad_iter = V.row(i).norm(); + if(rad_iter>rad)rad = rad_iter; + } + return rad; + } + + void normalize() {} + + // take d+1 points as input and compute the chebychev ball of the defined simplex + // done is true when the simplex is full dimensional and false if it is not + std::pair get_center_radius_inscribed_simplex(const typename std::vector::iterator it_beg, + const typename std::vector::iterator it_end) { + + Point p0 = *it_beg,p1,c; + unsigned int dim = p0.dimension(), i, j; + std::vector temp_p; + NT radius = 0.0, gi, sum = 0.0; + MT B(dim,dim), Bg(dim,dim), e(1,dim); + VT row(dim), g(dim); + std::pair result; + + for (j=1; jgetCoefficients() - p0.getCoefficients(); + } + + Bg = B; + B = B.inverse(); + for (i=0; i ComputeInnerBall() { + + std::vector verts(_d+1); + std::vector vecp(_d); + unsigned int vert_rand, pointer=0; + unsigned int i,j; + int m = num_of_vertices(); + std::vector x_vec(_d); + bool done=false; + unsigned seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(seed); + boost::random::uniform_int_distribution<> uidist(1, m); + + std::pair res; + // while d+1 points do not define a full dimensional simplex repeat + while(!done){ + pointer=0; + x_vec.assign(_d+1,0); + while(pointer!=_d+1) { + vert_rand = uidist(rng); + // Check if this vertex is selected first time + if (std::find(x_vec.begin(), x_vec.begin() + pointer, vert_rand) == x_vec.begin() + pointer) { + x_vec[pointer] = vert_rand; + pointer++; + } + } + + for(i=0; i<(_d+1); i++){ + for (j=0; j<_d; j++) { + vecp[j] = V(x_vec[i] - 1, j); + } + verts[i] = Point(_d,vecp.begin(),vecp.end()); + } + res=get_center_radius_inscribed_simplex(verts.begin(), verts.end(), done); + } + return res; + }*/ + + + std::pair ComputeInnerBall() { + + NT radius = std::numeric_limits::max(), min_plus; + Point center(_d); + + std::list randPoints; + if (!get_points_for_rounding(randPoints)) { + center = get_mean_of_vertices(); + } else { + + MT Ap(_d,randPoints.size()); + typename std::list::iterator rpit=randPoints.begin(); + + unsigned int i, j = 0; + for ( ; rpit!=randPoints.end(); rpit++, j++) { + const NT* point_data = rpit->getCoefficients().data(); + + for ( i=0; i < rpit->dimension(); i++){ + Ap(i,j)=double(*point_data); + point_data++; + } + } + MT Q(_d, _d); + VT c2(_d); + size_t w=1000; + + KhachiyanAlgo(Ap,0.01,w,Q,c2); // call Khachiyan algorithm + + //Get ellipsoid matrix and center as Eigen objects + for(unsigned int i=0; i<_d; i++) center.set_coord(i, NT(c2(i))); + } + + std::pair res; + Point v(_d); + for (unsigned int i = 0; i < _d; ++i) { + v.set_to_origin(); + v.set_coord(i, 1.0); + res = intersect_double_line_Vpoly(V, center, v, row, colno); + min_plus = std::min(res.first, -1.0*res.second); + if (min_plus < radius) radius = min_plus; + } + + radius = radius / std::sqrt(NT(_d)); + _inner_ball = std::pair (center, radius); + return std::pair (center, radius); + } + + + // check if point p belongs to the convex hull of V-Polytope P + int is_in(const Point &p, NT tol=NT(0)) const { + if (memLP_Vpoly(V, p, conv_mem, colno_mem)){ + return -1; + } + return 0; + } + + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v) const { + + return intersect_double_line_Vpoly(V, r, v, row, colno); + } + + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av) const { + return intersect_double_line_Vpoly(V, r, v, row, colno); + } + + // compute intersection point of ray starting from r and pointing to v + // with the V-polytope + std::pair line_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av, const NT &lambda_prev) const { + + return intersect_double_line_Vpoly(V, r, v, row, colno); + } + + + std::pair line_positive_intersect(const Point &r, const Point &v) const { + return std::pair (intersect_line_Vpoly(V, r, v, conv_comb, row, colno, false, false), 1); + } + + std::pair line_positive_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av) const { + return line_positive_intersect(r, v); + } + + + std::pair line_positive_intersect(const Point &r, const Point &v, const VT &Ar, + const VT &Av, const NT &lambda_prev) const { + return line_positive_intersect(r, v); + } + + //-------------------------accelarated billiard--------------------------------// + template + std::pair line_first_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + update_parameters ¶ms) const + { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters ¶ms) const + { + return line_positive_intersect(r, v); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters ¶ms) const + { + return line_positive_intersect(r, v); + } + //------------------------------------------------------------------------------// + + + // Compute the intersection of a coordinate ray + // with the V-polytope + std::pair line_intersect_coord(const Point &r, + const unsigned int rand_coord, + const VT &lamdas) const { + Point v(_d); + v.set_coord(rand_coord, 1.0); + return intersect_double_line_Vpoly(V, r, v, row, colno); + } + + + // Compute the intersection of a coordinate ray + // with the V-polytope + std::pair line_intersect_coord(const Point &r, + const Point &r_prev, + const unsigned int rand_coord, + const unsigned int rand_coord_prev, + const VT &lamdas) const { + return line_intersect_coord(r, rand_coord, lamdas); + } + + + //------------------------------oracles for exponential sampling---------------////// + + // compute intersection points of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + NT const& lambda_prev, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + + //------------oracle for exact hmc spherical gaussian sampling---------------// + std::pair trigonometric_positive_intersect(Point const& r, Point const& v, + NT const& omega, int &facet_prev) const + { + return std::make_pair(0, 0); + } + + + // shift polytope by a point c + void shift(const VT &c) { + MT V2 = V.transpose().colwise() - c; + V = V2.transpose(); + } + + + // apply linear transformation, of square matrix T, to the V-Polytope + void linear_transformIt(const MT &T) { + MT V2 = T.inverse() * V.transpose(); + V = V2.transpose(); + } + + + // consider an upper bound for the number of facets of a V-polytope + // for each facet consider a lower bound for the distance from the origin + // useful for CV algorithm to get the first gaussian + std::vector get_dists(const NT &radius) const { + std::vector res(upper_bound_of_hyperplanes(), radius); + return res; + } + + + // in number_of_vertices<=20*dimension use the vertices for the rounding + // otherwise you have to sample from the V-polytope + template + bool get_points_for_rounding (PointList &randPoints) { + if (num_of_vertices()>20*_d) { + return false; + } + unsigned int j; + + typename std::vector::iterator pointIt; + for (int i=0; i 0.0) { + Fmat2.row(count) = V.row(j); + count++; + } else { + outvert = j; + } + } + + VT a = Fmat2.colPivHouseholderQr().solve(VT::Ones(_d)); + if (a.dot(V.row(outvert)) > 1.0) a = -a; + a /= a.norm(); + + // compute reflection + a *= (-2.0 * v.dot(a)); + v += a; + } + + template + void compute_reflection(Point &v, const Point &p, update_parameters const& params) const { + + int count = 0, outvert; + MT Fmat2(_d,_d); + for (int j = 0; j < num_of_vertices(); ++j) { + if (*(conv_comb + j) > 0.0) { + Fmat2.row(count) = V.row(j); + count++; + } else { + outvert = j; + } + } + + VT a = Fmat2.colPivHouseholderQr().solve(VT::Ones(_d)); + if (a.dot(V.row(outvert)) > 1.0) a *= -1.0; + a /= a.norm(); + + // compute reflection + a *= (-2.0 * v.dot(a)); + v += a; + } + + void resetFlags() {} + + void update_position_internal(NT&){} + + template + std::tuple curve_intersect( + NT t_prev, + NT t0, + NT eta, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + NonLinearOracle &intersection_oracle, + int ignore_facet=-1) + { + return intersection_oracle.apply(t_prev, t0, eta, V, *this, + coeffs, phi, grad_phi, ignore_facet); + } + + +}; + +#endif diff --git a/src/volesti/include/convex_bodies/zonoIntersecthpoly.h b/src/volesti/include/convex_bodies/zonoIntersecthpoly.h new file mode 100644 index 00000000..bbad43a6 --- /dev/null +++ b/src/volesti/include/convex_bodies/zonoIntersecthpoly.h @@ -0,0 +1,272 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +#ifndef ZONOINTERSECTHPOLY_H +#define ZONOINTERSECTHPOLY_H + +/// This class represents the intersection of a Zonotope with an H-polytope +/// \tparam Zonotope Zonotope Type +/// \tparam HPolytope HPolytope Type +template +class ZonoIntersectHPoly { +private: + Zonotope Z; + HPolytope HP; +public: + typedef typename HPolytope::NT NT; + typedef typename HPolytope::VT VT; + typedef typename HPolytope::MT MT; + typedef typename Zonotope::PointType PointType; + + ZonoIntersectHPoly() + {} + + ZonoIntersectHPoly(Zonotope &Z1, HPolytope &HP1) + : Z(Z1) + , HP(HP1) + {} + + Zonotope first() const { return Z; } + HPolytope second() const { return HP; } + + int is_in(const PointType &p) const { + if(HP.is_in(p)==-1) + return Z.is_in(p); + return 0; + } + + int num_of_hyperplanes() const { + return HP.num_of_hyperplanes(); + } + + unsigned int dimension() const { + return HP.dimension(); + } + + unsigned int num_of_generators() const { + return Z.num_of_generators(); + } + + MT get_mat() const { + return HP.get_mat(); + } + + MT get_T() const { + return Z.get_mat(); + } + + MT get_vec() const { + return HP.get_vec(); + } + + std::pair InnerBall() const + { + return Z.InnerBall(); + } + + std::pair line_intersect(PointType const& r, + PointType const& v) const + { + + std::pair polypair = HP.line_intersect(r, v); + std::pair zonopair = Z.line_intersect(r, v); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av) const + { + std::pair polypair = HP.line_intersect(r, v, Ar, Av); + std::pair zonopair = Z.line_intersect(r, v); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + std::pair line_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev) const + { + std::pair polypair = HP.line_intersect(r, v, Ar, Av, lambda_prev); + std::pair zonopair = Z.line_intersect(r, v); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + //First coordinate ray shooting intersecting convex body + std::pair line_intersect_coord(PointType const& r, + unsigned int const& rand_coord, + VT& lamdas) const + { + std::pair polypair = HP.line_intersect_coord(r, rand_coord, lamdas); + std::pair zonopair = Z.line_intersect_coord(r, rand_coord, lamdas); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av) const + { + std::pair polypair = HP.line_positive_intersect(r, v, Ar, Av); + std::pair zonopair = Z.line_positive_intersect(r, v, Ar, Av); + int facet = HP.num_of_hyperplanes()+1; + + if (polypair.first < zonopair.first ) facet = polypair.second; + + return std::pair(std::min(polypair.first, zonopair.first), facet); + } + + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev) const + { + std::pair polypair = HP.line_positive_intersect(r, v, Ar, Av, + lambda_prev); + std::pair zonopair = Z.line_positive_intersect(r, v, Ar, Av); + int facet = HP.num_of_hyperplanes()+1; + + if (polypair.first < zonopair.first ) facet = polypair.second; + + return std::pair(std::min(polypair.first, zonopair.first), + facet); + } + + //---------------------accelarated billiard---------------------// + template + std::pair line_first_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + update_parameters& params) const + { + std::pair polypair = HP.line_first_positive_intersect(r, v, Ar, Av, params); + std::pair zonopair = Z.line_positive_intersect(r, v, Ar, Av); + int facet = HP.num_of_hyperplanes(); + params.facet_prev = polypair.second; + + if (polypair.first < zonopair.first ) { + facet = polypair.second; + params.hit_ball = false; + } else { + params.hit_ball = true; + } + + return std::pair(std::min(polypair.first, zonopair.first), facet); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters& params) const + { + std::pair polypair = HP.line_positive_intersect(r, v, Ar, Av, lambda_prev, params); + std::pair zonopair = Z.line_positive_intersect(r, v, Ar, Av); + int facet = HP.num_of_hyperplanes(); + params.facet_prev = polypair.second; + + if (polypair.first < zonopair.first ) { + facet = polypair.second; + params.hit_ball = false; + } else { + params.hit_ball = true; + } + + return std::pair(std::min(polypair.first, zonopair.first), facet); + } + + template + std::pair line_positive_intersect(PointType const& r, + PointType const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters& params) const + { + std::pair polypair = HP.line_positive_intersect(r, v, Ar, Av, lambda_prev, params); + std::pair zonopair = Z.line_positive_intersect(r, v, Ar, Av); + int facet = HP.num_of_hyperplanes(); + params.facet_prev = polypair.second; + + if (polypair.first < zonopair.first ) { + facet = polypair.second; + params.hit_ball = false; + } else { + params.hit_ball = true; + } + + return std::pair(std::min(polypair.first, zonopair.first), facet); + } +//-------------------------------------------------------------------------// + + //Not the first coordinate ray shooting intersecting convex body + std::pair line_intersect_coord(PointType const& r, + PointType const& r_prev, + unsigned int const& rand_coord, + unsigned int const& rand_coord_prev, + VT& lamdas) const + { + + std::pair polypair = HP.line_intersect_coord(r, r_prev, + rand_coord, + rand_coord_prev, + lamdas); + std::pair zonopair = Z.line_intersect_coord(r, rand_coord, + lamdas); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + std::pair query_dual(PointType const& p, + unsigned int const& rand_coord) const + { + std::pair polypair = HP.query_dual(p, rand_coord); + std::pair zonopair = Z.line_intersect_coord(p, rand_coord); + return std::pair(std::min(polypair.first, zonopair.first), + std::max(polypair.second, zonopair.second)); + } + + void compute_reflection (PointType &v, + PointType const& p, + int const& facet) const + { + if (facet == (HP.num_of_hyperplanes()+1)) { + Z.compute_reflection(v, p, facet); + } else { + HP.compute_reflection(v, p, facet); + } + + } + + template + void compute_reflection (PointType &v, const PointType &p, update_parameters const& params) const { + + if (params.hit_ball) { + Z.compute_reflection(v, p, params); + } else { + HP.compute_reflection(v, p, params.facet_prev); + } + + } + +}; + +#endif diff --git a/src/volesti/include/convex_bodies/zpolytope.h b/src/volesti/include/convex_bodies/zpolytope.h new file mode 100644 index 00000000..1ad8868e --- /dev/null +++ b/src/volesti/include/convex_bodies/zpolytope.h @@ -0,0 +1,619 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018-19 programs. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ZPOLYTOPE_H +#define ZPOLYTOPE_H + +#include + +#include +#include +#include "lp_oracles/vpolyoracles.h" +#include "lp_oracles/zpolyoracles.h" + +/// This class describes a zonotope i.e. the Minkowski sum of a set of line segments +/// \tparam Point Point type +template +class Zonotope { +public: + typedef Point PointType; + typedef typename Point::FT NT; + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + +private: + unsigned int _d; //dimension + MT V; //matrix V. Each row contains a vertex + VT b; // vector b that contains first column of ine file + MT T; + std::pair _inner_ball; + NT maxNT = std::numeric_limits::max(); + NT minNT = std::numeric_limits::lowest(); + + REAL *conv_comb, *row_mem, *row; + int *colno, *colno_mem; + MT sigma; + MT Q0; + + +public: + + Zonotope() {} + + Zonotope(const unsigned int &dim, const MT &_V, const VT &_b): + _d{dim}, V{_V}, b{_b}, + conv_comb{new REAL[V.rows() + 1]}, + row_mem{new REAL[V.rows()]}, + row{new REAL[V.rows() + 1]}, + colno{new int[V.rows() + 1]}, + colno_mem{new int[V.rows()]} + { + compute_eigenvectors(V.transpose()); + } + + /*Zonotope(unsigned int const dim, MT const& _V, VT const& _b) + { + _d = dim; + V = _V; + b = _b; + + conv_comb = new REAL[V.rows()+1]; + row_mem = new REAL[V.rows()]; + row = new REAL[V.rows() + 1]; + colno = new int[V.rows() + 1]; + colno_mem = new int[V.rows()]; + + //conv_comb = (REAL *) malloc((V.rows()+1) * sizeof(*conv_comb)); + //colno = (int *) malloc((V.rows()+1) * sizeof(*colno)); + //row = (REAL *) malloc((V.rows()+1) * sizeof(*row)); + //colno_mem = (int *) malloc((V.rows()) * sizeof(*colno_mem)); + //row_mem = (REAL *) malloc((V.rows()) * sizeof(*row_mem)); + + compute_eigenvectors(V.transpose()); + }*/ + + Zonotope(std::vector > const& Pin) + { + _d = Pin[0][1] - 1; + V.resize(Pin.size() - 1, _d); + b.resize(Pin.size() - 1); + for (unsigned int i = 1; i < Pin.size(); i++) + { + b(i - 1) = Pin[i][0]; + for (unsigned int j = 1; j < _d + 1; j++) + { + V(i - 1, j - 1) = Pin[i][j]; + } + } + + conv_comb = new REAL[Pin.size()]; + row_mem = new REAL[V.rows()]; + row = new REAL[V.rows() + 1]; + colno = new int[V.rows() + 1]; + colno_mem = new int[V.rows()]; + + compute_eigenvectors(V.transpose()); + } + + template + void copy_array(T* source, T* result, size_t count) + { + T* tarray; + tarray = new T[count]; + std::copy_n(source, count, tarray); + delete [] result; + result = tarray; + } + + Zonotope& operator=(const Zonotope& other) + { + if (this != &other) { // protect against invalid self-assignment + _d = other._d; + V = other.V; + b = other.b; + T = other.T; + + copy_array(other.conv_comb, conv_comb, V.rows() + 1); + copy_array(other.row_mem, row_mem, V.rows()); + copy_array(other.row, row, V.rows() + 1); + copy_array(other.colno, colno, V.rows() + 1); + copy_array(other.colno_mem, colno_mem, V.rows()); + } + return *this; + } + + Zonotope& operator=(Zonotope&& other) + { + if (this != &other) { // protect against invalid self-assignment + _d = other._d; + V = other.V; + b = other.b; + T = other.T; + + conv_comb = other.conv_comb; other.conv_comb = nullptr; + row_mem = other.row_mem; other.row_mem = nullptr; + row = other.row; other.row = nullptr; + colno = other.colno; colno = nullptr; + colno_mem = other.colno_mem; colno_mem = nullptr; + } + return *this; + } + + + Zonotope(const Zonotope& other) : + _d{other._d}, V{other.V}, b{other.b}, T{other.T}, + conv_comb{new REAL[V.rows() + 1]}, + row_mem{new REAL[V.rows()]}, + row{new REAL[V.rows() + 1]}, + colno{new int[V.rows() + 1]}, + colno_mem{new int[V.rows()]} + { + std::copy_n(other.conv_comb, V.rows() + 1, conv_comb); + std::copy_n(other.row_mem, V.rows(), row_mem); + std::copy_n(other.row, V.rows() + 1, row); + std::copy_n(other.colno, V.rows() + 1, colno); + std::copy_n(other.colno_mem, V.rows(), colno_mem); + } + + Zonotope(Zonotope&& other) : + _d{other._d}, V{other.V}, b{other.b}, T{other.T}, + conv_comb{nullptr}, row_mem{nullptr}, row{nullptr}, + colno{nullptr}, colno_mem{nullptr} + { + conv_comb = other.conv_comb; other.conv_comb = nullptr; + row_mem = other.row_mem; other.row_mem = nullptr; + row = other.row; other.row = nullptr; + colno = other.colno; colno = nullptr; + colno_mem = other.colno_mem; colno_mem = nullptr; + } + + ~Zonotope() { + delete [] conv_comb; + delete [] colno; + delete [] colno_mem; + delete [] row; + delete [] row_mem; + } + + void set_interior_point(Point const& r) + { + _inner_ball.first = r; + } + + // return the dimension + unsigned int dimension() const + { + return _d; + } + + // this function returns 0. The main sampler requests this function to set the length of lambdas vector + int num_of_hyperplanes() const + { + return 0; + } + + + // return the number of parallelopipeds. Used in get_dists fnction. + unsigned int upper_bound_of_hyperplanes() const + { + return 2*_d; + } + + void compute_eigenvectors(MT const& G) + { + + int k = G.cols(); + MT ps = G; + sigma.resize(k,k); + sigma = ps.transpose()*ps; + sigma = (sigma + sigma.transpose()) / 2; + Eigen::SelfAdjointEigenSolver es(sigma); + + MT D = es.eigenvalues().asDiagonal(); + MT Q2 = es.eigenvectors(); + + Q0.resize(k,k-_d); + int count=0; + for (int i = 0; i < k; ++i) + { + if (es.eigenvalues()[i]<0.0000001) + { + for (int j = 0; j < k; ++j) + { + Q0(j, count) = Q2(j, i); + } + count++; + } + } + Eigen::JacobiSVD svd(Q0, Eigen::ComputeFullU | Eigen::ComputeFullV); + MT T2 = svd.matrixU().transpose(); + T.resize(_d,k); + for (int i = k-_d; i < k; ++i) + { + for (int j = 0; j < k; ++j) + { + T(i-k+_d,j) = T2(i,j); + } + } + + for (int i1 = 0; i1 < k; ++i1) + { + sigma(i1,i1) = sigma(i1,i1) + 0.00000001; + } + } + + MT get_T() const + { + return T; + } + + MT get_Q0() const + { + return Q0; + } + + MT get_sigma() const + { + return sigma; + } + + // return the number of vertices + int num_of_vertices() const + { + return V.rows(); + } + + std::pair InnerBall() const + { + return _inner_ball; + } + + void set_InnerBall(std::pair const& innerball) //const + { + _inner_ball = innerball; + } + + // return the number of generators + int num_of_generators() const + { + return V.rows(); + } + + // return the matrix V + MT get_mat() const + { + return V; + } + + // return the vector b + VT get_vec() const + { + return b; + } + + // change the matrix V + void set_mat(MT const& V2) + { + V = V2; + } + + // change the vector b + void set_vec(VT const& b2) + { + b = b2; + } + + Point get_mean_of_vertices() const + { + return Point(_d); + } + + + NT get_max_vert_norm() const + { + return 0.0; + } + + + // print polytope in input format + void print() + { +#ifdef VOLESTI_DEBUG + std::cout << " " << V.rows() << " " << _d << " float" << std::endl; +#endif + for (unsigned int i = 0; i < V.rows(); i++) { + for (unsigned int j = 0; j < _d; j++) { +#ifdef VOLESTI_DEBUG + std::cout << V(i, j) << " "; +#endif + } +#ifdef VOLESTI_DEBUG + std::cout<<"\n"; +#endif + } + } + + + // check if point p belongs to the convex hull of V-Polytope P + int is_in(Point const& p, NT tol=NT(0)) const + { + if(memLP_Zonotope(V, p, row_mem, colno_mem)) + { + return -1; + } + return 0; + } + + + // Compute an inner ball of the zonotope + std::pair ComputeInnerBall() + { + std::vector temp(_d,0); + NT radius = maxNT, min_plus; + Point center(_d); + + for (unsigned int i = 0; i < _d; ++i) { + temp.assign(_d,0); + temp[i] = 1.0; + Point v(_d,temp.begin(), temp.end()); + min_plus = intersect_line_Vpoly(V, center, v, conv_comb, + row, colno, false, true); + if (min_plus < radius) radius = min_plus; + } + + radius = radius / std::sqrt(NT(_d)); + _inner_ball = std::pair (center, radius); + return _inner_ball; + } + + // compute intersection point of ray starting from r and pointing to v + // with the Zonotope + std::pair line_intersect(Point const& r, Point const& v) const + { + return intersect_line_zono(V, r, v, conv_comb, colno); + } + + + // compute intersection point of ray starting from r and pointing to v + // with the Zonotope + std::pair line_intersect(Point const& r, + Point const& v, + VT const& Ar, + VT const& Av) const + { + return intersect_line_zono(V, r, v, conv_comb, colno); + } + + // compute intersection point of ray starting from r and pointing to v + // with the Zonotope + std::pair line_intersect(Point const& r, + Point const& v, + VT const& Ar, + VT const& Av, + NT const& lambda_prev) const + { + return intersect_line_zono(V, r, v, conv_comb, colno); + } + + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT const& Ar, + VT const& Av) const + { + return std::pair (intersect_line_Vpoly(V, r, v, conv_comb, + row, colno, + false, true), 1); + } + + + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT const& Ar, + VT const& Av, + NT const& lambda_prev) const + { + return line_positive_intersect(r, v, Ar, Av); + } + + //---------------------------accelarated billiard-----------------------------// + template + std::pair line_first_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + update_parameters& params) const + { + return line_positive_intersect(r, v, Ar, Av); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + MT const& AA, + update_parameters& params) const + { + return line_positive_intersect(r, v, Ar, Av); + } + + template + std::pair line_positive_intersect(Point const& r, + Point const& v, + VT& Ar, + VT& Av, + NT const& lambda_prev, + update_parameters& params) const + { + return line_positive_intersect(r, v, Ar, Av); + } + //------------------------------------------------------------------------------// + + // Compute the intersection of a coordinate ray + // with the Zonotope + std::pair line_intersect_coord(Point const& r, + const unsigned int rand_coord, + VT const& lamdas) const + { + + std::vector temp(_d,0); + temp[rand_coord]=1.0; + Point v(_d,temp.begin(), temp.end()); + + return intersect_line_zono(V, r, v, conv_comb, colno); + + } + + + // Compute the intersection of a coordinate ray + // with the Zonotope + std::pair line_intersect_coord(Point const& r, + Point const& r_prev, + const unsigned int rand_coord, + const unsigned int rand_coord_prev, + VT const& lamdas) const + { + return line_intersect_coord(r, rand_coord, lamdas); + } + + + //------------------------------oracles for exponential sampling---------------////// + + // compute intersection points of a ray starting from r and pointing to v + // with polytope discribed by A and b + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + std::pair quadratic_positive_intersect(Point const& r, + Point const& v, + VT const& Ac, + NT const& T, + VT& Ar, + VT& Av, + NT const& lambda_prev, + int& facet_prev) const + { + throw std::runtime_error("Quadratic polynomial trajectories are supported only for H-polytopes"); + } + + + //------------oracle for exact hmc spherical gaussian sampling---------------// + std::pair trigonometric_positive_intersect(Point const& r, Point const& v, + NT const& omega, int &facet_prev) const + { + return std::make_pair(0, 0); + } + + // shift polytope by a point c + // vector c has to be always the zero vector + void shift(VT const& c) + { + return; + } + + + // get number of parallelopipeds + // for each parallelopiped consider a lower bound for the distance from the origin + // useful for CG algorithm to get the first gaussian + std::vector get_dists(NT const& radius) const + { + std::vector res(upper_bound_of_hyperplanes(), radius); + return res; + } + + // apply linear transformation, of square matrix T, to thr Zonotope + void linear_transformIt(MT const& T) + { + MT V2 = T.inverse() * V.transpose(); + V = V2.transpose(); + } + + // return false to the rounding function + // no points are given so they have o be sampled + template + bool get_points_for_rounding (T const& randPoints) + { + return false; + } + + void normalize() {} + + void compute_reflection(Point &v, Point const& p, int const& facet) const + { + //compute_reflection(v, p, 0.0); + + int count = 0; + MT Fmat(_d-1,_d); + const NT e = 0.0000000001; + for (int j = 0; j < num_of_generators(); ++j) + { + if (((1.0 - *(conv_comb + j) ) > e || (1.0 - *(conv_comb + j) ) + > e*std::abs(*(conv_comb + j))) && + ((1.0 + *(conv_comb + j) ) > e || (1.0 + *(conv_comb + j) ) + > e*std::abs(*(conv_comb + j)))) + { + Fmat.row(count) = V.row(j); + count++; + } + } + + VT a = Fmat.fullPivLu().kernel(); + + if(p.getCoefficients().dot(a) < 0.0) a *= -1.0; + + a = a/a.norm(); + + // compute reflection + a *= (-2.0 * v.dot(a)); + v += a; + } + + void resetFlags() {} + + void update_position_internal(NT&){} + + template + void compute_reflection(Point &v, const Point &p, update_parameters const& params) const { + + int count = 0; + MT Fmat(_d-1,_d); + const NT e = 0.0000000001; + for (int j = 0; j < num_of_generators(); ++j) { + if (((1.0 - *(conv_comb + j) ) > e || (1.0 - *(conv_comb + j) ) > e*std::abs(*(conv_comb + j))) && + ((1.0 + *(conv_comb + j) ) > e || (1.0 + *(conv_comb + j) ) > e*std::abs(*(conv_comb + j)))) { + Fmat.row(count) = V.row(j); + count++; + } + } + + VT a = Fmat.fullPivLu().kernel(); + + if(p.getCoefficients().dot(a) < 0.0) a *= -1.0; + + a = a/a.norm(); + + // compute reflection + a *= (-2.0 * v.dot(a)); + v += a; + } + +}; + +#endif diff --git a/src/volesti/include/diagnostics/diagnostics.hpp b/src/volesti/include/diagnostics/diagnostics.hpp new file mode 100644 index 00000000..64905875 --- /dev/null +++ b/src/volesti/include/diagnostics/diagnostics.hpp @@ -0,0 +1,24 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef DIAGNOSTICS_DIAGNOSTICS_HPP +#define DIAGNOSTICS_DIAGNOSTICS_HPP + +#include "misc/print_table.hpp" +#include "diagnostics/multivariate_psrf.hpp" +#include "diagnostics/univariate_psrf.hpp" +#include "diagnostics/interval_psrf.hpp" +#include "diagnostics/geweke.hpp" +#include "diagnostics/raftery.hpp" +#include "diagnostics/effective_sample_size.hpp" +#include "diagnostics/thin_samples.hpp" +#include "diagnostics/print_diagnostics.hpp" + +#endif diff --git a/src/volesti/include/diagnostics/effective_sample_size.hpp b/src/volesti/include/diagnostics/effective_sample_size.hpp new file mode 100644 index 00000000..c372eb69 --- /dev/null +++ b/src/volesti/include/diagnostics/effective_sample_size.hpp @@ -0,0 +1,123 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#include + +#ifndef DIAGNOSTICS_EFFECTIVE_SAMPLE_SIZE_HPP +#define DIAGNOSTICS_EFFECTIVE_SAMPLE_SIZE_HPP + +template +void cummulative_minimum(std::vector &v) { + unsigned int N = v.size(); + for (unsigned int i = 1; i < N; i++) { + if (v[i] > v[i - 1]) v[i] = v[i - 1]; + } +} + +template +VT effective_sample_size(MT const& samples, unsigned int &min_ess) { + typedef Eigen::FFT EigenFFT; + typedef std::complex CNT; + EigenFFT fft; + + // Sample matrix is provided as d x n_samples + unsigned int d = samples.rows(); + unsigned int N = samples.cols(); + unsigned int N_even = N - N % 2; + + // Calculate effective sample size per dimension + VT ess; + ess.resize(d); + + // Autocorrelation vector + std::vector autocorrelation(N_even, NT(0)); + std::vector min_auto_correlation(N_even / 2, NT(0)); + + + // Z-normalized samples + std::vector normalized_sample_row(2 * N); + + // FFT vector + std::vector fft_vec(2 * N); + std::vector fft_inv_vec(2 * N); + std::vector psd(2 * N); + + // Helper variables + NT row_mean; + NT variance; + NT gap; + + for (unsigned int i = 0; i < d; i++) { + + // Z-normalization + row_mean = samples.row(i).mean(); + + for (int j = 0; j < N; j++) { + normalized_sample_row[j] = samples(i, j) - row_mean; + // Zero-padding + normalized_sample_row[j + N] = NT(0); + } + + variance = NT(0); + + for (int j = 0; j < N; j++) { + variance += std::pow(normalized_sample_row[j], 2); + } + + variance *= (1.0 / N); + + for (int j = 0; j < N; j++) { + normalized_sample_row[j] /= NT(1e-16 + sqrt(variance)); + } + + // Perform FFT on 2N points + fft.fwd(fft_vec, normalized_sample_row); + + // Calculate PSD which is the norm squared of the FFT of the sequence + for (int j = 0; j < 2 * N; j++) { + psd[j].real(std::norm(fft_vec[j])); + psd[j].imag(NT(0)); + } + + // Invert fft to get autocorrelation function + fft.inv(fft_inv_vec, psd); + + for (unsigned int j = 0; j < N_even; j++) { + autocorrelation[j] = std::real(fft_inv_vec[j]) / N; + } + + // Calculate minimum autocorrelation + for (int j = 0; j < N_even; j += 2) { + min_auto_correlation[j/2] = autocorrelation[j] + autocorrelation[j + 1]; + } + + gap = NT(0); + cummulative_minimum(min_auto_correlation); + + for (int j = 0; j < N_even / 2; j++) { + if (min_auto_correlation[j] > 0) gap += min_auto_correlation[j]; + } + + gap = 2 * gap - NT(1); + if (gap < NT(1)) gap = NT(1); + + ess(i) = (1.0 * N) / gap; + + // Store minimum effective sample size as integer (in general ess is not int) + // for the thinning process + if (i == 0) min_ess = (unsigned int) ceil(ess(i)); + else if ((unsigned int)ceil(ess(i)) < min_ess) min_ess = (unsigned int)ceil(ess(i)); + + } + + return ess; +} + +#endif diff --git a/src/volesti/include/diagnostics/ess_updater_autocovariance.hpp b/src/volesti/include/diagnostics/ess_updater_autocovariance.hpp new file mode 100644 index 00000000..4023a51c --- /dev/null +++ b/src/volesti/include/diagnostics/ess_updater_autocovariance.hpp @@ -0,0 +1,69 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef DIAGNOSTICS_ESS_UPDATER_AUTOCOVARIANCE_HPP +#define DIAGNOSTICS_ESS_UPDATER_AUTOCOVARIANCE_HPP + +#include +#include +#include +#include +#include +#include + +/** + Compute autocovariance estimates for every lag for the input sequence + using the Geyer's stable estimator given in + Charles J. Geyer, Practical Markov Chain Monte Carlo, Statistical Science 1992. + + * @tparam NT number type + * @tparam VT vector type + * @param samples the sequence of correlated samples + * @param auto_cov the autocovariance estimates +*/ +template +void compute_autocovariance(VT const& samples, VT &auto_cov) +{ + const NT eps = 1e-16; + typedef Eigen::FFT EigenFFT; + typedef Eigen::Matrix, Eigen::Dynamic, 1> CVT; + EigenFFT fft; + + unsigned int N = samples.size(); + NT samples_mean = samples.mean(); + auto_cov.setZero(N); + + // compute normalized samples + VT normalized_samples(2 * N); + normalized_samples.setZero(); + normalized_samples.head(N) = samples.array() - samples_mean; + + NT variance = (normalized_samples.cwiseProduct(normalized_samples)).sum(); + variance *= (1.0 / N); + variance += eps * (samples_mean*samples_mean); + normalized_samples.head(N) = normalized_samples.head(N).array() / sqrt(variance); + + // Perform FFT on 2N points + CVT frequency(2 * N); + fft.fwd(frequency, normalized_samples); + + // Invert fft to get autocorrelation function + CVT auto_cov_tmp(2 * N); + frequency = frequency.cwiseAbs2(); + fft.inv(auto_cov_tmp, frequency); + + auto_cov = auto_cov_tmp.head(N).real().array() / N; + + boost::accumulators::accumulator_set> accumulator; + for (int i = 0; i < samples.size(); ++i) + { + accumulator(samples.coeff(i)); + } + + auto_cov = auto_cov.array() * boost::accumulators::variance(accumulator); +} + +#endif diff --git a/src/volesti/include/diagnostics/ess_window_updater.hpp b/src/volesti/include/diagnostics/ess_window_updater.hpp new file mode 100644 index 00000000..99edb5aa --- /dev/null +++ b/src/volesti/include/diagnostics/ess_window_updater.hpp @@ -0,0 +1,154 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef DIAGNOSTICS_ESS_UPDATER_HPP +#define DIAGNOSTICS_ESS_UPDATER_HPP + +#include "ess_updater_autocovariance.hpp" + + +/** + This is a class that updates the effective sample size (ess) of a sample given a new chain + using Welford's algorithm to update the average values and the variance estimates where needed. + The chains has to be of the same length. The ess estimation exploits Geyer's stable estimator + for the autocovariance and the Geyer's conversion to a monotone sequence, given in, + + Charles J. Geyer, Practical Markov Chain Monte Carlo, Statistical Science 1992. + + * @tparam NT number type + * @tparam VT vector type + * @tparam MT matrix type +*/ +template +class ESSestimator { + +private: + unsigned int num_draws, max_s, s, d, num_chains, jj; + VT cm_mean, cm_var, cv_mean, draws, var_plus, ess, auto_cov; + NT oldM, rho_hat_odd, rho_hat_even, mean_var, M2, delta, new_elem; + MT acov_s_mean, rho_hat_s; + +public: + ESSestimator() {} + + ESSestimator(unsigned int const& _ndraws, unsigned int const& _dim) + { + num_draws = _ndraws; + d = _dim; + num_chains = 0; + + cm_mean.setZero(d); + cm_var.setZero(d); + cv_mean.setZero(d); + var_plus.setZero(d); + ess.setZero(d); + draws.setZero(num_draws); + acov_s_mean.setZero(num_draws-3, d); + rho_hat_s.setZero(num_draws, d); + } + + void update_estimator(MT const& samples) + { + num_chains++; + + for (int i = 0; i < d; i++) + { + draws = samples.row(i).transpose(); + compute_autocovariance(draws, auto_cov); + + new_elem = draws.mean(); + delta = new_elem - cm_mean.coeff(i); + cm_mean(i) += delta / NT(num_chains); + cm_var(i) += delta * (new_elem - cm_mean(i)); + + new_elem = auto_cov.coeff(0) * NT(num_draws) / (NT(num_draws) - 1.0); + delta = new_elem - cv_mean.coeff(i); + cv_mean(i) += delta / NT(num_chains); + + new_elem = auto_cov.coeff(1); + delta = new_elem - acov_s_mean.coeff(0, i); + acov_s_mean(0, i) += delta / NT(num_chains); + jj = 1; + while (jj < num_draws-4) + { + new_elem = auto_cov.coeff(jj+1); + delta = new_elem - acov_s_mean.coeff(jj, i); + acov_s_mean(jj, i) += delta / NT(num_chains); + + new_elem = auto_cov.coeff(jj+2); + delta = new_elem - acov_s_mean.coeff(jj+1, i); + acov_s_mean(jj+1, i) += delta / NT(num_chains); + + jj += 2; + } + } + } + + + void estimate_effective_sample_size() + { + rho_hat_s.setZero(num_draws, d); + + var_plus = cv_mean * (NT(num_draws) - 1.0) / NT(num_draws); + if (num_chains > 1) + { + VT cm_var_temp = cm_var * (1.0 / (NT(num_chains)-1.0)); + var_plus += cm_var_temp; + } + + for (int i = 0; i < d; i++) + { + rho_hat_even = 1.0; + rho_hat_s(0, i) = rho_hat_even; + rho_hat_odd = 1 - (cv_mean.coeff(i) - acov_s_mean.coeff(0, i)) / var_plus.coeff(i); + rho_hat_s(1, i) = rho_hat_odd; + + s = 1; + while (s < (num_draws - 4) && (rho_hat_even + rho_hat_odd) > 0) + { + rho_hat_even = 1.0 - (cv_mean.coeff(i) - acov_s_mean.coeff(s, i)) / var_plus.coeff(i); + rho_hat_odd = 1.0 - (cv_mean.coeff(i) - acov_s_mean.coeff(s+1, i)) / var_plus.coeff(i); + if ((rho_hat_even + rho_hat_odd) >= 0) + { + rho_hat_s(s + 1, i) = rho_hat_even; + rho_hat_s(s + 2, i) = rho_hat_odd; + } + s += 2; + } + + max_s = s; + // this is used in the improved estimate + if (rho_hat_even > 0) + { + rho_hat_s(max_s + 1, i) = rho_hat_even; + } + + // Convert Geyer's positive sequence into a monotone sequence + for (jj = 1; jj <= max_s - 3; jj += 2) + { + if (rho_hat_s(jj + 1, i) + rho_hat_s.coeff(jj + 2, i) > rho_hat_s.coeff(jj - 1, i) + rho_hat_s.coeff(jj, i)) + { + rho_hat_s(jj + 1, i) = (rho_hat_s.coeff(jj - 1, i) + rho_hat_s.coeff(jj, i)) / 2.0; + rho_hat_s(jj + 2, i) = rho_hat_s.coeff(jj + 1, i); + } + } + NT num_total_draws = NT(num_chains) * NT(num_draws); + NT tau_hat = -1.0 + 2.0 * rho_hat_s.col(i).head(max_s).sum() + rho_hat_s.coeff(max_s + 1, i); + ess(i) = num_total_draws / tau_hat; + } + } + + + VT get_effective_sample_size() + { + return ess; + } + +}; + + +#endif + diff --git a/src/volesti/include/diagnostics/geweke.hpp b/src/volesti/include/diagnostics/geweke.hpp new file mode 100644 index 00000000..7bfd291a --- /dev/null +++ b/src/volesti/include/diagnostics/geweke.hpp @@ -0,0 +1,79 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +//Licensed under GNU LGPL.3, see LICENCE file + +/* + This function implements a multivariate version of the Geweke diagnostic. + It is reduced to Hotelling's Two Sample test, which is a multivariate + extension of the common two sample Student's t-test. The null hypothesis + is that there is no difference between sample means. + + It is based on "Evaluating the accuracy of sampling-based approaches + to the calculation of posterior moments, 1992" by J. Geweke + + Inputs: samples, a matrix that contains sample points column-wise + frac_first, the portion of the first in order points in matrix samples + frac_last, the portion of the last in order points in matrix samples + alpha, the confidence level for the statistical test + + Output: A boolean to denote the result of Geweke diagnostic: + (i) false if the null hypothesis is rejected + (ii) true if the null hypothesis is not rejected +*/ + + +#ifndef DIAGNOSTICS_GEWEKE_HPP +#define DIAGNOSTICS_GEWEKE_HPP + +#include + +template +bool perform_geweke(MT const& samples, + NT frac_first = 0.1, + NT frac_last = 0.5, + NT alpha = 0.05) +{ + unsigned int d = samples.rows(), N = samples.cols(); + unsigned int N1 = N * frac_first; + unsigned int N2 = N * frac_last; + + // Compute sample means and covariances + VT mean1 = samples.block(0, 0, d, N1).rowwise().mean(); + VT mean2 = samples.block(0, N - N2, d, N2).rowwise().mean(); + + MT norm_chain1 = samples.block(0, 0, d, N1).colwise() - mean1; + MT norm_chain2 = samples.block(0, N - N2, d, N2).colwise() - mean2; + + MT sigma1 = (norm_chain1 * norm_chain1.transpose()) / (NT(N1) - 1.0); + MT sigma2 = (norm_chain2 * norm_chain2.transpose()) / (NT(N2) - 1.0); + + // Compute the pooled covariance matrix + MT S_pl = ((NT(N1) - NT(1)) * sigma1 + (NT(N2) - 1.0) * sigma2) / (NT(N1) + NT(N2) - NT(2)); + + // T2 follows Hotelling's T-squared distribution under the assumption of + // equal covariances and when the null hypothesis is true + NT T2 = (mean1 - mean2).transpose() * S_pl.inverse() * (mean1 - mean2); + T2 = ((NT(N1) * NT(N2)) / (NT(N1) + NT(N2))) * T2; + + // U follows Fischer distribution + // We use this transformation to check the null hypothesis more easily + NT U = ((NT(N1) + NT(N2) - NT(d) - 1.0) / ((NT(N1) + NT(N2) - 2.0) * NT(d))) * T2; + + boost::math::fisher_f dist(d, int(N1) + int(N2) - d - 1); + + NT F1 = boost::math::quantile(dist, alpha / 2.0); + NT F2 = boost::math::quantile(boost::math::complement(dist, alpha / 2.0)); + + if (U <= F1 || U > F2) { // reject null hypothesis + return false; + } + return true; // do not reject null hypothesis + +} + +#endif diff --git a/src/volesti/include/diagnostics/interval_psrf.hpp b/src/volesti/include/diagnostics/interval_psrf.hpp new file mode 100644 index 00000000..cbcdfd10 --- /dev/null +++ b/src/volesti/include/diagnostics/interval_psrf.hpp @@ -0,0 +1,79 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +//Licensed under GNU LGPL.3, see LICENCE file + +/* + This function implements the interval diagnostic of Brooks & Gelman. + It is based on "General Methods for Monitoring Convergence of Iterative Simulations, 1998" by S. Brooks and A. Gelman + + For each coordinate the sample is splitted into two parts. + Then the psrf of S. Brooks and A. Gelman is computed for each coordinate. + + Inputs: samples, a matrix that contains sample points column-wise + + Output: The value of interval PSRF of S. Brooks and A. Gelman for each coordinate +*/ + +#ifndef DIAGNOSTICS_INTERVAL_PSRF_HPP +#define DIAGNOSTICS_INTERVAL_PSRF_HPP + +template +VT interval_psrf(MT const& samples, NT alpha = 0.05) +{ + MT runs = samples.transpose(); + unsigned int N = samples.cols(), d = samples.rows(); + unsigned int N1 = N / 2; + unsigned int N2 = N - N1; + VT sorted_samples(N), marginal_samples(N), sorted_subsamples1(N1), sorted_subsamples2(N2), results(d); + std::vector temp_col(N); + + for (int i = 0; i < d; i++) + { + sorted_samples = runs.col(i); + marginal_samples = runs.col(i); + + temp_col.resize(N); + temp_col = std::vector(&sorted_samples[0], sorted_samples.data() + sorted_samples.cols() * + sorted_samples.rows()); + std::sort(temp_col.begin(), temp_col.end()); + sorted_samples = Eigen::Map(&temp_col[0], temp_col.size()); + + int n1 = N * (alpha / NT(2)), n2 = N - N * (alpha / NT(2)); + + NT len_total_sequence_interval = sorted_samples(n2) - sorted_samples(n1); + + sorted_subsamples1 = marginal_samples.block(0,0,N1,1); + temp_col.resize(N1); + temp_col = std::vector(&sorted_subsamples1[0], sorted_subsamples1.data() + + sorted_subsamples1.cols() * sorted_subsamples1.rows()); + std::sort(temp_col.begin(), temp_col.end()); + sorted_subsamples1 = Eigen::Map(&temp_col[0], temp_col.size()); + + sorted_subsamples2 = marginal_samples.block(N1,0,N2,1); + temp_col.resize(N2); + temp_col = std::vector(&sorted_subsamples2[0], sorted_subsamples2.data() + + sorted_subsamples2.cols() * sorted_subsamples2.rows()); + std::sort(temp_col.begin(), temp_col.end()); + sorted_subsamples2 = Eigen::Map(&temp_col[0], temp_col.size()); + + n1 = N1 * (alpha / NT(2)), n2 = N1 - N1 * (alpha / NT(2)); + NT len_sequence_interval1 = sorted_subsamples1(n2) - sorted_subsamples1(n1); + + n1 = N2 * (alpha / NT(2)), n2 = N2 - N2 * (alpha / NT(2)); + NT len_sequence_interval2 = sorted_subsamples2(n2) - sorted_subsamples2(n1); + + NT R = (len_total_sequence_interval) / + ((len_sequence_interval1 + len_sequence_interval2) / NT(2)); + + results(i) = std::abs(1.0 - R) + NT(1); + } + return results; +} + + +#endif diff --git a/src/volesti/include/diagnostics/multivariate_psrf.hpp b/src/volesti/include/diagnostics/multivariate_psrf.hpp new file mode 100644 index 00000000..883b28d6 --- /dev/null +++ b/src/volesti/include/diagnostics/multivariate_psrf.hpp @@ -0,0 +1,55 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +//Licensed under GNU LGPL.3, see LICENCE file + +/* + This function implements a multivariate version of the Rubin & Gelman diagnostic. + It is based on "Inference from iterative simulation using multiple sequences, 1992" by D. B. Rubin and A. Gelman + and "General Methods for Monitoring Convergence of Iterative Simulations, 1998" by S. Brooks and A. Gelman + + The sample is splitted into two parts. Then a multivariate psrf is computed as proposed by S. Brooks and A. Gelman + + Inputs: samples, a matrix that contains sample points column-wise + + Output: The value of multivariate PSRF by S. Brooks and A. Gelman +*/ + +#ifndef DIAGNOSTICS_PSRF_HPP +#define DIAGNOSTICS_PSRF_HPP + +template +NT multivariate_psrf(MT const& samples) +{ + unsigned int N = samples.cols(), d = samples.rows(); + unsigned int N1 = N / 2; + unsigned int N2 = N - N1; + + VT mean1 = samples.block(0, 0, d, N1).rowwise().mean(); + VT mean2 = samples.block(0, N1, d, N - N1).rowwise().mean(); + + MT norm_chain1 = samples.block(0, 0, d, N1).colwise() - mean1; + MT norm_chain2 = samples.block(0, N1, d, N - N1).colwise() - mean2; + + MT W = ((norm_chain1 * norm_chain1.transpose()) / (NT(N1) - 1.0) + + (norm_chain2 * norm_chain2.transpose()) / (NT(N2) - 1.0)) / NT(2); + + VT mean00 = (mean1 + mean2) / 2.0; + + MT B = (mean1 - mean00) * (mean1 - mean00).transpose() + + (mean2 - mean00) * (mean2 - mean00).transpose(); + + MT WB = W.inverse() * B; + Eigen::SelfAdjointEigenSolver eigensolver(WB); + NT l_max = eigensolver.eigenvalues().maxCoeff(); + + NT R = (NT(N1) - NT(1))/NT(N1) + 1.5 * l_max; + return R; +} + + +#endif diff --git a/src/volesti/include/diagnostics/print_diagnostics.hpp b/src/volesti/include/diagnostics/print_diagnostics.hpp new file mode 100644 index 00000000..b009d6d0 --- /dev/null +++ b/src/volesti/include/diagnostics/print_diagnostics.hpp @@ -0,0 +1,56 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef DIAGNOSTICS_PRINT_DIAGNOSTICS_HPP +#define DIAGNOSTICS_PRINT_DIAGNOSTICS_HPP + +template +void print_diagnostics(MT const& samples, unsigned int &min_ess, StreamType &stream) { + + unsigned int d = samples.rows(); + unsigned int N = samples.cols(); + + VariadicTable vt( + {"Dimension", + "Average", + "Standard Deviation", + "Effective Sample Size", + "Interval PSRF (50%)" + }); + + VT ess = effective_sample_size(samples, min_ess); + VT intv_psrf = interval_psrf(samples); + + NT row_mean, row_std; + + vt.setColumnPrecision({1, 3, 3, 3, 3}); + + vt.setColumnFormat({VariadicTableColumnFormat::AUTO, + VariadicTableColumnFormat::SCIENTIFIC, + VariadicTableColumnFormat::SCIENTIFIC, + VariadicTableColumnFormat::SCIENTIFIC, + VariadicTableColumnFormat::SCIENTIFIC}); + + for (unsigned int i = 0; i < d; i++) { + row_mean = samples.row(i).mean(); + row_std = NT(0); + for (int j = 0; j < N; j++) { + row_std += std::pow(samples(i, j) - row_mean, 2); + } + row_std = sqrt(row_std / N); + vt.addRow(i + 1, row_mean, row_std, ess(i), intv_psrf(i)); + } + + vt.print(stream); + std::cout << "interval_psrf = " << intv_psrf.maxCoeff() << "us" << std::endl; +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery.hpp b/src/volesti/include/diagnostics/raftery.hpp new file mode 100644 index 00000000..1fa087a5 --- /dev/null +++ b/src/volesti/include/diagnostics/raftery.hpp @@ -0,0 +1,142 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +//Licensed under GNU LGPL.3, see LICENCE file + +/* + This function implements a multivariate version of the raftery & Lewis diagnostic. + It is based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + and "How many iterations in the Gibbs sampler?, 1992" by A. Raftery and S. Lewis + + Inputs: samples, a matrix that contains sample points column-wise + q, the quantile of the quantity of interest. The default value is 0.025. + r, the level of precision desired. The default value is 0.01. + s, the probability associated with r. The default value is 0.95. + + Outputs: (i) The number of draws required for burn-in + (ii) The skip parameter for 1st-order Markov chain + (iii) The skip parameter sufficient to get independence chain + (iv) The number of draws required to achieve r precision + (v) The number of draws if the chain is white noise + (vi) The I-statistic from Raftery and Lewis (1992) +*/ + +#ifndef DIAGNOSTICS_RAFTERY_HPP +#define DIAGNOSTICS_RAFTERY_HPP + +template +NT round_to_zero(NT x) +{ + return (x > 0.0) ? std::floor(x) : std::ceil(x); +} + +#include "raftery_subroutines/empquant.hpp" +#include "raftery_subroutines/indtest.hpp" +#include "raftery_subroutines/mctest.hpp" +#include "raftery_subroutines/mcest.hpp" +#include "raftery_subroutines/thin.hpp" +#include "raftery_subroutines/ppnd.hpp" + + +template +MT perform_raftery(MT const& samples, NT const& q, NT const& r, NT const& s) +{ + MT runs = samples.transpose(); + + typedef Eigen::Matrix MTint; + typedef Eigen::Matrix VTint; + + unsigned int n = runs.rows(), d = runs.cols(), kthin, kmind; + MT results(d, 6); + MTint work = MTint::Zero(n, d); + VTint tmp = VTint::Zero(n); + std::pair xy; + std::pair g2bic; + + NT cutpt, alpha, beta, g2, bic, epss; + int tcnt; + + MT sorted_samples(n, d); + VT a(n); + std::vector temp_col(n); + + for (int i = 0; i < d; i++) + { + a = runs.col(i); + temp_col = std::vector(&a[0], a.data() + a.cols() * a.rows()); + std::sort(temp_col.begin(), temp_col.end()); + sorted_samples.col(i) = Eigen::Map(&temp_col[0], temp_col.size()); + } + + for (int i = 0; i < d; i++) + { + cutpt = empquant(sorted_samples.col(i), q); + for (int j = 0; j < n; j++) + { + if (runs(j, i) <= cutpt) work(j, i) = 1; + } + kthin = 1; bic = 1.0; epss = 0.001; + + while(bic > 0.0) + { + xy = thin(work.col(i), n, kthin); + tcnt = xy.first; + tmp = xy.second; + g2bic = mctest(tmp, tcnt); + g2 = g2bic.first; + bic = g2bic.second; + kthin++; + if (kthin > n / 2) { + break; + } + } + + kthin--; + g2bic = mcest(tmp, tcnt); + alpha = g2bic.first; + beta = g2bic.second; + kmind = kthin; + g2bic = indtest(tmp, tcnt); + g2 = g2bic.first; + bic = g2bic.second; + + while (bic > 0.0) + { + xy = thin(work.col(i), n, kmind); + tcnt = xy.first; + tmp = xy.second; + g2bic = indtest(tmp, tcnt); + g2 = g2bic.first; + bic = g2bic.second; + kmind++; + if (kmind > n) { + break; + } + } + + NT psum = alpha + beta; + NT tmp1 = std::log(psum * epss / std::max(alpha, beta)) / std::log(std::abs(1.0 - psum)); + NT nburn = round_to_zero((tmp1 + 1.0) * NT(kthin)); + NT phi = ppnd((s + 1.0) / 2.0); + NT tmp2 = (2.0 - psum) * alpha * beta * (phi * phi) / (psum * psum * psum * (r * r)); + NT nprec = round_to_zero(tmp2 + 1.0) * kthin; + NT nmin = round_to_zero(((1.0 - q) * q * (phi * phi) / (r * r)) + 1.0); + NT irl = (nburn + nprec) / nmin; + NT kind = std::max(round_to_zero(irl + 1.0), NT(kmind)); + + results(i, 0) = NT(kthin); + results(i, 1) = NT(nburn); + results(i, 2) = kind; + results(i, 3) = NT(nburn) + nprec; + results(i, 4) = nmin; + results(i, 5) = irl; + } + return results; +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery_subroutines/empquant.hpp b/src/volesti/include/diagnostics/raftery_subroutines/empquant.hpp new file mode 100644 index 00000000..6604f0ad --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/empquant.hpp @@ -0,0 +1,32 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef EMPQUANT_HPP +#define EMPQUANT_HPP + + +template +NT empquant(VT const& sorted_samples, NT const& q) +{ + unsigned int n = sorted_samples.rows(); + + NT order = (n - 1) * q + 1.0; + NT fract = order - NT(int(order)); + int low = std::max(round_to_zero(order), 1.0); + int high = std::min(low + 1.0, NT(n)); + + NT y = (1.0 - fract) * sorted_samples(low - 1) + fract * sorted_samples(high - 1); + + return y; +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery_subroutines/indtest.hpp b/src/volesti/include/diagnostics/raftery_subroutines/indtest.hpp new file mode 100644 index 00000000..8af7a329 --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/indtest.hpp @@ -0,0 +1,43 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef INDTEST_HPP +#define INDTEST_HPP + + +template +std::pair indtest(VT const& d, int const& n) +{ + MT t = MT::Zero(2, 2); + int t1, t2; + NT fitted, focus; + for (int i1 = 1; i1 < n; i1++){ + t(d(i1 - 1), d(i1))=t(d(i1 - 1), d(i1)) + 1; + } + NT dcm1 = NT(n) - 1.0, g2 = 0.0; + for (int i1 = 0; i1 < 2; i1++){ + for (int i2 = 0; i2 < 2; i2++){ + if (t(i1, i2) != 0){ + t1 = t(i1, 0) + t(i1, 1); + t2 = t(0, i2) + t(1, i2); + fitted = (NT(t1) * NT(t2)) / dcm1; + focus = NT(t(i1, i2)); + g2 = g2 + std::log(focus / fitted) * focus; + } + } + } + g2 = g2 * 2.0; + NT bic = g2 - std::log(dcm1); + return std::pair(g2, bic); +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery_subroutines/mcest.hpp b/src/volesti/include/diagnostics/raftery_subroutines/mcest.hpp new file mode 100644 index 00000000..8f29021b --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/mcest.hpp @@ -0,0 +1,28 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef MCEST_HPP +#define MCEST_HPP + +template +std::pair mcest(VT const& d, int const& n) +{ + MT t = MT::Zero(2, 2); + for (int i1 = 1; i1 < n; i1++){ + t(d(i1 - 1), d(i1)) = t(d(i1 - 1), d(i1)) + 1; + } + NT alpha = NT(t(0, 1)) / NT((t(0, 0) + t(0, 1))), beta = NT(t(1, 0)) / NT((t(1, 0)+t(1, 1))); + + return std::pair(alpha, beta); +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery_subroutines/mctest.hpp b/src/volesti/include/diagnostics/raftery_subroutines/mctest.hpp new file mode 100644 index 00000000..f5108db8 --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/mctest.hpp @@ -0,0 +1,63 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef MCTEST_HPP +#define MCTEST_HPP + + +template +std::pair mctest(VT const& d, int const& n) +{ + MT m1 = MT::Zero(2,2), m2 = MT::Zero(2,2); + NT g2 = 0.0, bic = 0.0, fitted; + int i1, i2, i3, t1, t2, t3, t4, focus; + + for (int i=2; i(g2, bic); +} + + +#endif diff --git a/src/volesti/include/diagnostics/raftery_subroutines/ppnd.hpp b/src/volesti/include/diagnostics/raftery_subroutines/ppnd.hpp new file mode 100644 index 00000000..c2c8b630 --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/ppnd.hpp @@ -0,0 +1,56 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef PPND_HPP +#define PPND_HPP + +template +NT ppnd(NT const& p) +{ + NT split1 = 0.425, split2 = 5.0, const1 = 0.180625, const2 = 1.6, a0=3.3871327179e+00, + a1 = 5.0434271938e+01, a2 = 1.5929113202e+02, a3 = 5.9109374720e+01, b1 = 1.7895169469e+01, + b2 = 7.8757757664e+01, b3 = 6.7187563600e+01, c0 = 1.4234372777e+00, c1 = 2.7568153900e+00, + c2 = 1.3067284816e+00, c3 = 1.7023821103e-01, d1 = 7.3700164250e-01, d2 = 1.2021132975e-01, + e0 = 6.6579051150e+00, e1 = 3.0812263860e+00, e2 = 4.2868294337e-01, e3 = 1.7337203997e-02, + f1 = 2.4197894225e-01, f2 = 1.2258202635e-02; + + NT q = p - 0.5, r, y; + + if (std::abs(q) <= split1){ + r = const1 - q * q; + y = q * (((a3 * r + a2) * r + a1) * r + a0) / (((b3 * r + b2) * r + b1) * r + 1.0); + return y; + } else if (q < 0.0){ + r = p; + } else{ + r = 1 - p; + } + if (r <= 0.0){ + return 0.0; + } + + r = std::sqrt(-1.0 * std::log(r)); + + if (r <= split2){ + r = r - const2; + y = (((c3 * r + c2) * r + c1) * r + c0) / ((d2 * r + d1) * r + 1.0); + } else{ + r = r - split2; + y = (((e3 * r + e2) * r + e1) * r + e0)/((f2 * r + f1) * r + 1.0); + } + + if (q < 0.0) return -y; + + return y; +} + +#endif + diff --git a/src/volesti/include/diagnostics/raftery_subroutines/thin.hpp b/src/volesti/include/diagnostics/raftery_subroutines/thin.hpp new file mode 100644 index 00000000..c9b5ec10 --- /dev/null +++ b/src/volesti/include/diagnostics/raftery_subroutines/thin.hpp @@ -0,0 +1,32 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +//Based on Matlab version of coda package in http://www.spatial-econometrics.com/gibbs/ + +#ifndef THIN_HPP +#define THIN_HPP + +template +std::pair thin(VT const& work, unsigned int const& n, unsigned int const& kthin) +{ + VT y((n-1) / kthin + 1); + + int i = 0, j = 0; + while (i < n) + { + y(j) = work(i); + j++; + i += kthin; + } + + return std::pair((n-1) / kthin + 1, y); +} + + +#endif diff --git a/src/volesti/include/diagnostics/thin_samples.hpp b/src/volesti/include/diagnostics/thin_samples.hpp new file mode 100644 index 00000000..8e41fe2f --- /dev/null +++ b/src/volesti/include/diagnostics/thin_samples.hpp @@ -0,0 +1,37 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef DIAGNOSTICS_THIN_SAMPLES_HPP +#define DIAGNOSTICS_THIN_SAMPLES_HPP + +template +MT thin_samples(MT const& samples, NT const& min_ess) { + + // Sample matrix is provided as d x n_samples + unsigned int d = samples.rows(); + unsigned int N = samples.cols(); + unsigned int gap; + unsigned int N_gap; + + // Thin samples are the initial samples which are N / min_ess apart + gap = N / min_ess; + N_gap = N - N % gap; + + MT thin_samples; + thin_samples.resize(d, N_gap / gap); + + for (int i = 0; i < N_gap; i += gap) { + thin_samples.col(i / gap) = samples.col(i); + } + + return thin_samples; +} + +#endif diff --git a/src/volesti/include/diagnostics/univariate_psrf.hpp b/src/volesti/include/diagnostics/univariate_psrf.hpp new file mode 100644 index 00000000..3a863908 --- /dev/null +++ b/src/volesti/include/diagnostics/univariate_psrf.hpp @@ -0,0 +1,67 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +//Licensed under GNU LGPL.3, see LICENCE file + +/* + This function implements the Rubin & Gelman diagnostic. + It is based on "Inference from iterative simulation using multiple sequences, 1992" by D. B. Rubin and A. Gelman + + For each coordinate the sample is splitted into two parts. + Then the psrf of D.B. Rubin and A. Gelman is computed for each coordinate + + Inputs: samples, a matrix that contains sample points column-wise + + Output: The value of PSRF of D.B. Rubin and A. Gelman for each coordinate +*/ + +#ifndef DIAGNOSTICS_MARGINAL_PSRF_HPP +#define DIAGNOSTICS_MARGINAL_PSRF_HPP + +template +VT univariate_psrf(MT const& samples) +{ + MT runs = samples.transpose(); + unsigned int N = samples.cols(), d = samples.rows(); + unsigned int N1 = N / 2; + unsigned int N2 = N - N1; + VT coord_samples(N), results(d); + NT mean1, mean2, mean00, sum, R, W, B, sigma; + + for (int i = 0; i < d; i++) + { + coord_samples = runs.col(i); + mean1 = coord_samples.block(0,0,N1,1).mean(); + mean2 = coord_samples.block(N1,0,N2,1).mean(); + + sum = NT(0); + for (int j = 0; j < N1; j++) + { + sum += (coord_samples(j) - mean1) * (coord_samples(j) - mean1); + } + W = sum / (NT(N1) - NT(1)); + + sum = NT(0); + for (int j = N1; j < N; j++) + { + sum += (coord_samples(j) - mean2) * (coord_samples(j) - mean2); + } + W += (sum / (NT(N2) - NT(1))); + W = W / NT(2); + + mean00 = coord_samples.mean(); + + B = (mean1 - mean00) * (mean1 - mean00) + (mean2 - mean00) * (mean2 - mean00); + sigma = ((NT(N1) - NT(1)) / NT(N1)) * W + B; + R = std::sqrt(sigma / W); + + results(i) = R; + } + return results; +} + +#endif diff --git a/src/volesti/include/generators/boost_random_number_generator.hpp b/src/volesti/include/generators/boost_random_number_generator.hpp new file mode 100644 index 00000000..d15fee15 --- /dev/null +++ b/src/volesti/include/generators/boost_random_number_generator.hpp @@ -0,0 +1,95 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020 Vissarion Fisikopoulos + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef GENERATORS_BOOST_RANDOM_NUMBER_GENERATOR_HPP +#define GENERATORS_BOOST_RANDOM_NUMBER_GENERATOR_HPP + +#include +#include + +/////////////////// Random numbers generator +/// +/// \tparam RNGType +/// \tparam NT +/// \tparam Ts + +template +struct BoostRandomNumberGenerator; + +template +struct BoostRandomNumberGenerator +{ + BoostRandomNumberGenerator(int d) + : _rng(std::chrono::system_clock::now().time_since_epoch().count()) + , _urdist(0, 1) + , _uidist(0, d-1) + , _ndist(0, 1) + {} + + NT sample_urdist() + { + return _urdist(_rng); + } + + NT sample_uidist() + { + return _uidist(_rng); + } + + NT sample_ndist() + { + return _ndist(_rng); + } + + void set_seed(unsigned rng_seed){ + _rng.seed(rng_seed); + } + +private : + RNGType _rng; + boost::random::uniform_real_distribution _urdist; + boost::random::uniform_int_distribution<> _uidist; + boost::random::normal_distribution _ndist; +}; + + +template +struct BoostRandomNumberGenerator +{ + BoostRandomNumberGenerator(int d=1) + : _rng(Seed) + , _urdist(0, 1) + , _uidist(0, d-1) + , _ndist(0, 1) + {} + + NT sample_urdist() + { + return _urdist(_rng); + } + + NT sample_uidist() + { + return _uidist(_rng); + } + + NT sample_ndist() + { + return _ndist(_rng); + } + + void set_seed(unsigned rng_seed){ + _rng.seed(rng_seed); + } + +private : + RNGType _rng; + boost::random::uniform_real_distribution _urdist; + boost::random::uniform_int_distribution<> _uidist; + boost::random::normal_distribution _ndist; +}; + +#endif // GENERATORS_BOOST_RANDOM_NUMBER_GENERATOR_HPP diff --git a/src/volesti/include/generators/convex_bodies_generator.h b/src/volesti/include/generators/convex_bodies_generator.h new file mode 100644 index 00000000..f80b98d1 --- /dev/null +++ b/src/volesti/include/generators/convex_bodies_generator.h @@ -0,0 +1,124 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef CONVEX_BODIES_GEN_H +#define CONVEX_BODIES_GEN_H + +#include + +#include "convex_bodies/convex_body.h" + +#ifndef isnan + using std::isnan; +#endif + +/// This function generates a unit ball of given dimension +/// @tparam ConvexBody Type of returned Convex Body +template +ConvexBody generate_unit_ball(unsigned int dim) { + + typedef typename ConvexBody::MT MT; + typedef typename ConvexBody::VT VT; + typedef typename ConvexBody::NT NT; + typedef typename ConvexBody::PointType Point; + typedef std::function func; + typedef std::function grad; + + func unit_ball_func = [](const Point &x) { + return x.dot(x) - 1; + }; + + grad unit_ball_grad = [](const Point &x) { + return 2 * x; + }; + + std::vector unit_ball_funcs{unit_ball_func}; + std::vector unit_ball_grads{unit_ball_grad}; + + return ConvexBody(unit_ball_funcs, unit_ball_grads, dim); +} + +/// This function generates a unit ball of given dimension intersected by a hyperplane +/// @tparam ConvexBody Type of returned Convex Body +template +ConvexBody generate_unit_ball_intersect_hyperplane(unsigned int dim) { + + typedef typename ConvexBody::MT MT; + typedef typename ConvexBody::VT VT; + typedef typename ConvexBody::NT NT; + typedef typename ConvexBody::PointType Point; + typedef std::function func; + typedef std::function grad; + + func unit_ball_func = [](const Point &x) { + return x.dot(x) - 1; + }; + + func hyperplane_func = [](const Point &x) { + return x[0] - 0.5; + }; + + grad unit_ball_grad = [](const Point &x) { + return 2 * x; + }; + + grad hyperplane_grad = [](const Point &x) { + Point v = Point(x.dimension()); + v.set_coord(0, 1); + return v; + }; + + std::vector unit_ball_funcs{unit_ball_func, hyperplane_func}; + std::vector unit_ball_grads{unit_ball_grad, hyperplane_grad}; + + return ConvexBody(unit_ball_funcs, unit_ball_grads, dim); +} + +/// This function generates a unit ball of given dimension intersected by a logsum exponential function +/// @tparam ConvexBody Type of returned Convex Body +template +ConvexBody generate_unit_ball_intersect_logsumexp(unsigned int dim) { + + typedef typename ConvexBody::MT MT; + typedef typename ConvexBody::VT VT; + typedef typename ConvexBody::NT NT; + typedef typename ConvexBody::PointType Point; + typedef std::function func; + typedef std::function grad; + + func unit_ball_func = [](const Point &x) { + return x.dot(x) - pow(x.dimension(), 2); + }; + + func logsumexp_func = [](const Point &x) { + typedef typename Point::FT NT; + NT s = 0; + for (unsigned int i = 0; i < x.dimension(); i++) { + s += exp(x[i]); + } + return log(s); + }; + + grad unit_ball_grad = [](const Point &x) { + return 2 * x; + }; + + grad logsumexp_grad = [](const Point &x) { + Point z(x.dimension()); + for (unsigned int i = 0; i < x.dimension(); i++) { + z.set_coord(i, exp(x[i])); + } + return (1 / z.sum()) * z; + }; + + std::vector unit_ball_funcs{unit_ball_func, logsumexp_func}; + std::vector unit_ball_grads{unit_ball_grad, logsumexp_grad}; + + return ConvexBody(unit_ball_funcs, unit_ball_grads, dim); +} + +#endif diff --git a/src/volesti/include/generators/h_polytopes_generator.h b/src/volesti/include/generators/h_polytopes_generator.h new file mode 100644 index 00000000..dae44bdc --- /dev/null +++ b/src/volesti/include/generators/h_polytopes_generator.h @@ -0,0 +1,61 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef H_POLYTOPES_GEN_H +#define H_POLYTOPES_GEN_H + +#include + + +#ifndef isnan + using std::isnan; +#endif + +/// This function generates a random H-polytope of given dimension and number of hyperplanes $m$ +/// @tparam Polytope Type of returned polytope +/// @tparam RNGType RNGType Type +template +Polytope random_hpoly(unsigned int dim, unsigned int m, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType Point; + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + + MT A(m, dim); + VT b(m); + Point p(dim); + + for (int i = 0; i < m; ++i) { + boost::normal_distribution<> rdist(0, 1); + NT normal = NT(0); + NT *data = p.pointerToData(); + + //RNGType rng2 = var.rng; + for (unsigned int i = 0; i < dim; ++i) { + *data = rdist(rng); + normal += *data * *data; + data++; + } + + normal = 1.0 / std::sqrt(normal); + p *= normal; + A.row(i) = p.getCoefficients(); + b(i) = 10.0; + } + + return Polytope(dim, A, b); +} + +#endif diff --git a/src/volesti/include/generators/known_polytope_generators.h b/src/volesti/include/generators/known_polytope_generators.h new file mode 100644 index 00000000..050918ad --- /dev/null +++ b/src/volesti/include/generators/known_polytope_generators.h @@ -0,0 +1,361 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef KNOWN_POLYTOPE_GENERATORS_H +#define KNOWN_POLYTOPE_GENERATORS_H + +#include + +#include "convex_bodies/hpolytope.h" +#include "convex_bodies/vpolytope.h" + +/// This function generates a hypercube of given dimension +/// The result can be either in V-representation (Vpoly=true) or in H-representation (V-poly-false) +/// @tparam Polytope Type of returned polytope +template +Polytope generate_cube(const unsigned int& dim, const bool& Vpoly) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + MT A; + VT b; + unsigned int m; + + if (!Vpoly) { + + A.resize(2 * dim, dim); + b.resize(2 * dim); + for (unsigned int i = 0; i < dim; ++i) { + b(i) = 1.0; + for (unsigned int j = 0; j < dim; ++j) { + if (i == j) { + A(i, j) = 1.0; + } else { + A(i, j) = 0.0; + } + } + } + for (unsigned int i = 0; i < dim; ++i) { + b(i + dim) = 1.0; + for (unsigned int j = 0; j < dim; ++j) { + if (i == j) { + A(i + dim, j) = -1.0; + } else { + A(i + dim, j) = 0.0; + } + } + } + } else { + + m = 2 << (dim - 1); + A.resize(m, dim); + b.resize(m); + for(unsigned int i=0; i> 1; + ++j; + } + for(; j +Polytope generate_cross(const unsigned int &dim, const bool &Vpoly) { + + unsigned int m; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + + MT A; + VT b; + if (!Vpoly) { + + m = 2 << (dim - 1); + A.resize(m, dim); + b.resize(m); + for(unsigned int i=0; i> 1; + ++j; + } + for(; j +Polytope generate_simplex(const unsigned int &dim, const bool &Vpoly){ + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + + MT A; + A.resize(dim+1, dim); + VT b; + b.resize(dim+1); + + for(unsigned int i=0; i +Polytope generate_prod_simplex(const unsigned int &dim, bool Vpoly = false){ + + Polytope Perr; + try + { + if(Vpoly) throw false; + } + catch (bool e) { + #ifdef VOLESTI_DEBUG + std::cout<<"Only prod simplices in H-representation can be generated.."< +Polytope generate_skinny_cube(const unsigned int &dim, bool Vpoly = false) { + + Polytope Perr; + try + { + if(Vpoly) throw false; + } + catch (bool e) { + #ifdef VOLESTI_DEBUG + std::cout<<"Only prod simplices in H-representation can be generated.."< +Polytope generate_birkhoff(unsigned int const& n) { + + unsigned int m = n * n; + unsigned int d = n * n - 2 * n + 1; + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + + MT A = MT::Zero(m, d); + VT b(m); + + b(d) = -1.0 * int(n - 2); + + for (int i = 0; i < d; ++i) { + A(d, i) = -1; + } + + for (int i = 0; i < d; ++i) { + b(i) = 0; + A(i, i) = -1; + } + + for (int i = d+1; i < d+1+n-1; ++i) { + b(i) = 1; + for (int counter = 0; counter < n-1; ++counter) { + A(i, counter * (n-1) + (i-d-1)) = 1; + } + } + + for (int i = d+n; i < m; ++i) { + b(i) = 1; + for (int counter = 0; counter < n-1; ++counter) { + A(i, counter + (i-d-n) * (n-1)) = 1; + } + } + + Polytope P(d, A, b); + + return P; +} + +#endif diff --git a/src/volesti/include/generators/order_polytope_generator.h b/src/volesti/include/generators/order_polytope_generator.h new file mode 100644 index 00000000..bfece556 --- /dev/null +++ b/src/volesti/include/generators/order_polytope_generator.h @@ -0,0 +1,63 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2021 Vissarion Fisikopoulos +// Copyright (c) 2018-2021 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ORDER_POLYTOPES_GEN_H +#define ORDER_POLYTOPES_GEN_H + +#include +#include +#include "misc.h" +#include "misc/poset.h" + + +// Instances taken from: https://github.com/ttalvitie/le-counting-practice +static const std::unordered_map instances = +{ + {"bipartite_0.5_008_0", R"(0 0 0 0 1 0 1 0 + 0 0 0 0 1 0 0 0 + 0 0 0 0 1 1 0 1 + 0 0 0 0 1 0 1 0 + 0 0 0 0 0 0 0 0 + 0 0 0 0 0 0 0 0 + 0 0 0 0 0 0 0 0 + 0 0 0 0 0 0 0 0)"}, + + {"bayesiannetwork_andes_008_0", R"(0 0 0 0 0 0 0 0 + 1 0 0 0 0 0 0 0 + 0 1 0 0 0 0 0 0 + 0 0 0 0 0 0 0 0 + 0 0 1 1 0 0 0 0 + 0 0 0 0 1 0 0 0 + 0 0 0 0 0 0 0 0 + 0 0 0 0 1 1 1 0)"}, + +}; + +// generates an Order Polytope from an instance name +// Instances taken from: https://github.com/ttalvitie/le-counting-practice +/// @tparam Polytope Type of returned polytope +template +Polytope generate_orderpoly(std::string& instance_name) { + std::stringstream in_ss(instances.at(instance_name)); + Poset poset = read_poset_from_file_adj_matrix(in_ss).second; + return Polytope(poset); +} + +// Generates a cube as an Order Polytope +/// @tparam Polytope Type of returned polytope +template +Polytope generate_cube_orderpoly(unsigned int dim) { + typedef typename Poset::RV RV; + + RV order_relations; + Poset poset(dim, order_relations); + Polytope OP(poset); + return OP; +} + +#endif diff --git a/src/volesti/include/generators/sdp_generator.h b/src/volesti/include/generators/sdp_generator.h new file mode 100644 index 00000000..c2b3a154 --- /dev/null +++ b/src/volesti/include/generators/sdp_generator.h @@ -0,0 +1,151 @@ +// +// Created by panagiotis on 9/7/2019. +// + +#ifndef VOLESTI_SDP_GENERATOR_H +#define VOLESTI_SDP_GENERATOR_H + +#include /* srand, rand */ +#include /* time */ + +typedef boost::mt19937 RNGType; + +/// Generates a random matrix +/// @tparam NT Number type +template +void randomMatrixGOE(Eigen::Matrix& M) { + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + unsigned m = M.rows(); + boost::normal_distribution<> rdist(0,1); + unsigned seed = std::chrono::system_clock::now().time_since_epoch().count();//4 if fixed for debug + RNGType rng(seed); + + for (unsigned int i=0; i +Spectrahedron, Eigen::Matrix > generateSDP(int n, int m) { + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + MT ones = MT::Ones(m, m); + MT M = 2* Eigen::MatrixXd::Random(m,m) - ones; + + MT I = Eigen::MatrixXd::Identity(m, m); + std::vector matrices(n + 1); + matrices[0] = -(M * M.transpose()) - I; + + std::cout<<"A0 = "< lmi(matrices); + Spectrahedron spectrahedron(lmi); + return spectrahedron; + + //return optimization::sdp_problem(spectrahedron, obj); +} + +/// Generates a random spectahedron S(n, m) +/// @tparam NT Number type +template +Spectrahedron, Eigen::Matrix > generateSDP2(int n, int m) { + + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + MT ones = MT::Ones(m, m); + MT M = 2* Eigen::MatrixXd::Random(m,m) - ones; + + MT I = Eigen::MatrixXd::Identity(m, m); + std::vector matrices(n + 1); + matrices[0] = -(M * M.transpose()) - I; + + //std::cout<<"A0 = "< lmi(matrices); + Spectrahedron spectrahedron(lmi); + return spectrahedron; + + //return optimization::sdp_problem(spectrahedron, obj); +} + + +#endif //VOLESTI_SDP_GENERATOR_H diff --git a/src/volesti/include/generators/v_polytopes_generators.h b/src/volesti/include/generators/v_polytopes_generators.h new file mode 100644 index 00000000..3977e948 --- /dev/null +++ b/src/volesti/include/generators/v_polytopes_generators.h @@ -0,0 +1,175 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef V_POLYTOPES_GEN_H +#define V_POLYTOPES_GEN_H + +#include + +#ifndef isnan + using std::isnan; +#endif + +template +void removeRow(MT &matrix, unsigned int rowToRemove) +{ + unsigned int numRows = matrix.rows()-1; + unsigned int numCols = matrix.cols(); + + if( rowToRemove < numRows ) + matrix.block(rowToRemove,0,numRows-rowToRemove,numCols) = matrix.bottomRows(numRows-rowToRemove); + + matrix.conservativeResize(numRows,numCols); +} + +/// Generates a random V-polytope +/// @tparam Polytope polytope type +/// @tparam RNGType RNGType type +template +Polytope random_vpoly(unsigned int dim, unsigned int k, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType PointType; + typedef PointType Point; + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + boost::normal_distribution<> rdist(0,1); + + typename std::vector::iterator pit; + MT V(k, dim); + unsigned int j; + + + std::vector Xs(dim,0); + NT normal = NT(0); + + for (unsigned int i = 0; i < k; ++i) { + + normal = NT(0); + for (unsigned int i=0; i +Polytope random_vpoly_incube(unsigned int d, unsigned int k, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType PointType; + typedef PointType Point; + + REAL *conv_mem; + int *colno_mem; + + conv_mem = (REAL *) malloc(k * sizeof(*conv_mem)); + colno_mem = (int *) malloc(k * sizeof(*colno_mem)); + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + boost::random::uniform_real_distribution<> urdist1(-1, 1); + + Point p(d); + typename std::vector::iterator pit; + MT V(k, d); + unsigned int j, count_row,it=0; + std::vector indices; + + VT b = VT::Ones(k); + + for (unsigned int i = 0; i < k; ++i) { + for (int j = 0; j < d; ++j) { + V(i, j) = urdist1(rng); + } + } + if(k==d+1){ + return Polytope(d, V, b); + } + + MT V2(k,d); + V2 = V; + indices.clear(); + while(it<20) { + V.resize(V2.rows(), d); + V = V2; + for (int i = 0; i < indices.size(); ++i) { + V.conservativeResize(V.rows()+1, d); + for (int j = 0; j < d; ++j) { + V(V.rows()-1, j) = urdist1(rng); + } + } + indices.clear(); + V2.resize(k, d); + V2 = V; + + for (int i = 0; i < k; ++i) { + for (int j = 0; j < d; ++j) { + p.set_coord(j, V(i, j)); + } + removeRow(V2, i); + if (memLP_Vpoly(V2, p, conv_mem, colno_mem)){ + indices.push_back(i); + } + V2.resize(k, d); + V2 = V; + } + if (indices.size()==0) { + return Polytope(d, V, b); + } + V2.resize(k - indices.size(), d); + count_row =0; + for (int i = 0; i < k; ++i) { + if(std::find(indices.begin(), indices.end(), i) != indices.end()) { + continue; + } else { + for (int j = 0; j < d; ++j) V2(count_row, j) = V(i,j); + count_row++; + } + } + it++; + } + + + free(colno_mem); + free(conv_mem); + + return Polytope(d, V2, VT::Ones(V2.rows())); +// return VP; + +} + +#endif diff --git a/src/volesti/include/generators/z_polytopes_generators.h b/src/volesti/include/generators/z_polytopes_generators.h new file mode 100644 index 00000000..8afbc6d1 --- /dev/null +++ b/src/volesti/include/generators/z_polytopes_generators.h @@ -0,0 +1,145 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos +// Copyright (c) 2018 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef Z_POLYTOPES_GEN_H +#define Z_POLYTOPES_GEN_H + +#include + +#ifndef isnan + using std::isnan; +#endif + +/// Generates a random Zonotope with generators draw from Gaussian distribution +/// @tparam Polytope polytope type +/// @tparam RNGType RNGType type +template +Polytope gen_zonotope_gaussian(int dim, int m, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + boost::normal_distribution<> rdist(0, 1); + boost::normal_distribution<> rdist2(50, 33.3); + + MT A; + VT b; + A.resize(m, dim); + b.resize(m); + NT rand_gaus; + + for (unsigned int i = 0; i < m; ++i) { + b(i) = 1.0; + for (unsigned int j = 0; j < dim; ++j) { + A(i,j) = rdist(rng); + } + A.row(i)=A.row(i)/A.row(i).norm(); + while(true){ + rand_gaus = rdist2(rng); + if (rand_gaus > 0.0 && rand_gaus<100.0){ + A.row(i) = A.row(i) * rand_gaus; + break; + } + } + } + + Polytope P(dim, A, b); + return P; +} + + +/// Generates a random Zonotope with generators draw from uniform distribution +/// @tparam Polytope polytope type +/// @tparam RNGType RNGType type +template +Polytope gen_zonotope_uniform(int dim, int m, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + boost::normal_distribution<> rdist(0, 1); + boost::random::uniform_real_distribution<> urdist1(0, 100); + + MT A; + VT b; + A.resize(m, dim); + b.resize(m); + + for (unsigned int i = 0; i < m; ++i) { + b(i) = 1.0; + for (unsigned int j = 0; j < dim; ++j) { + A(i,j) = rdist(rng); + } + A.row(i)=A.row(i)/A.row(i).norm(); + A.row(i) = A.row(i) * urdist1(rng); + } + + Polytope P(dim, A, b); + return P; + +} + + +/// Generates a random Zonotope with generators draw from exponential distribution +/// @tparam Polytope polytope type +/// @tparam RNGType RNGType type +template +Polytope gen_zonotope_exponential(int dim, int m, double seed = std::numeric_limits::signaling_NaN()) { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed = seed; + rng.seed(rng_seed); + } + boost::normal_distribution<> rdist(0, 1); + boost::normal_distribution<> expdist(1.0/30.0); + + MT A; + VT b; + A.resize(m, dim); + b.resize(m); + NT rand_exp; + + for (unsigned int i = 0; i < m; ++i) { + b(i) = 1.0; + for (unsigned int j = 0; j < dim; ++j) { + A(i,j) = rdist(rng); + } + A.row(i)=A.row(i)/A.row(i).norm(); + while(true){ + rand_exp = expdist(rng); + if (rand_exp > 0.0 && rand_exp<100.0){ + A.row(i) = A.row(i) * rand_exp; + break; + } + } + } + + Polytope P(dim, A, b); + return P; +} + +#endif diff --git a/src/volesti/include/integration/simple_MC_integration.hpp b/src/volesti/include/integration/simple_MC_integration.hpp new file mode 100644 index 00000000..58739650 --- /dev/null +++ b/src/volesti/include/integration/simple_MC_integration.hpp @@ -0,0 +1,276 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2021 Vissarion Fisikopoulos +// Copyright (c) 2018-2021 Apostolos Chalkis + +// Contributed and/or modified by Suraj Choubey, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// Monte Carlo Integration algorithm used here : https://en.wikipedia.org/wiki/Monte_Carlo_integration#Overview + +#ifndef SIMPLE_MC_INTEGRATION_HPP +#define SIMPLE_MC_INTEGRATION_HPP + +#include +#include +#include +#include +#include "convex_bodies/hpolytope.h" +#include "Eigen/Eigen" +#include "generators/known_polytope_generators.h" +#include "boost_random_number_generator.hpp" +#include "cartesian_geom/cartesian_kernel.h" +#include "random_walks/random_walks.hpp" +#include "volume/volume_sequence_of_balls.hpp" +#include "volume/volume_cooling_gaussians.hpp" +#include "volume/volume_cooling_balls.hpp" +#include "misc.h" + +typedef double NT; +typedef Cartesian Kernel; +typedef typename Kernel::Point Point; +typedef HPolytope Hpolytope; +typedef boost::mt19937 RNGType; +typedef BoostRandomNumberGenerator RandomNumberGenerator; +typedef typename HPolytope::MT MT; +typedef typename HPolytope::VT VT; + +enum volumetype {CB ,CG ,SOB}; // Volume type for polytope +typedef typename std::vector Limit; // Standard way for user to use limits +// E.g. Limits LL{0.5, 1.5, 2.5} ; Limits UL{1.2, 1.8 , 2.8 } ; + +const Limit lt{0}; // To initialize non-initialized limits +const Point pt{0}; // To initialize non-initialized points + +// To check if two n-dimensional points ensure valid limits in integration +template +< + typename Point = Point, + typename NT = NT +> +bool valid_limits(Point LL, Point UL) { + if (UL.dimension() == LL.dimension()) { + for (int i = 0; i < LL.dimension(); i++) { + if (LL[i] > UL[i]) { + std::cerr << "Invalid integration limits\n"; + return false; + } + } + return true; + } else { + std::cerr << "Invalid integration limits\n"; + return false; + } +} + +// Initialize Limit Point +template +< + typename Point = Point, + typename NT = NT +> +Point init_limit(Limit L, int dim) { + Point pt(dim); + for (int i = 0; i < dim; i++) { + pt.set_coord(i, L[i]); + } + return pt; +} + +//Initialize to [-1,1]^n +template +< + typename Point = Point, + typename NT = NT +> +void initiate_unit_limits(Point& LL, Point& UL, int dim) { + LL.set_dimension(dim); + UL.set_dimension(dim); + LL.set_to_origin(); + UL.set_to_origin(); + + for (int i = 0 ; i < dim ; i++) { + LL.set_coord(i, -1); + UL.set_coord(i, 1); + } +} + +// Simple MC Integration Over Polytopes +template +< + typename WalkType = BallWalk, + typename Polytope = Hpolytope, + typename VolumeWalkType = BallWalk, + typename RNG = RandomNumberGenerator, + typename NT = NT, + typename Functor +> +NT simple_mc_polytope_integrate(Functor Fx, + Polytope &P, + RNG &rng, + int N = 10000, + volumetype voltype = SOB, + int walk_length = 1, + NT e = 0.1, + Point Origin = pt) +{ + + int dim = P.dimension(); + // P.print(); + + // Check if ShiftPoint is shifted with accurate dimensions + if (Origin.dimension() == 0 && dim > 0) { + Origin.set_dimension(dim); + Origin.set_to_origin(); + } else if (Origin.dimension() != dim && dim > 0) { + std::cerr << "Polytope Dimension != Shiftpoint Dimension" << std::endl; + return -1; + } + + // std::cout << "Origin.dimension() = " << Origin.dimension() << std::endl; + // std::cout << "P.dimension() = " << P.dimension() << std::endl; + + // Volume calculation for HPolytope + NT volume; + + switch (voltype) { + case CB: + volume = volume_cooling_balls (P, rng, e, walk_length).second; + break; + case CG: + volume = volume_cooling_gaussians (P, rng, e, walk_length); + break; + case SOB: + volume = volume_sequence_of_balls (P, rng, e, walk_length); + break; + default: + std::cerr << "Error in volume type: CB / SOB / CG" << std::endl; + return -1; + } + + // std::cout << "Volume of the convex body = " << volume << std::endl; + + // For implementing Uniform Walks + std::pair inner_ball = P.ComputeInnerBall(); + Point x0 = inner_ball.first; + typename WalkType::template Walk walk(P, x0, rng); + + NT sum = 0; + + // Applying and walking through Uniform Walks + Storing Points in Vector + for (int i = 0; i < N; i++) { + walk.apply(P, x0, walk_length, rng); + sum += Fx(x0 + Origin); + + // (x0 + Origin).print(); + } + + // Integration Value + NT integration_value = volume * sum / N ; + return integration_value; +} + +template +< + typename WalkType = BallWalk, + typename Polytope = Hpolytope, + typename VolumeWalkType = BallWalk, + typename RNG = RandomNumberGenerator, + typename NT = NT, + typename Functor +> +NT simple_mc_polytope_integrate(Functor Fx, + Polytope &P, + int N = 10000, + volumetype voltype = SOB, + int walk_length = 1, + NT e = 0.1, + Point Origin = pt) +{ + RNG rng(P.dimension()); + return simple_mc_polytope_integrate(Fx, P, rng, N, voltype, + walk_length, e, Origin); +} + +// Simple MC Integration over Hyper-Rectangles +template +< + typename WalkType = BallWalk, + typename RNG = RandomNumberGenerator, + typename NT = NT, + typename Functor +> +NT simple_mc_integrate(Functor Fx, + int dim, + RNG &rng, + int N = 10000, + volumetype voltype = SOB, + Limit LowLimit = lt, + Limit UpLimit = lt, + int walk_length = 10, + NT e = 0.1) +{ + + // Setting up integration limits + Point LL, UL; + if (LowLimit.size() == 1 && UpLimit.size() == 1 && LowLimit[0] == 0 && UpLimit[0] == 0) { + initiate_unit_limits(LL, UL, dim); + } else if (LowLimit.size() == UpLimit.size() && LowLimit.size() == dim) { + LL = init_limit (LowLimit, dim); + UL = init_limit (UpLimit, dim); + } else { + std::cerr << "Invalid limits entered"; + return -1; + } + + NT sum = 0; + + if (valid_limits(LL, UL)) { + + // Creating an MT & VT for HPolytope(Hyper-Rectangle) for integration limits using LL & UL + MT mt(dim*2, dim); + mt.setZero(); + VT vt(dim*2); + vt.setZero(); + + for (int i=0; i (Fx, P, rng, N, voltype, walk_length, e); + return integration_value; + + } else { + std::cerr << "Invalid integration limits" << std::endl; + return -1; + } +} + +template +< + typename WalkType = BallWalk, + typename RNG = RandomNumberGenerator, + typename NT = NT, + typename Functor +> +NT simple_mc_integrate(Functor Fx, + int dim, + int N = 10000, + volumetype voltype = SOB, + Limit LowLimit = lt, + Limit UpLimit = lt, + int walk_length = 10, + NT e = 0.1) +{ + RNG rng(dim); + return simple_mc_integrate(Fx, dim, rng, N, voltype, LowLimit, UpLimit, walk_length, e); +} +#endif diff --git a/src/volesti/include/lp_oracles/misc_lp.h b/src/volesti/include/lp_oracles/misc_lp.h new file mode 100644 index 00000000..9e8e2528 --- /dev/null +++ b/src/volesti/include/lp_oracles/misc_lp.h @@ -0,0 +1,145 @@ +#ifndef MISC_LP_H +#define MISC_LP_H + + +#include +#include +#include +#undef Realloc +#undef Free +#include "lp_lib.h" + + +// Computes the Chebychev ball of an H-polytope described by a dxd matrix A and d-dimensional vector b, s.t.: Ax<=b +/// @tparam NT Number type +/// @tparam Point Point type +/// @tparam MT Matrix type +/// @tparam VT Vector type +template +std::pair ComputeChebychevBall(MT &A, VT &b){ + + lprec *lp; + int d = A.cols(); + int Ncol=d+1, j, m=A.rows(), i; + int *colno = NULL; + + REAL *row = NULL; + std::pair exception_pair(Point(1),-1.0); + + try + { + lp = make_lp(m, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for chebychev center "< res; + + std::vector temp_p(d,0); + get_variables(lp, row); + for(j = 0; j < d; j++){ + temp_p[j]=NT(row[j]); + } + Point xc( d , temp_p.begin() , temp_p.end() ); + NT r=NT(get_objective(lp)); + res = std::pair (xc,r); + delete_lp(lp); + free(row); + free(colno); + + return res; +} + + + +#endif diff --git a/src/volesti/include/lp_oracles/solve_lp.h b/src/volesti/include/lp_oracles/solve_lp.h new file mode 100644 index 00000000..f7013e99 --- /dev/null +++ b/src/volesti/include/lp_oracles/solve_lp.h @@ -0,0 +1,296 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// VolEsti is free software: you can redistribute it and/or modify it +// under the terms of the GNU Lesser General Public License as published by +// the Free Software Foundation, either version 3 of the License, or (at +// your option) any later version. +// +// VolEsti is distributed in the hope that it will be useful, but WITHOUT +// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +// for more details. +// +// See the file COPYING.LESSER for the text of the GNU Lesser General +// Public License. If you did not receive this file along with HeaDDaCHe, +// see . + + +#ifndef SOLVE_LP_H +#define SOLVE_LP_H + + +#include +#include +#include +#undef Realloc +#undef Free +#include "lp_lib.h" + + +// compute the chebychev ball of an H-polytope described by a dxd matrix A and d-dimensional vector b, s.t.: Ax<=b +template +std::pair ComputeChebychevBall(const MT &A, const VT &b){ + + lprec *lp; + int d = A.cols(); + int Ncol=d+1, j, m=A.rows(), i; + int *colno = NULL; + + REAL *row = NULL; + std::pair exception_pair(Point(1),-1.0); + + try + { + lp = make_lp(m, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { + #ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for chebychev center "< res; + + std::vector temp_p(d,0); + get_variables(lp, row); + for(j = 0; j < d; j++){ + temp_p[j]=NT(row[j]); + } + Point xc( d , temp_p.begin() , temp_p.end() ); + NT r = NT(get_objective(lp)); + res = std::pair (xc,r); + delete_lp(lp); + + return res; +} + + +template +Point PointInIntersection(MT V1, MT V2, Point direction, bool &empty) { + + typedef typename Point::FT NT; + unsigned int d = V1.cols(); + unsigned int k1 = V1.rows(); + unsigned int k2 = V2.rows(); + unsigned int k = k1 + k2; + VT cb(k1); + lprec *lp; + int Ncol=k, *colno = NULL, j, i; + REAL *row = NULL; + Point p(d); + + try + { + lp = make_lp(d+2, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for membership "<. + + +#ifndef VPOLYORACLES_H +#define VPOLYORACLES_H + + +#include +#include +#include +#undef Realloc +#undef Free +#include "lp_lib.h" + + +// return true if q belongs to the convex hull of the V-polytope described by matrix V +// otherwise return false +template +bool memLP_Vpoly(const MT &V, const Point &q, NT *row, int *colno){ + + //typedef typename Point::FT NT; + int d=q.dimension(); + lprec *lp; + int Ncol=d+1, j, i, m=V.rows(); + m++; + + try + { + lp = make_lp(m, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for membership "<0.0){ + return false; + } + return true; +} + + + +// compute the intersection of a ray with a V-polytope +// if maxi is true compute positive lambda, when the ray is p + lambda \cdot v +// otherwise compute the negative lambda +template +NT intersect_line_Vpoly(const MT &V, const Point &p, const Point &v, + NT *conv_comb, NT *row, int *colno, bool maxi, bool zonotope){ + + int d=v.dimension(), i; + lprec *lp; + int m=V.rows(); + m++; + int Ncol=m, j, Nrows; + NT res; + if(!zonotope) { + Nrows = d+1; + } else { + Nrows = d; + } + + try + { + lp = make_lp(Nrows, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for ray-shooting "<. + + +#ifndef ZPOLYORACLES_H +#define ZPOLYORACLES_H + + +#include +#include +#include +#undef Realloc +#undef Free +#include "lp_lib.h" + + +template +bool memLP_Zonotope(const MT &V, const Point &q, NT *row, int *colno){ + + //typedef typename Point::FT NT; + int d=q.dimension(); + lprec *lp; + int Ncol=V.rows(), j, i; + + try + { + lp = make_lp(d, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for membership "< +std::pair intersect_line_zono(const MT &V, const Point &p, const Point &v, NT *row, int *colno){ + + std::pair pair_res; + int d=v.dimension(), i; + lprec *lp;//, *lp2; + int m=V.rows(); + m++; + int Ncol=m, j, Nrows; + NT res; + Nrows = d; + + try + { + lp = make_lp(Nrows, Ncol); + //lp2 = make_lp(Nrows, Ncol); + if(lp == NULL) throw false; + } + catch (bool e) { +#ifdef VOLESTI_DEBUG + std::cout<<"Could not construct Linear Program for ray-shooting "< +class DenseProductMatrix { +public: + /// Eigen matrix type + typedef Eigen::Matrix MT; + /// Eigen vector type + typedef Eigen::Matrix VT; + + /// The number of rows + int _rows; + /// The number of cols + int _cols; + + /// Pointer to matrix A + MT const *A; + /// Pointer to matrix B + MT const *B; + + /// The decomposition we will use + /// If PARTIAL_LU_DECOMPOSITION is defined, use the Eigen partial LU decomposition, + /// otherwise full. The partial is faster but assumes that the matrix has full rank. +#if defined(PARTIAL_LU_DECOMPOSITION) + typedef Eigen::PartialPivLU Decomposition; +#else + typedef Eigen::FullPivLU Decomposition; +#endif + + /// The LU decomposition of B + Decomposition Blu; + + /// Constructs an object of this class and computes the LU decomposition of B. + /// + /// \param[in] A The matrix A + /// \param[in] B The matrix B + DenseProductMatrix(MT const *A, MT const *B) : A(A), B(B) { + Blu = Decomposition(*B); + _rows = A->rows(); + _cols = B->cols(); + } + + ///Required by Spectra + /// \return The number of rows + int rows() { + return _rows; + } + + ///Required by Spectra + /// \return The number of columns + int cols() { + return _cols; + } + + /// Required by Spectra. + /// Computes the product Cx = y, i.e. @f[ (B^-1 A)v = y@$]. But B = LU, so Ax = LUy. + /// Let Ax = v, then LUy = v. Then Lw = v and finally Uy = w to get y; + /// \param[in] x_in + /// \param[out] y_out + void perform_op(NT const * x_in, NT* y_out) { + + // Declaring the vectors like this, we don't copy the values of x_in to v + // and next of y to y_out + Eigen::Map const x(const_cast(x_in), _rows); + VT const v = *A * x; + + Eigen::Map y(y_out, _rows); + y = Blu.solve(v); + } + + /// Required by arpack. + /// Computes the product Cx = y, i.e. @f[ (B^-1 A)v = y@$]. But B = LU, so Ax = LUy. + /// Let Ax = v, then LUy = v. Then Lw = v and finally Uy = w to get y; + /// \param[in] x_in + /// \param[out] y_out + void MultMv(NT * x_in, NT* y_out) { + + // Declaring the vectors like this, we don't copy the values of x_in to v + // and next of y to y_out + Eigen::Map const x(const_cast(x_in), _rows); + VT const v = *A * x; + + Eigen::Map y(y_out, _rows); + y = Blu.solve(v); + } +}; +#endif //VOLESTI_DENSEPRODUCTMATRIX_H diff --git a/src/volesti/include/matrix_operations/EigenDenseMatrix.h b/src/volesti/include/matrix_operations/EigenDenseMatrix.h new file mode 100644 index 00000000..2a3582e3 --- /dev/null +++ b/src/volesti/include/matrix_operations/EigenDenseMatrix.h @@ -0,0 +1,76 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_EIGENDENSEMATRIX_H +#define VOLESTI_EIGENDENSEMATRIX_H + +/// A wrap class to use Eigen dense matrices when solving Eigenvalue problems with ARPACK++ +/// \tparam NT Numeric Type +template +class EigenDenseMatrix { +public: + + /// Eigen matrix type + typedef Eigen::Matrix MT; + /// Eigen vector type + typedef Eigen::Matrix VT; + + /// The matrix + MT const * M; + + /// number of columns + int n; + /// number of rows + int m; + + /// \return Number of rows + int nrows() { return m;} + + /// \return Number of columns + int ncols() { return n;} + + /// \return Number of rows + int rows() { return m;} + + /// \return Number of columns + int cols() { return n;} + + /// Required by ARPACK++ : Multiplies the matrix with vector v + /// \param[in] v The input vector, for example double* + /// \param[out] w The result of M*v + void MultMv(NT* v, NT* w) { + // Declaring the vectors like this, we don't copy the values of v and after to w + Eigen::Map _v(v, m); + Eigen::Map _w(w, m); + + _w = *M * _v; + } + + /// Required by ARPACK++ : Multiplies the matrix with vector v + /// \param[in] v The input vector, for example double* + /// \param[out] w The result of M*v + void perform_op(NT* v, NT* w) { + // Declaring the vectors like this, we don't copy the values of v and after to w + Eigen::Map _v(v, m); + Eigen::Map _w(w, m); + + _w = *M * _v; + } + + + /// Constructs an object + /// \param[in] M An Eigen Matrix + EigenDenseMatrix(MT const * M) { + this->M = M; + n = M->cols(); + m = M->rows(); + } + +}; +#endif //VOLESTI_EIGENDENSEMATRIX_H diff --git a/src/volesti/include/matrix_operations/EigenvaluesProblems.h b/src/volesti/include/matrix_operations/EigenvaluesProblems.h new file mode 100644 index 00000000..4012177f --- /dev/null +++ b/src/volesti/include/matrix_operations/EigenvaluesProblems.h @@ -0,0 +1,450 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. +// Contributed and modified by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_EIGENVALUESPROBLEMS_H +#define VOLESTI_EIGENVALUESPROBLEMS_H + +/// Uncomment the solver the function minPosGeneralizedEigenvalue uses +/// Eigen solver for generalized eigenvalue problem +//#define EIGEN_EIGENVALUES_SOLVER +/// Spectra standard eigenvalue problem +#define SPECTRA_EIGENVALUES_SOLVER +/// ARPACK++ standard eigenvalues solver +//#define ARPACK_EIGENVALUES_SOLVER + +#include +#include "DenseProductMatrix.h" +#include "EigenDenseMatrix.h" + +#include "Spectra/include/Spectra/SymGEigsSolver.h" +#include "Spectra/include/Spectra/GenEigsSolver.h" + +/// Solve eigenvalues problems +/// \tparam NT Numeric Type +/// \tparam MT Matrix Type +/// \tparam VT Vector Type +template +class EigenvaluesProblems { + +}; + + +/// A specialization of the template class EigenvaluesProblems for dense Eigen matrices and vectors. +/// \tparam NT Numer Type +template +class EigenvaluesProblems, Eigen::Matrix > { +public: + /// The type for Eigen Matrix + typedef Eigen::Matrix MT; + /// The type for Eigen vector + typedef Eigen::Matrix VT; + /// The type of a complex Eigen Vector for handling eigenvectors +#if defined(EIGEN_EIGENVALUES_SOLVER) || defined (SPECTRA_EIGENVALUES_SOLVER) + typedef typename Eigen::GeneralizedEigenSolver::ComplexVectorType CVT; +#elif defined(ARPACK_EIGENVALUES_SOLVER) + typedef Eigen::Matrix CVT; +#endif + + /// The type of a pair of NT + typedef std::pair NTpair; + + + /// Find the smallest eigenvalue of M + /// \param M a symmetric matrix + /// \return smallest eigenvalue + NT findSymEigenvalue(MT const & M) { + EigenDenseMatrix _M(&M); + +//#define NOT_WORKING +#ifdef NOT_WORKING + // Creating an eigenvalue problem and defining what we need: + // the smallest eigenvalue of M. + ARNonSymStdEig > + dprob(M.cols(), 1, &_M, &EigenDenseMatrix::MultMv, std::string ("LR"), 8, 0.0, 100*15); + + // compute + if (dprob.FindEigenvectors() == 0) { + std::cout << "Failed in findSymEigenvalue\n"; + // if failed with default (and fast) parameters, try with stable (and slow) + dprob.ChangeNcv(M.cols()/10); + if (dprob.FindEigenvectors() == 0) { + std::cout << "\tFailed Again\n"; + return NT(0); + } + } + + if (!dprob.EigenvaluesFound()) { + // if failed to find eigenvalues + return NT(0); + } + + // retrieve eigenvalue of the original system + return dprob.EigenvalueReal(0); +#elif defined(SPECTRA) + // This parameter is for Spectra. It must be larger than #(requested eigenvalues) + 2 + // and smaller than the size of matrix; + int ncv = M.cols()/10 + 5; + if (ncv > M.cols()) ncv = M.cols(); + + Spectra::SymEigsSolver > eigs(&_M, 1, ncv); + // compute + eigs.init(); + eigs.compute(50000); + if(eigs.info() == Spectra::SUCCESSFUL) { + return eigs.eigenvalues()(0); + } + else { + std::cout << "Spectra failed\n"; + return NT(0); + } +#else + Eigen::SelfAdjointEigenSolver solver; + solver.compute(M, Eigen::EigenvaluesOnly); +// typename Eigen::GeneralizedEigenSolver::ComplexVectorType eivals = solver.eigenvalues(); +// NT max = eivals(0).real(); +// +// for (int i = 1; i < eivals.rows(); i++) +// if (eivals(i).real() > max) +// max = eivals(i).real(); + + return solver.eigenvalues().maxCoeff(); +#endif + } + + /// Find the minimum positive and maximum negative eigenvalues of the generalized eigenvalue + /// problem A + lB, where A, B symmetric and A negative definite. + /// \param[in] A Input matrix + /// \param[in] B Input matrix + /// \return The pair (minimum positive, maximum negative) of eigenvalues + NTpair symGeneralizedProblem(MT const & A, MT const & B) const { + + int matrixDim = A.rows(); + + // Spectra solves Xv=lYv, where Y positive definite + // Set X = B, Y=-A. Then, the eigenvalues we want are the minimum negative + // and maximum positive eigenvalues of Xv=lYv. + + // Construct matrix operation object using the wrapper classes provided by Spectra + Spectra::DenseSymMatProd op(B); + Spectra::DenseCholesky Bop(-A); + + // Construct generalized eigen solver object + // requesting the minmum negative and largest positive eigenvalues + Spectra::SymGEigsSolver, Spectra::DenseCholesky, Spectra::GEIGS_CHOLESKY> + geigs(&op, &Bop, 2, 5 < matrixDim ? 5 : matrixDim); + + // Initialize and compute + geigs.init(); + int nconv = geigs.compute(); + + // Retrieve results + if (geigs.info() != Spectra::SUCCESSFUL) + return {NT(0), NT(0)}; + + Eigen::VectorXd evalues; + double lambdaMinPositive, lambdaMaxNegative; + + evalues = geigs.eigenvalues(); + + // get the eigenvalues of the original problem + lambdaMinPositive = 1 / evalues(0); + lambdaMaxNegative = 1 / evalues(1); + + return {lambdaMinPositive, lambdaMaxNegative}; + } + + NT minPosLinearEigenvalue(MT const & A, MT const & B, VT &eigvec) { + int matrixDim = A.rows(); + double lambdaMinPositive; + + Spectra::DenseSymMatProd op(B); + Spectra::DenseCholesky Bop(-A); + + // Construct generalized eigen solver object, requesting the largest three generalized eigenvalues + Spectra::SymGEigsSolver, Spectra::DenseCholesky, Spectra::GEIGS_CHOLESKY> + geigs(&op, &Bop, 1, 15 < matrixDim ? 15 : matrixDim); + + // Initialize and compute + geigs.init(); + int nconv = geigs.compute(); + + VT evalues; + if (geigs.info() == Spectra::SUCCESSFUL) { + evalues = geigs.eigenvalues(); + eigvec = geigs.eigenvectors().col(0); + } + + lambdaMinPositive = 1 / evalues(0); + + return lambdaMinPositive; + } + + /// Finds the minimum positive real eigenvalue of the generalized eigenvalue problem A + lB and + /// the corresponding eigenvector. + /// If the macro EIGEN_EIGENVALUES_SOLVER is defined, the Generalized Solver of Eigen is used. + /// Otherwise, we transform the generalized to a standard eigenvalue problem and use Spectra. + /// Warning: With Spectra we might get a value smaller than the minimum positive real eigenvalue (the real part + /// of a complex eigenvalue). + /// No restriction on the matrices! + /// \param[in] A Input matrix + /// \param[in] B Input matrix + /// \param[out] eigenvector The eigenvector corresponding to the minimum positive eigenvalue + /// \return The minimum positive eigenvalue + NT minPosGeneralizedEigenvalue(MT const & A, MT const & B, CVT& eigenvector) { + NT lambdaMinPositive = std::numeric_limits::max(); + +#if defined(EIGEN_EIGENVALUES_SOLVER) + // use the Generalized eigenvalue solver of Eigen + + // compute generalized eigenvalues with Eigen solver + Eigen::GeneralizedEigenSolver ges(A, -B); + + // retrieve minimum positive eigenvalue + typename Eigen::GeneralizedEigenSolver::ComplexVectorType alphas = ges.alphas(); + VT betas = ges.betas(); + int index = 0; + + for (int i = 0; i < alphas.rows(); i++) { + + if (betas(i) == 0 || alphas(i).imag() != 0) + continue; + + double lambda = alphas(i).real() / betas(i); + if (lambda > 0 && lambda < lambdaMinPositive) { + lambdaMinPositive = lambda; + index = i; + } + } + + // retrieve corresponding eigenvector + eigenvector = ges.eigenvectors().col(index); +#elif defined(SPECTRA_EIGENVALUES_SOLVER) + // Transform the problem to a standard eigenvalue problem and use the general eigenvalue solver of Spectra + + // This makes the transformation to standard eigenvalue problem. See class for more info. + // We have the generalized problem A + lB, or Av = -lBv + // This class computes the matrix product vector Mv, where M = -B * A^[-1] + MT _B = -1 * B; // TODO avoid this allocation + DenseProductMatrix M(&_B, &A); + + // This parameter is for Spectra. It must be larger than #(requested eigenvalues) + 2 + // and smaller than the size of matrix; + int ncv = 3; + + // Prepare to solve Mx = (1/l)x + // we want the smallest positive eigenvalue in the original problem, + // so in this the largest positive eigenvalue; + Spectra::GenEigsSolver > eigs(&M, 1, ncv); + + // compute + eigs.init(); + eigs.compute(); + + //retrieve result and invert to get required eigenvalue of the original problem + if (eigs.info() != Spectra::SUCCESSFUL) { + eigenvector.setZero(A.rows()); + return NT(0); + } + + lambdaMinPositive = 1/((eigs.eigenvalues())(0).real()); + + // retrieve corresponding eigenvector + int matrixDim = A.rows(); + eigenvector.resize(matrixDim); + for (int i = 0; i < matrixDim; i++) + eigenvector(i) = (eigs.eigenvectors()).col(0)(i); + +#elif defined(ARPACK_EIGENVALUES_SOLVER) + // Transform the problem to a standard eigenvalue problem and use the general eigenvalue solver of ARPACK++ + + // This makes the transformation to standard eigenvalue problem. See class for more info. + // We have the generalized problem A + lB, or Av = -lBv + // This class computes the matrix product vector Mv, where M = -B * A^[-1] + MT _B = -1 * B; // TODO avoid this allocation + DenseProductMatrix M(&_B, &A); + + // Creating an eigenvalue problem and defining what we need: + // the eigenvector of A with largest real. + ARNonSymStdEig > + + dprob(A.cols(), 1, &M, &DenseProductMatrix::MultMv, std::string ("LR"), 8 op(B); + Spectra::DenseCholesky Bop(-A); + + // Construct generalized eigen solver object, requesting the largest three generalized eigenvalues + Spectra::SymGEigsSolver, Spectra::DenseCholesky, Spectra::GEIGS_CHOLESKY> + geigs(&op, &Bop, 1, 15 < matrixDim ? 15 : matrixDim); + + // Initialize and compute + geigs.init(); + int nconv = geigs.compute(); + + // Retrieve results + VT evalues; + + if (geigs.info() == Spectra::SUCCESSFUL) { + evalues = geigs.eigenvalues(); + eigvec = geigs.eigenvectors().col(0); + } + + lambdaMinPositive = 1 / evalues(0); + + return lambdaMinPositive; + } + + /// Transform the quadratic eigenvalue problem \[At^2 + Bt + c\] to + /// the generalized eigenvalue problem X+lY. + /// If the updateOnly flag is false, compute matrices X,Y from scratch; + /// otherwise update them. + /// \param[in] A + /// \param[in] B + /// \param[in] C + /// \param[in, out] X + /// \param[in, out] Y + /// \param[in, out] updateOnly True if X,Y were previously computed and only B,C changed + void linearization(const MT &A, const MT &B, const MT &C, MT &X, MT &Y, bool &updateOnly) { + unsigned int matrixDim = A.rows(); + + // check if the matrices X,Y are computed. + //if yes, update them; otherwise compute them from scratch + if (!updateOnly) { + X.resize(2 * matrixDim, 2 * matrixDim); + Y.resize(2 * matrixDim, 2 * matrixDim); + + Y.block(matrixDim, matrixDim, matrixDim, matrixDim) = -1 * C; + Y.block(0, matrixDim, matrixDim, matrixDim) = MT::Zero(matrixDim, matrixDim); + Y.block(matrixDim, 0, matrixDim, matrixDim) = MT::Zero(matrixDim, matrixDim); + Y.block(0, 0, matrixDim, matrixDim) = A; + + X.block(0, matrixDim, matrixDim, matrixDim) = C; + X.block(0, 0, matrixDim, matrixDim) = B; + X.block(matrixDim, 0, matrixDim, matrixDim) = C; + X.block(matrixDim, matrixDim, matrixDim, matrixDim) = MT::Zero(matrixDim, matrixDim); + } else { + Y.block(matrixDim, matrixDim, matrixDim, matrixDim) = -1 * C; + + X.block(0, matrixDim, matrixDim, matrixDim) = C; + X.block(0, 0, matrixDim, matrixDim) = B; + X.block(matrixDim, 0, matrixDim, matrixDim) = C; + } + } + + /// Find the minimum positive real eigenvalue of the quadratic eigenvalue problem \[At^2 + Bt + c\]. + /// First transform it to the generalized eigenvalue problem X+lY. + /// If the updateOnly flag is false, compute matrices X,Y from scratch; + /// otherwise only update them. + /// \param[in] A Input matrix + /// \param[in] B Input matrix + /// \param[in] C Input matrix + /// \param[in, out] X + /// \param[in, out] Y + /// \param[out] eigenvector The eigenvector corresponding to the minimum positive eigenvalue + /// \param[in, out] updateOnly True if X,Y were previously computed and only B,C changed + /// \return Minimum positive eigenvalue + NT minPosQuadraticEigenvalue(MT const & A, MT const &B, MT const &C, MT &X, MT &Y, VT &eigenvector, bool &updateOnly) { + // perform linearization and create generalized eigenvalue problem X+lY + linearization(A, B, C, X, Y, updateOnly); + + // solve generalized problem + CVT eivector; + NT lambdaMinPositive = minPosGeneralizedEigenvalue(X, Y, eivector); + + if (lambdaMinPositive == 0) + return 0; + + int matrixDim = A.rows(); + + // the eivector has dimension 2*matrixDim + // while the eigenvector of the original problem has dimension matrixDim + // retrieve the eigenvector by keeping only #matrixDim coordinates. + eigenvector.resize(matrixDim); + +#if defined(EIGEN_EIGENVALUES_SOLVER) || defined (SPECTRA_EIGENVALUES_SOLVER) + for (int i = 0; i < matrixDim; i++) + eigenvector(i) = eivector(matrixDim + i).real(); +#elif defined(ARPACK_EIGENVALUES_SOLVER) + for (int i = 0; i < matrixDim; i++) + eigenvector(i) = eivector(matrixDim + i); +#endif + + return lambdaMinPositive; + } + + // Using LDLT decomposition to check membership + // Faster than computing the largest eigenvalue with Spectra + // more numerically stable for singular matrices + bool isPositiveSemidefinite(MT const &A) const { + Eigen::LDLT A_ldlt(A); + if (A_ldlt.info() != Eigen::NumericalIssue && A_ldlt.isPositive()) + return true; + return false; + } + + /// Minimum positive eigenvalue of the generalized eigenvalue problem A - lB + /// Use Eigen::GeneralizedSelfAdjointEigenSolver ges(B,A) (faster) + /// \param[in] A: symmetric positive definite matrix + /// \param[in] B: symmetric matrix + /// \return The minimum positive eigenvalue and the corresponding eigenvector + NT minPosLinearEigenvalue_EigenSymSolver(MT const & A, MT const & B, VT &eigvec) const { + NT lambdaMinPositive = NT(0); + Eigen::GeneralizedSelfAdjointEigenSolver ges(B,A); + lambdaMinPositive = 1/ges.eigenvalues().reverse()[0]; + eigvec = ges.eigenvectors().reverse().col(0).reverse(); + return lambdaMinPositive; + } +}; + +#endif //VOLESTI_EIGENVALUESPROBLEMS_H diff --git a/src/volesti/include/misc/linear_extensions.h b/src/volesti/include/misc/linear_extensions.h new file mode 100644 index 00000000..13866816 --- /dev/null +++ b/src/volesti/include/misc/linear_extensions.h @@ -0,0 +1,92 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2018 Vissarion Fisikopoulos + +// Licensed under GNU LGPL.3, see LICENCE file + +inline void linear_extensions_to_order_polytope(std::istream &is, + std::ostream &os){ + + std::string ns; + //read n: the number of elements in the poset + std::getline(is, ns, ' '); + std::istringstream buffer(ns); + int n; + buffer >> n; + //read m: the number of edges/relations in the poset + std::getline(is, ns, '\n'); + std::istringstream bufferm(ns); + int m; + bufferm >> m; + + os << "order_"< ipoint; + point.erase( std::remove( point.begin(), point.end(), ' ' ), + point.end() ); + point.erase( std::remove( point.begin(), point.end(), '[' ), + point.end() ); + std::string coord; + if (point[0]==',') + point.erase(0,1); + std::stringstream stream(point); + if (!point.empty()){ + int i=1; + os<<" 0 "; + //read point 1 + std::getline(stream, coord, ','); + std::istringstream buffer(coord); + int temp; + buffer >> temp; + ipoint.push_back(temp); + //os << temp << " "; + //write inequality 1 + for(;i> temp2; + ipoint.push_back(temp2); + //os << temp2 << "\n"; + //write inequality 2 + for(;i +#include +#include +#include +#include "poset.h" + +//function to print rounding to double coordinates +template +void round_print(T p) { + std::cout<<"test version.."< Vpolys; + +int Minkowski_sum_naive(V_polytope &P1, V_polytope &P2, V_polytope &Msum){ + std::cout<<(!P1.empty() && !P2.empty())< ep(P1[0].dimension()); + ep.insert(Msum_all.begin(),Msum_all.end()); + //std::vector extreme_points; + ep.get_extreme_points(std::back_inserter(Msum)); + return Msum.size(); + } + return -1; +} +*/ + +// polymake file to compute exact volume +template +void print_polymake_volfile(T &P, + std::ostream& os){ + std::cout<<"test version.."<;\n"; + os << "$p->POINTS=<<'.';\n"; + for (typename T::iterator vit = P.begin(); vit != P.end(); vit++){ + os << "1 "; + for (Point::Cartesian_const_iterator cit=vit->cartesian_begin(); + cit != vit->cartesian_end(); + cit++){ + os << *cit; + if (cit - vit->cartesian_begin() != vit->dimension()-1) + os << " "; + } + //os << "|" << vit->point().index(); + os << "\n"; + } + os << ".\n"; + os << "print ' ';\n"; + os << "print $p->N_POINTS;\n"; + os << "print ' ';\n"; + os << "print $p->N_VERTICES;\n"; + os << "print ' ';\n"; + os << "print $p->DIM;\n"; + os << "print ' ';\n"; + os << "my $t0 = [gettimeofday];\n"; + os << "my $f=$p->VOLUME;\n"; + os << "print $f;\n"; + os << "print ' ';\n"; + os << "print tv_interval($t0,[gettimeofday]);\n"; + os << "print \"\n\";\n"; + os << std::endl; +}*/ + +// polymake file to compute exact volume +template +void print_polymake_volfile2(T &P, + std::ostream& os){ + std::cout<<"test version.."<INEQUALITIES=<<'.';\n"; + //os << "my $p=new Polytope;\n"; + //os << "$p->POINTS=<<'.';\n"; + for (typename T::iterator vit = P.begin(); vit != P.end(); vit++){ + Hyperplane::Coefficient_const_iterator cit_end = vit->coefficients_end(); + os << *(--cit_end)<<" "; + //os << "0 "; + Hyperplane::Coefficient_const_iterator cit = vit->coefficients_begin(); + //++cit; + for (; cit != cit_end; cit++){ + //std::cout<<*cit<<" "; + os <<(*cit)<<" "; + if (cit - vit->coefficients_begin() != vit->dimension()-1) + os << " "; + } + //os << "|" << vit->point().index(); + os << "\n"; + } + os << ".\n"; + //$p=new Polytope(INEQUALITIES=>$inequalities); + + os << "print ' ';\n"; + os << "print $p->N_POINTS;\n"; + os << "print ' ';\n"; + os << "print $p->N_VERTICES;\n"; + os << "print ' ';\n"; + os << "print $p->DIM;\n"; + os << "print ' ';\n"; + os << "my $t0 = [gettimeofday];\n"; + os << "my $f=$p->VOLUME;\n"; + os << "print $f;\n"; + os << "print ' ';\n"; + os << "print tv_interval($t0,[gettimeofday]);\n"; + os << "print \"\n\";\n"; + os << std::endl; +}*/ +template +void read_pointset(std::istream &is, + std::vector > &Input){ + + std::string point; + + while(!std::getline(is, point, '\n').eof()) { + //std::cout< input; + while (found2!=std::string::npos || point[found]=='-') + { + //std::cout<<"*"<<(point[found]!='-')<<"*"< +void read_objective(std::istream &is, std::vector &obj) { + NT element; + while (is >> element) obj.push_back(element); + +} + +template +std::pair read_inner_ball(std::istream &is) { + std::vector obj; + read_objective(is, obj); + unsigned int dim = obj.size() - 1; + Point center(dim); + NT radius = obj[dim]; + for (unsigned int i = 0; i < dim; i++) { + center.set_coord(i, obj[i]); + } + + return std::make_pair(center, radius); +} + +/* read a poset given in the following format: + - First line contains a single positive integer 'n' - number of elements + - Next `m` lines follow containing a pair 'i j' in each line to signify A_i <= A_j + i.e i_th element is less than or equal to the j_th element +*/ +Poset read_poset_from_file(std::istream &data_file) { + typedef typename Poset::RT RT; + typedef typename Poset::RV RV; + + // read number of elements + unsigned int n; + data_file >> n; + + // read relations line by line + RT curr_relation; + RV relations; + while(data_file >> curr_relation.first >> curr_relation.second) + relations.push_back(curr_relation); + + return Poset(n, relations); +} + + +// read a poset given as an adjacency matrix +std::pair read_poset_from_file_adj_matrix(std::istream &in) { + typedef typename Poset::RV RV; + + RV edges; + unsigned int x, n = 0; + + // read a single line + std::string line; + std::getline(in, line); + std::stringstream line_ss(line); + while(line_ss >> x) { + if(x) { + edges.emplace_back(0, n); + } + ++n; + } + + // read rest of the lines + for(unsigned int a = 1; a < n; ++a) { + for(unsigned int b = 0; b < n; ++b) { + if(!(in >> x)) { + std::cerr << "Invalid adjacency matrix"; + return std::pair(false, Poset()); + } + + if(x) { + edges.emplace_back(a, b); + } + } + } + + return std::pair(true, Poset(n, edges)); +} + +#endif //MISC_H diff --git a/src/volesti/include/misc/poset.h b/src/volesti/include/misc/poset.h new file mode 100644 index 00000000..69ba02ec --- /dev/null +++ b/src/volesti/include/misc/poset.h @@ -0,0 +1,131 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2021 Vaibhav Thakkar + +// Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef POSET_H +#define POSET_H + +#include +#include +#include + +/// A class to represent a partial order set aka Poset +class Poset { +public: + typedef std::pair RT; // relation type + typedef std::vector RV; // relations vector + +private: + unsigned int n; // elements will be from 0 to n-1 + RV order_relations; // pairs of form a <= b + +public: + Poset() {} + + Poset(unsigned int _n, RV& _order_relations) : + n(_n), order_relations(verify(_order_relations, n)) + {} + + + // verify if the relations are valid + static RV verify(const RV& relations, unsigned int n) + { + for (int i = 0; i < relations.size(); i++) { + if (relations[i].first < 0 || relations[i].first >= n) + throw "invalid elements in order relations"; + + if (relations[i].second < 0 || relations[i].second >= n) + throw "invalid elements in order relations"; + } + + // TODO: Check if corresponding DAG is actually acyclic + + return relations; + } + + + void print() + { + std::cout << "Number of elements: " << num_elem() << std::endl; + + unsigned int count = num_relations(); + for(int i=0; i + bool is_in(const std::vector& pt_coeffs, NT tol=NT(0)) const + { + for (int i = 0; i < order_relations.size(); i++) { + unsigned int a = order_relations[i].first; + unsigned int b = order_relations[i].second; + + if (! (pt_coeffs[a] - pt_coeffs[b] <= tol) ) { + return false; + } + } + + return true; + } + + + std::vector topologically_sorted_list() const + { + std::vector > adjList(n); + std::vector indegree(n, 0); + + for(auto x: order_relations) { + adjList[x.first].push_back(x.second); + indegree[x.second] += 1; + } + + std::queue q; + for(unsigned int i=0; i res; + while(!q.empty()) { + unsigned int curr = q.front(); + res.push_back(curr); + q.pop(); + + for(unsigned int i=0; i +#include +#include +#include +#include +#include +#include +#include + +/** + * Used to specify the column format + */ +enum class VariadicTableColumnFormat +{ + AUTO, + SCIENTIFIC, + FIXED, + PERCENT +}; + +/** + * A class for "pretty printing" a table of data. + * + * Requries C++11 (and nothing more) + * + * It's templated on the types that will be in each column + * (all values in a column must have the same type) + * + * For instance, to use it with data that looks like: "Fred", 193.4, 35, "Sam" + * with header names: "Name", "Weight", "Age", "Brother" + * + * You would invoke the table like so: + * VariadicTable vt("Name", "Weight", "Age", "Brother"); + * + * Then add the data to the table: + * vt.addRow("Fred", 193.4, 35, "Sam"); + * + * And finally print it: + * vt.print(); + */ +template +class VariadicTable +{ +public: + /// The type stored for each row + typedef std::tuple DataTuple; + + /** + * Construct the table with headers + * + * @param headers The names of the columns + * @param static_column_size The size of columns that can't be found automatically + */ + VariadicTable(std::vector headers, + unsigned int static_column_size = 0, + unsigned int cell_padding = 1) + : _headers(headers), + _num_columns(std::tuple_size::value), + _static_column_size(static_column_size), + _cell_padding(cell_padding) + { + assert(headers.size() == _num_columns); + } + + /** + * Add a row of data + * + * Easiest to use like: + * table.addRow({data1, data2, data3}); + * + * @param data A Tuple of data to add + */ + void addRow(Ts... entries) { _data.emplace_back(std::make_tuple(entries...)); } + + /** + * Pretty print the table of data + */ + template + void print(StreamType & stream) + { + size_columns(); + + // Start computing the total width + // First - we will have _num_columns + 1 "|" characters + unsigned int total_width = _num_columns + 1; + + // Now add in the size of each colum + for (auto & col_size : _column_sizes) + total_width += col_size + (2 * _cell_padding); + + // Print out the top line + stream << std::string(total_width, '-') << "\n"; + + // Print out the headers + stream << "|"; + for (unsigned int i = 0; i < _num_columns; i++) + { + // Must find the center of the column + auto half = _column_sizes[i] / 2; + half -= _headers[i].size() / 2; + + stream << std::string(_cell_padding, ' ') << std::setw(_column_sizes[i]) << std::left + << std::string(half, ' ') + _headers[i] << std::string(_cell_padding, ' ') << "|"; + } + + stream << "\n"; + + // Print out the line below the header + stream << std::string(total_width, '-') << "\n"; + + // Now print the rows of the table + for (auto & row : _data) + { + stream << "|"; + print_each(row, stream); + stream << "\n"; + } + + // Print out the line below the header + stream << std::string(total_width, '-') << "\n"; + } + + /** + * Set how to format numbers for each column + * + * Note: this is ignored for std::string columns + * + * @column_format The format for each column: MUST be the same length as the number of columns. + */ + void setColumnFormat(const std::vector & column_format) + { + assert(column_format.size() == std::tuple_size::value); + + _column_format = column_format; + } + + /** + * Set how many digits of precision to show for floating point numbers + * + * Note: this is ignored for std::string columns + * + * @column_format The precision for each column: MUST be the same length as the number of columns. + */ + void setColumnPrecision(const std::vector & precision) + { + assert(precision.size() == std::tuple_size::value); + _precision = precision; + } + +protected: + // Just some handy typedefs for the following two functions + typedef decltype(&std::right) right_type; + typedef decltype(&std::left) left_type; + + // Attempts to figure out the correct justification for the data + // If it's a floating point value + template ::type>::value>::type> + static right_type justify(int /*firstchoice*/) + { + return std::right; + } + + // Otherwise + template + static left_type justify(long /*secondchoice*/) + { + return std::left; + } + + /** + * These three functions print out each item in a Tuple into the table + * + * Original Idea From From https://stackoverflow.com/a/26908596 + * + * BTW: This would all be a lot easier with generic lambdas + * there would only need to be one of this sequence and then + * you could pass in a generic lambda. Unfortunately, that's C++14 + */ + + /** + * This ends the recursion + */ + template + void print_each(TupleType &&, + StreamType & /*stream*/, + std::integral_constant< + size_t, + std::tuple_size::type>::value>) + { + } + + /** + * This gets called on each item + */ + template ::type>::value>::type> + void print_each(TupleType && t, StreamType & stream, std::integral_constant) + { + auto & val = std::get(t); + + // Set the precision + if (!_precision.empty()) + { + assert(_precision.size() == + std::tuple_size::type>::value); + + stream << std::setprecision(_precision[I]); + } + + // Set the format + if (!_column_format.empty()) + { + assert(_column_format.size() == + std::tuple_size::type>::value); + + if (_column_format[I] == VariadicTableColumnFormat::SCIENTIFIC) + stream << std::scientific; + + if (_column_format[I] == VariadicTableColumnFormat::FIXED) + stream << std::fixed; + + if (_column_format[I] == VariadicTableColumnFormat::PERCENT) + stream << std::fixed << std::setprecision(2); + } + + stream << std::string(_cell_padding, ' ') << std::setw(_column_sizes[I]) + << justify(0) << val << std::string(_cell_padding, ' ') << "|"; + + // Unset the format + if (!_column_format.empty()) + { + // Because "stream << std::defaultfloat;" won't compile with old GCC or Clang + stream.unsetf(std::ios_base::floatfield); + } + + // Recursive call to print the next item + print_each(std::forward(t), stream, std::integral_constant()); + } + + /** + * his is what gets called first + */ + template + void print_each(TupleType && t, StreamType & stream) + { + print_each(std::forward(t), stream, std::integral_constant()); + } + + /** + * Try to find the size the column will take up + * + * If the datatype has a size() member... let's call it + */ + template + size_t sizeOfData(const T & data, decltype(((T *)nullptr)->size()) * /*dummy*/ = nullptr) + { + return data.size(); + } + + /** + * Try to find the size the column will take up + * + * If the datatype is an integer - let's get it's length + */ + template + size_t sizeOfData(const T & data, + typename std::enable_if::value>::type * /*dummy*/ = nullptr) + { + if (data == 0) + return 1; + + return std::log10(data) + 1; + } + + /** + * If it doesn't... let's just use a statically set size + */ + size_t sizeOfData(...) { return _static_column_size; } + + /** + * These three functions iterate over the Tuple, find the printed size of each element and set it + * in a vector + */ + + /** + * End the recursion + */ + template + void size_each(TupleType &&, + std::vector & /*sizes*/, + std::integral_constant< + size_t, + std::tuple_size::type>::value>) + { + } + + /** + * Recursively called for each element + */ + template ::type>::value>::type> + void + size_each(TupleType && t, std::vector & sizes, std::integral_constant) + { + sizes[I] = sizeOfData(std::get(t)); + + // Override for Percent + if (!_column_format.empty()) + if (_column_format[I] == VariadicTableColumnFormat::PERCENT) + sizes[I] = 6; // 100.00 + + // Continue the recursion + size_each(std::forward(t), sizes, std::integral_constant()); + } + + /** + * The function that is actually called that starts the recursion + */ + template + void size_each(TupleType && t, std::vector & sizes) + { + size_each(std::forward(t), sizes, std::integral_constant()); + } + + /** + * Finds the size each column should be and set it in _column_sizes + */ + void size_columns() + { + _column_sizes.resize(_num_columns); + + // Temporary for querying each row + std::vector column_sizes(_num_columns); + + // Start with the size of the headers + for (unsigned int i = 0; i < _num_columns; i++) + _column_sizes[i] = _headers[i].size(); + + // Grab the size of each entry of each row and see if it's bigger + for (auto & row : _data) + { + size_each(row, column_sizes); + + for (unsigned int i = 0; i < _num_columns; i++) + _column_sizes[i] = std::max(_column_sizes[i], column_sizes[i]); + } + } + + /// The column headers + std::vector _headers; + + /// Number of columns in the table + unsigned int _num_columns; + + /// Size of columns that we can't get the size of + unsigned int _static_column_size; + + /// Size of the cell padding + unsigned int _cell_padding; + + /// The actual data + std::vector _data; + + /// Holds the printable width of each column + std::vector _column_sizes; + + /// Column Format + std::vector _column_format; + + /// Precision For each column + std::vector _precision; +}; diff --git a/src/volesti/include/nlp_oracles/nlp_hpolyoracles.hpp b/src/volesti/include/nlp_oracles/nlp_hpolyoracles.hpp new file mode 100644 index 00000000..d7394ca2 --- /dev/null +++ b/src/volesti/include/nlp_oracles/nlp_hpolyoracles.hpp @@ -0,0 +1,572 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +/* This file contains the implementation of boundary oracles between an + d-dimensional H-polytope P: Ax <= b with a curve p(t) = sum c_j phi_j(t) + where c_j are d-dimensional coefficients and phi_j(t) are basis functions + (e.g. polynomials, rational functions, splines). The boundary oracle + returns one root (in general multiple roots exist) assuming that for t >= t_p + the curve p(t) penetrates the H-polytope. The problem reduces to the following + non-linear optimization problem for finding the maximum t such that p(t) penetrates + the polytope where t lies inside [t0, t0 + eta] for some eta > 0 + + max t + + subject to + t >= t0 + t <= t0 + eta + A p(t) <= b + + The second constraint can be rewritten as (A*C) * Phi <= b which is eventually + the optimization problem we solve where the vector Phi contains all the basis + functions and C has c_j's as columns. + + The second optimization problem that can be solved is the following + + min_i min t_i + + subject to + + t0 <= t_i <= t0 + eta + A_i^T p(t_i) = b_i + A_j^T p(t_i) <= b_i j neq i + + + + We use interior-point methods to solve the non-linear optimization program + using COIN-OR ipopt and the ifopt interface for Eigen + ipopt. + +*/ + +#ifndef NLP_HPOLYORACLES_HPP +#define NLP_HPOLYORACLES_HPP + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "root_finders/root_finders.hpp" +#define MAX_NR_TRIES 10000 + +#define POLYTOL 1e-7 + +#ifndef isnan + using std::isnan; +#endif + +#ifndef isinf + using std::isinf; +#endif + + +using namespace ifopt; + +/// Define the variable t we use in the optimization +/// \tparam VT Vector Type +/// \tparam NT Numeric Type +template +class HPolyOracleVariables : public VariableSet { +public: + NT t, tb, eta; + + HPolyOracleVariables(NT t_prev, NT tb_, NT eta_=-1): + VariableSet(1, "t"), t(t_prev), tb(tb_), eta(eta_) {}; + + void SetVariables(const VT& T) override { + t = T(0); + } + + VectorXd GetValues() const override { + VectorXd T(1); + T(0) = t; + return T; + } + + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + NT tu = eta > 0 ? NT(tb + eta) : NT(inf); + + bounds.at(0) = Bounds(tb, tu); + return bounds; + }; + +}; + +/// Define the cost function f(t) = t (ipopt takes minimization so it is -t) +/// \tparam VT Vector Type +/// \tparam NT Numeric Type +template +class HPolyOracleCost : public CostTerm { +public: + std::string method; + + HPolyOracleCost(std::string method_) : + CostTerm("h_poly_cost"), method(method_) {}; + + NT GetCost() const override { + VectorXd T = GetVariables()->GetComponent("t")->GetValues(); + if (method == "max_pos") return - T(0); + else if (method == "min_pos") return T(0); + } + + void FillJacobianBlock (std::string var_set, Jacobian& jac) const override { + if (var_set == "t") { + if (method == "max_pos") jac.coeffRef(0, 0) = (NT) (-1.0); + else if (method == "min_pos") jac.coeffRef(0, 0) = (NT) (1.0); + } + } + +}; + +/// Define the feasibility constraint A p(t) <= b which we translate +/// to (A * C) * Phi <= b +/// \tparam MT Matrix Type +/// \tparam VT Vector Type +/// \tparam NT Numeric Type +/// \tparam bfunc feasibility constraint type +template +class HPolyOracleFeasibility : public ConstraintSet { +public: + MT &C; + VT &b; + bfunc phi, grad_phi; + NT t0; + int m, M; + std::string method; + int index; + int ignore_facet; + + HPolyOracleFeasibility( + MT &C_, + VT &b_, + NT t0_, + bfunc basis, + bfunc basis_grad, + std::string method_, + int i, + int ignore_facet_=-1) : + C(C_), b(b_), t0(t0_), phi(basis), grad_phi(basis_grad), + ConstraintSet(C_.rows(), "h_poly_feasibility"), + method(method_), index(i), ignore_facet(ignore_facet_) { + m = C_.rows(); + M = C_.cols(); + }; + + // Define bounds for feasibility + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + for (int i = 0; i < m; i++) { + if (method == "min_pos" && i == index) { + bounds.at(i) = Bounds(b(i), b(i)); + } + else { + bounds.at(i) = Bounds((NT) (-inf), b(i)); + } + + if (i == ignore_facet) { + bounds.at(i) = Bounds((NT) (-inf), (NT) (inf)); + } + + } + return bounds; + } + + VectorXd GetValues() const override { + VectorXd T = GetVariables()->GetComponent("t")->GetValues(); + NT t = T(0); + VectorXd phis; + phis.resize(M); + + for (int i = 0; i < M; i++) { + phis(i) = (NT) (phi(t, t0, i, M)); + } + return C * phis; + } + + // Calculate jacobian matrix of constraints + void FillJacobianBlock (std::string var_set, Jacobian& jac_block) const override { + + if (var_set == "t") { + VectorXd T = GetVariables()->GetComponent("t")->GetValues(); + NT t = T(0); + NT temp; + + for (int i = 0; i < m; i++) { + jac_block.coeffRef(i, 0) = NT(0); + for (int j = 0; j < M; j++) { + temp = grad_phi(t, t0, j, M); + if (isinf(temp)) continue; + else jac_block.coeffRef(i, 0) += temp; + } + } + } + } + + +}; + +/// Oracle that uses the COIN-OR ipopt solver +/// \tparam Polytope Polytope Type +/// \tparam bfunc feasibility constraint type +template +struct IpoptHPolyoracle { + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType Point; + + + std::tuple apply( + NT t_prev, + NT t0, + NT eta, + MT &A, + VT &b, + Polytope &P, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + int ignore_facet=-1, + std::string solution="max_pos") + { + + MT C, C_tmp; + C_tmp.resize(coeffs[0].dimension(), coeffs.size()); + + + for (int i = 0; i < coeffs.size(); i++) { + C_tmp.col(i) = coeffs[i].getCoefficients(); + } + + // C_tmp: dimension x num_coeffs + // A: constraints x dimension + // C: constraints x num_coeffs + C = A * C_tmp; + + // Initialize COIN-OR ipopt solver + IpoptSolver ipopt; + ipopt.SetOption("linear_solver", "mumps"); + ipopt.SetOption("jacobian_approximation", "exact"); + ipopt.SetOption("tol", POLYTOL); + ipopt.SetOption("acceptable_tol", 100 * POLYTOL); + ipopt.SetOption("max_iter", 1000000); + + ipopt.SetOption("print_level", 0); + ipopt.SetOption("sb", "yes"); + + NT t, t_tmp; + + if (solution == "max_pos") { + + Problem nlp; + std::shared_ptr> + hpolyoraclevariables (new HPolyOracleVariables(t_prev, t0, eta)); + std::shared_ptr> + hpolyoraclecost (new HPolyOracleCost(solution)); + std::shared_ptr> + hpolyoraclefeasibility (new HPolyOracleFeasibility + (C, b, t0, phi, grad_phi, "max_pos", 0, ignore_facet)); + + nlp.AddVariableSet (hpolyoraclevariables); + nlp.AddCostSet (hpolyoraclecost); + nlp.AddConstraintSet(hpolyoraclefeasibility); + + ipopt.Solve(nlp); + + t = nlp.GetOptVariables()->GetValues()(0); + + } + + if (solution == "min_pos") { + int m = A.rows(); + + t = eta > 0 ? t0 + eta : std::numeric_limits::max(); + + for (int i = 0; i < m; i++) { + + if (i == ignore_facet) continue; + + Problem nlp; + std::shared_ptr> + hpolyoraclevariables (new HPolyOracleVariables(t_prev, t0, eta)); + std::shared_ptr> + hpolyoraclecost (new HPolyOracleCost(solution)); + std::shared_ptr> + hpolyoraclefeasibility (new HPolyOracleFeasibility + (C, b, t0, phi, grad_phi, "min_pos", i)); + + nlp.AddVariableSet (hpolyoraclevariables); + nlp.AddCostSet (hpolyoraclecost); + nlp.AddConstraintSet(hpolyoraclefeasibility); + + t_tmp = nlp.GetOptVariables()->GetValues()(0); + + std::cout << "t is " << t_tmp << std::endl; + + if (t_tmp < t && t_tmp > t0) t = t_tmp; + + + } + } + + Point p(coeffs[0].dimension()); + + for (unsigned int i = 0; i < coeffs.size(); i++) { + p += phi(t, t0, i, coeffs.size()) * coeffs[i]; + } + + const NT* b_data = b.data(); + + int f_min = -1; + NT dist_min = std::numeric_limits::max(); + + for (int i = 0; i < A.rows(); i++) { + if (*b_data == 0 && std::abs(*b_data - A.row(i) * p.getCoefficients()) < dist_min) { + f_min = i; + dist_min = std::abs(*b_data - A.row(i) * p.getCoefficients()); + } + else if (*b_data != 0 && std::abs(*b_data - A.row(i) * p.getCoefficients()) / *b_data < dist_min) { + f_min = i; + dist_min = std::abs(*b_data - A.row(i) * p.getCoefficients()) / *b_data; + } + b_data++; + } + + return std::make_tuple(t, p, f_min); + }; + +}; + +/// Compute intersection of H-polytope P := Ax <= b +/// with polynomial curve p(t) = sum a_j (t - t0)^j +/// Uses the MPsolve library +/// \tparam Polytope Polytope Type +/// \tparam bfunc feasibility constraint type +template +struct MPSolveHPolyoracle { + + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType Point; + + std::tuple apply( + NT t_prev, + NT t0, + NT eta, + MT &A, + VT &b, + Polytope &P, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + int ignore_facet=-1, + bool positive_real=true) + { + NT maxNT = std::numeric_limits::max(); + NT minNT = std::numeric_limits::lowest(); + + NT tu = eta > 0 ? t0 + eta : NT(maxNT); + NT t = tu; + Point dummy(coeffs[0].dimension()); + + for (unsigned int j = 0; j < coeffs.size(); j++) { + dummy = dummy + pow(tu - t0, NT(j)) * coeffs[j]; + } + + std::tuple result = std::make_tuple(tu, dummy, -1); + + int m = A.rows(); + + // Keeps constants A_i^T C_j + std::vector Z(coeffs.size(), NT(0)); + + // std::vector> solutions; + + // Iterate over all hyperplanes + for (int i = 0; i < m; i++) { + + if (i == ignore_facet) continue; + + for (unsigned int j = 0; j < coeffs.size(); j++) { + Z[j] = A.row(i) * coeffs[j].getCoefficients(); + + #ifdef VOLESTI_DEBUG + std::cout << "Z [ " << j << " ] = " << Z[j] << std::endl; + #endif + } + + // Find point projection on m-th hyperplane + Z[0] -= b(i); + + std::vector> solutions = mpsolve(Z, positive_real); + + for(std::pair sol: solutions) { + + #ifdef VOLESTI_DEBUG + std::cout << "Facet: " << i << " Candidate root is " << sol.first + t0 << std::endl; + #endif + + // Check if solution is in the desired range [t0, t0 + eta] and if it is the current minimum + if (t0 + sol.first <= tu && t0 + sol.first < std::get<0>(result)) { + t = t0 + sol.first; + + // Calculate point from this root + Point p = Point(coeffs[0].dimension()); + + for (unsigned int j = 0; j < coeffs.size(); j++) { + p += pow(t - t0, NT(j)) * coeffs[j] ; + } + + #ifdef VOLESTI_DEBUG + std::cout << "Calculcated point is " << std::endl << p.getCoefficients() << std::endl; + #endif + + // Check if point satisfies Ax <= b up to some tolerance and change current solution + if (P.is_in(p, 1e-6)) { + result = std::make_tuple(t, p, i); + } + } + } + + } + + return result; + }; + +}; + +/// Compute intersection of H-polytope P := Ax <= b +/// with curve p(t) = sum a_j phi_j(t) where phi_j are basis +/// functions (e.g. polynomials) +/// Uses Newton-Raphson to solve the transcendental equation +/// \tparam Polytope Polytope Type +/// \tparam bfunc feasibility constraint type +template +struct NewtonRaphsonHPolyoracle { + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType Point; + + std::tuple apply( + NT t_prev, + NT t0, + NT eta, + MT &A, + VT &b, + Polytope &P, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + int ignore_facet=-1) + { + + // Keep results in a vector (in case of multiple roots) + // The problem has O(m * len(coeffs)) solutions if phi's are polys + // due to the Fundamental Theorem of Algebra + // Some roots may be common for more than one hyperplanes + NT maxNT = std::numeric_limits::max(); + NT minNT = std::numeric_limits::lowest(); + + // Root + NT t = t_prev; + NT tu = eta > 0 ? t0 + eta : NT(maxNT); + + Point dummy(coeffs[0].dimension()); + + for (unsigned int j = 0; j < coeffs.size(); j++) { + dummy += coeffs[j] * phi(tu, t0, j, coeffs.size()); + } + + + std::tuple result = std::make_tuple(tu, dummy, -1); + + // Helper variables for Newton-Raphson + NT dot_u, num, den, den_tmp; + + // Regularization for NR (e.g. at critical points where grad = 0) + NT reg = (NT) 1e-7; + VT Z; + int m = A.rows(); + + // Keeps constants A_i^T C_j + Z.resize(coeffs.size()); + + // Iterate over all hyperplanes + for (int i = 0; i < m; i++) { + + if (i == ignore_facet) continue; + + // Calculate constants + start_iter: t_prev = t0 + reg; + + + for (unsigned int j = 0; j < coeffs.size(); j++) { + Z(j) = A.row(i) * coeffs[j].getCoefficients(); + } + + + for (int tries = 0; tries < MAX_NR_TRIES; tries++) { + + num = - b(i); + den = (NT) 0; + + // Calculate numerator f(t) and denominator f'(t) + for (int j = 0; j < coeffs.size(); j++) { + num += Z(j) * phi(t_prev, t0, j, coeffs.size()); + + // Avoid ill-posed derivative (e.g. 0^{-1}) + if (j > 0) den += Z(j) * grad_phi(t_prev, t0, j, coeffs.size()); + } + + // Regularize denominator if near 0 + if (std::abs(den) < 10 * reg) den += reg; + + // Newton-Raphson Iteration t = t_old - f(t) / f'(t) + t = t_prev - num / den; + + if (t < 0 && t_prev < 0) continue; + + if (std::abs(t - t_prev) < 1e-6 && t > t0) { + // Add root (as t) and facet + + Point p = Point(coeffs[0].dimension()); + + for (unsigned int j = 0; j < coeffs.size(); j++) { + p += coeffs[j] * phi(t, t0, j, coeffs.size()); + } + + if (P.is_in(p) && t < std::get<0>(result)) + result = std::make_tuple(t, p, i); + + } + + t_prev = t; + + } + + } + + return result; + + }; + +}; + + +#endif diff --git a/src/volesti/include/nlp_oracles/nlp_oracles.hpp b/src/volesti/include/nlp_oracles/nlp_oracles.hpp new file mode 100644 index 00000000..11f978e8 --- /dev/null +++ b/src/volesti/include/nlp_oracles/nlp_oracles.hpp @@ -0,0 +1,22 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#include +#include +#include +#include + +#include "nlp_hpolyoracles.hpp" +#include "nlp_vpolyoracles.hpp" + +#ifndef NLP_POLYORACLES_HPP +#define NLP_POLYORACLES_HPP + +#endif diff --git a/src/volesti/include/nlp_oracles/nlp_vpolyoracles.hpp b/src/volesti/include/nlp_oracles/nlp_vpolyoracles.hpp new file mode 100644 index 00000000..9e921d9c --- /dev/null +++ b/src/volesti/include/nlp_oracles/nlp_vpolyoracles.hpp @@ -0,0 +1,314 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +/* This file contains the implementation of boundary oracles between an + d-dimensional V-polytope given by V with a curve p(t) = sum c_j phi_j(t) + where c_j are d-dimensional coefficients and phi_j(t) are basis functions + (e.g. polynomials, rational functions, splines). The boundary oracle + returns one root (in general multiple roots exist) assuming that for t >= t_p + the curve p(t) penetrates the V-polytope. The problem reduces to the following + non-linear optimization problem + + max t + + subject to + t >= 0 + lambda_i >= 0 i \in [V] + \sum_{i = 1}^m lambda_i = 1 + \sum_{i = 1}^m lambda_i v_i - \sum_{i = 1}^M c_j phi_j(t) = 0 + + We use interior-point methods to solve the non-linear optimization program + using COIN-OR ipopt and the ifopt interface for Eigen + ipopt. + +*/ + +#ifndef NLP_VPOLYORACLES_HPP +#define NLP_VPOLYORACLES_HPP + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace ifopt; + +/// Define the variable t we use in the optimization +/// \tparam VT Vector Type +/// \tparam MT Matrix Type +template +class VPolyOracleVariableT : public VariableSet { +public: + NT t, tb, eta; + + VPolyOracleVariableT(NT t_prev, NT tb_, NT eta_=-1): VariableSet(1, "t"), t(t_prev), tb(tb_), eta(eta_) {}; + + void SetVariables(const VT& T) override { + t = T(0); + } + + VectorXd GetValues() const override { + VectorXd T(1); + T(0) = t; + return T; + } + + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + NT tu = eta > 0 ? NT(tb + eta) : NT(inf); + + bounds.at(0) = Bounds(tb, tu); + return bounds; + }; + +}; + + +/// Define the variable t we use in the optimization +/// \tparam VT Vector Type +/// \tparam MT Matrix Type +template +class VPolyOracleVariableLambdas : public VariableSet { +public: + VectorXd lambdas; + int m; + + VPolyOracleVariableLambdas(int m_): m(m_), VariableSet(m_, "lambdas") { + lambdas.resize(m_); + for (int i = 0; i < m_; i++) lambdas(i) = NT(0); + }; + + void SetVariables(const VectorXd& lambdas_) override { + // for (int i = 0; i < m; i++) lambdas(i) = lambdas_(i); + lambdas = lambdas_; + } + + VectorXd GetValues() const override { + return lambdas; + } + + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + for (int i = 0; i < m; i++) bounds.at(i) = Bounds(NT(0), NT(inf)); + return bounds; + }; + +}; + +/// Define the cost function f(t) = t (ipopt takes minimization so it is -t) +/// \tparam VT Vector Type +/// \tparam MT Matrix Type +template +class VPolyOracleCost : public CostTerm { +public: + int m; + + VPolyOracleCost(int m_) : CostTerm("v_poly_cost"), m(m_) {}; + + NT GetCost() const override { + VectorXd T = GetVariables()->GetComponent("t")->GetValues(); + return - T(0); + } + + void FillJacobianBlock (std::string var_set, Jacobian& jac) const override { + if (var_set == "t") jac.coeffRef(0, 0) = NT(-1.0); + if (var_set == "lambdas") { + for (int i = 0; i < m; i++) { + jac.coeffRef(0, i) = NT(0.0); + } + } + } + + +}; + +/// Define the feasibility lambdas +/// \tparam VT Vector Type +/// \tparam MT Matrix Type +template +class VPolyoracleFeasibilityLambdas : public ConstraintSet { +public: + int m; + + VPolyoracleFeasibilityLambdas(int m_) : ConstraintSet(1, "lambdas_simplex"), m(m_) {}; + + + VectorXd GetValues() const override { + VectorXd lambdas = GetVariables()->GetComponent("lambdas")->GetValues(); + VectorXd S(1); + S(0) = NT(0); + for (int i = 0; i < m; i++) S(0) += lambdas(i); + return S; + } + + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + bounds.at(0) = Bounds(NT(1.0), NT(1.0)); + return bounds; + } + + void FillJacobianBlock (std::string var_set, Jacobian& jac) const override { + if (var_set == "lambdas") + for (int i = 0; i < m; i++) { + jac.coeffRef(0, i) = NT(1.0); + } + if (var_set == "t") jac.coeffRef(0, 0) = NT(0); + } + + +}; + +template +class VPolyOracleFeasibilityCurve : public ConstraintSet { +public: + int m; // number of lambdas + int M; // number of coefficients + int d_; // dimension + std::vector &coeffs; + MT &V; + NT t0; + + bfunc phi, grad_phi; + +// V, coeffs, t0, phi, grad_phi + + VPolyOracleFeasibilityCurve(MT &V_, std::vector &coeffs_, NT t0_, bfunc basis, bfunc basis_grad) : + V(V_), coeffs(coeffs_), t0(t0_), phi(basis), grad_phi(basis_grad), ConstraintSet(V_.cols() ,"curve_feasibility") { + m = V.rows(); + M = (int) (coeffs.size()); + d_ = V.cols(); + } + + VecBound GetBounds() const override { + VecBound bounds(GetRows()); + for (int i = 0; i < d_; i++) bounds.at(i) = Bounds(NT(0), NT(0)); + return bounds; + } + + VectorXd GetValues() const override { + VectorXd values_(d_); + NT t = GetVariables()->GetComponent("t")->GetValues()(0); + VectorXd lambdas = GetVariables()->GetComponent("lambdas")->GetValues(); + + for (int i = 0; i < d_; i++) { + values_(i) = NT(0); + + for (int j = 0; j < m; j++) { + values_(i) += lambdas(j) * V(j, i); + } + + for (int j = 0; j < M; j++) { + values_(i) -= phi(t, t0, j, M) * coeffs[j][i]; + } + + } + + return values_; + + } + + void FillJacobianBlock (std::string var_set, Jacobian& jac) const override { + if (var_set == "t") { + NT t = GetVariables()->GetComponent("lambdas")->GetValues()(0); + + for (int i = 0; i < d_; i++) { + for (int j = 0; j < M; j++) { + jac.coeffRef(i, 0) -= grad_phi(t, t0, j, M) * coeffs[j][i]; + } + } + } + + if (var_set == "lambdas") { + for (int i = 0; i < d_; i++) { + for (int j = 0; j < m; j++) { + jac.coeffRef(i, j) = V(j, i); + } + } + } + + } + +}; + +/// Oracle for V-polytopes +/// \tparam Polytope Polytope Type +/// \tparam bfunc feasibility constraint type +template +struct IpoptVPolyoracle { + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Polytope::NT NT; + typedef typename Polytope::PointType Point; + + std::tuple apply( + NT t_prev, + NT t0, + NT eta, + MT &V, + Polytope &P, + std::vector &coeffs, + bfunc phi, + bfunc grad_phi, + int ignore_facet=-1) { // ignore facet not supported + + Problem nlp; + + int m = V.rows(); + + std::shared_ptr> vpolyoraclevariablet (new VPolyOracleVariableT(t_prev, t0, eta)); + std::shared_ptr> vpolyoraclevariable_lambdas (new VPolyOracleVariableLambdas(m)); + + nlp.AddVariableSet(vpolyoraclevariablet); + nlp.AddVariableSet(vpolyoraclevariable_lambdas); + + std::shared_ptr> vpolyoraclecost (new VPolyOracleCost(m)); + + nlp.AddCostSet(vpolyoraclecost); + + std::shared_ptr> vpolyoraclefeasibility_lambdas (new VPolyoracleFeasibilityLambdas(m)); + std::shared_ptr> vpolyoraclefeasibility_curve (new VPolyOracleFeasibilityCurve(V, coeffs, t0, phi, grad_phi)); + + nlp.AddConstraintSet(vpolyoraclefeasibility_lambdas); + nlp.AddConstraintSet(vpolyoraclefeasibility_curve); + + nlp.PrintCurrent(); + + IpoptSolver ipopt; + ipopt.SetOption("linear_solver", "mumps"); + + // TODO fix exact jacobian + ipopt.SetOption("jacobian_approximation", "finite-difference-values"); + ipopt.SetOption("tol", 1e-7); + ipopt.SetOption("print_level", 0); + ipopt.SetOption("sb", "yes"); + + ipopt.Solve(nlp); + + NT t = nlp.GetOptVariables()->GetValues()(0); + + Point p(coeffs[0].dimension()); + + for (unsigned int i = 0; i < coeffs.size(); i++) { + p += phi(t, t0, i, coeffs.size()) * coeffs[i]; + } + + return std::make_tuple(t, p, NT(-1)); + + } + + + +}; + +#endif diff --git a/src/volesti/include/ode_solvers/basis.hpp b/src/volesti/include/ode_solvers/basis.hpp new file mode 100644 index 00000000..f51e92f7 --- /dev/null +++ b/src/volesti/include/ode_solvers/basis.hpp @@ -0,0 +1,186 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_BASIS_HPP +#define ODE_SOLVERS_BASIS_HPP + +enum BasisType { + DERIVATIVE = 0, + FUNCTION = 1, + INTEGRAL = 2 +}; + + +template +struct PolynomialBasis { + BasisType basis_type; + + PolynomialBasis(BasisType basis_type_) : basis_type(basis_type_) {} + + NT operator() (NT t, const NT t0, unsigned int j, const unsigned int ord) const { + switch (basis_type) { + case FUNCTION: + return pow(t - t0, NT(j)); + case DERIVATIVE: + return NT(j) * pow(t - t0, NT(j - 1)); + case INTEGRAL: + return pow(t - t0, NT(j + 1)) / NT(j + 1); + } + } + +}; + +template +struct Polynomial { + + BasisType basis_type; + PolynomialBasis basis; + std::vector coeffs; + NT result; + unsigned int ord; + + Polynomial(std::vector coeffs_, BasisType basis_type_) : + basis_type(basis_type_), coeffs(coeffs_), basis(basis_type_) { + ord = coeffs.size(); + } + + NT operator() (NT t, NT t0, unsigned int j=-1, unsigned int ord=-1) { + result = NT(0); + + for (unsigned int i = 0; i < ord; i++) { + result += coeffs[i] * basis(t, t0, i, ord); + } + + return result; + } + + static std::vector convolve(std::vector const& p, std::vector const& q) { + // Performs direct convolution of p and q (less round-off error than FFT) + unsigned int n = p.size(); + unsigned int m = q.size(); + unsigned int r = n + m - 1; + + std::vector result(r, NT(0)); + + unsigned int j, k; + + for (unsigned int i = 0; i < r; i++) { + j = (i >= m - 1)? i - (m - 1) : 0; + k = (i < n - 1)? i : n - 1; + for (unsigned int z = j; z <= k; z++) result[i] += (p[z] * q[i - z]); + } + + return result; + } + + static std::vector multi_convolve(std::vector> &seq) { + + std::vector result; + result = seq[0]; + + for (unsigned int i = 1; i < seq.size(); i++) { + result = convolve(result, seq[i]); + } + + return result; + + } + + +}; + +template +struct RationalFunctionBasis { + bfunc p, q; + bfunc grad_p, grad_q; + NT reg = 1e-6; + BasisType basis_type; + NT num, den, grad_num, grad_den; + + RationalFunctionBasis(bfunc num, bfunc grad_num, bfunc den, bfunc grad_den, BasisType basis_type_) : + p(num), grad_p(grad_num), q(den), grad_q(grad_den), basis_type(basis_type_) {}; + + NT operator()(NT t, NT t0, unsigned int j, unsigned int ord) { + + switch (basis_type) { + case FUNCTION: + num = p(t, t0, j, ord); + den = q(t, t0, j, ord); + if (std::abs(den) < reg) den += reg; + return num / den; + case DERIVATIVE: + num = p(t, t0, j, ord); + grad_num = grad_p(t, t0, j, ord); + den = q(t, t0, j, ord); + grad_den = grad_q(t, t0, j, ord); + if (std::abs(den * den) < reg) den += reg; + return (grad_num / den) - (grad_den * num) / den; + case INTEGRAL: + throw true; + } + } + +}; + +template +struct LagrangePolynomial { + + VT coeffs; + VT nodes; + int basis = -1; + + LagrangePolynomial() {} + + int order() const { + return nodes.rows(); + } + + void set_nodes(VT nodes_) { + nodes = nodes_; + } + + void set_basis(int basis_) { + basis = basis_; + } + + void set_coeffs(VT coeffs_) { + coeffs = coeffs_; + } + + NT operator() (NT t) const { + + int j = (int) round((order() * acos(t)) / M_PI - 0.5); + NT temp = cos((j+0.5) * M_PI / order()); + + if (abs(temp - nodes(j)) < 1e-6) { + if (basis == -1) return coeffs(j); + else return NT(1); + } else { + throw true; + return NT(-1); + } + + } +}; + +template +void degree_doubling_chebyshev(std::vector &coeffs, + std::vector &result) { + unsigned int N = coeffs.size() - 1; + + for (int i = 0; i <= 2 * N; i++) { + if (i > N) result[i] = coeffs[i - N]; + else if (i == N) result[i] = 2 * coeffs[0]; + else result[i] = coeffs[N - i]; + } + +} + +#endif diff --git a/src/volesti/include/ode_solvers/collocation.hpp b/src/volesti/include/ode_solvers/collocation.hpp new file mode 100644 index 00000000..3b025289 --- /dev/null +++ b/src/volesti/include/ode_solvers/collocation.hpp @@ -0,0 +1,298 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_COLLOCATION_HPP +#define ODE_SOLVERS_COLLOCATION_HPP + +#include "nlp_oracles/nlp_hpolyoracles.hpp" +#include "nlp_oracles/nlp_vpolyoracles.hpp" + +template < + typename Point, + typename NT, + typename Polytope, + typename bfunc, + typename func, + typename NontLinearOracle=MPSolveHPolyoracle< + Polytope, + bfunc + > +> +struct CollocationODESolver { + + + // Vectors of points + typedef std::vector pts; + typedef std::vector ptsv; + + // typedef from existing templates + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef std::vector MTs; + typedef std::vector bounds; + typedef std::vector coeffs; + + unsigned int dim; + + NT eta; + NT t, t_prev, dt, t_temp; + const NT tol = 1e-3; + + // If set to true the solver assumes linearity of the field + // Otherwise it approximates the constant vector with Euler method + const bool exact = false; + + // If set to true it enables precomputation (does not recompute A and b at every step) + const bool precompute = true; + + bool precompute_flag = false; + + // Function oracles x'(t) = F(x, t) + func F; + + // Basis functions + bfunc phi, grad_phi; + + bounds Ks; + + // Contains the sub-states + pts xs, xs_prev, zs; + Point y; + + NT temp_grad; + NT temp_func; + + // Basis coefficients + ptsv as; + + // Temporal coefficients + coeffs cs; + + // Matrices for collocation methods + MTs As, Bs; + + // Keeps the solution to Ax = b temporarily + MTs temps; + + VT Ar, Av; + + NontLinearOracle oracle; + + int prev_facet = -1; + Point prev_point; + + CollocationODESolver(NT initial_time, NT step, pts initial_state, func oracle, + bounds boundaries, coeffs c_coeffs, bfunc basis, bfunc grad_basis) : + t(initial_time), xs(initial_state), F(oracle), eta(step), Ks(boundaries), + cs(c_coeffs), phi(basis), grad_phi(grad_basis) { + dim = xs[0].dimension(); + initialize_matrices(); + }; + + unsigned int order() const { + return cs.size(); + } + + void initialize_matrices() { + As = MTs(xs.size()); + Bs = MTs(xs.size()); + temps = MTs(xs.size()); + as = ptsv(xs.size(), pts(order(), Point(xs[0].dimension()))); + for (unsigned int i = 0; i < xs.size(); i++) { + // Gradient matrix is of size (order - 1) x (order - 1) + As[i].resize(order()-1, order()-1); + // Constants matrix is of size (order - 1) x dim + Bs[i].resize(order()-1, xs[0].dimension()); + temps[i].resize(order()-1, xs[0].dimension()); + } + + zs = pts{Point(1)}; + } + + void step() { + t_prev = t; + xs_prev = xs; + + for (unsigned int ord = 0; ord < order(); ord++) { + // Calculate t_ord + t = t_prev + cs[ord] * eta; + + for (int i = 0; i < xs.size(); i++) { + // a0 = F(x0, t0) + y = F(i,xs, t); + + if (ord == 0) { + as[i][0] = xs_prev[i]; + + if (exact) { + for (unsigned int j = 0; j < order() - 1; j++) { + Bs[i].row(j) = y.getCoefficients(); + } + } + + } + // Construct matrix A + else { + if (exact) { + + if (!precompute || (precompute && !precompute_flag)) { + for (unsigned int j = 0; j < order() - 1; j++) { + temp_grad = grad_phi(t, t_prev, order() - j - 1, order()); + zs[0].set_coord(0, phi(t, t_prev, order() - j - 1, order())); + temp_func = F(i,zs, t)[0]; + As[i](ord-1, j) = temp_grad - temp_func; + } + } + } + else { + + // Compute new derivative (inter-point) + dt = (cs[ord] - cs[ord-1]) * eta; + y = dt * y; + + // Do not take into account reflections + xs[i] += y; + + // Keep grads for matrix B + Bs[i].row(ord-1) = y.getCoefficients().transpose(); + + // Construct matrix A that contains the gradients of the basis functions + if (!precompute || (precompute && !precompute_flag)) { + for (unsigned int j = 0; j < order() - 1; j++) { + As[i](ord-1, j) = grad_phi(t, t_prev, order() - j - 1, order()); + } + } + } + } + + } + } + + // Solve linear systems + for (int i = 0; i < xs.size(); i++) { + // temp contains solution in decreasing order of bases + temps[i] = As[i].colPivHouseholderQr().solve(Bs[i]); + + for (int j = 0; j < order() - 1; j++) { + // TODO Add vectorized implementation + // as[i][order() - j - 1] += temp(j); + for (int k = 0; k < xs[0].dimension(); k++) { + as[i][order() - j - 1].set_coord(k, temps[i](j, k)); + } + } + } + + if (!exact) { + for (int r = 0; r < (int) (eta / tol); r++) { + for (unsigned int i = 0; i < xs.size(); i++) { + xs[i] = Point(xs[i].dimension()); + for (unsigned int ord = 0; ord < order(); ord++) { + xs[i] += as[i][ord] * phi(t_prev + eta, t_prev, ord, order()); + } + } + + for (unsigned int i = 0; i < xs.size(); i++) { + for (int ord = 1; ord < order(); ord++) { + t_temp = cs[ord] * eta; + y = F(i,xs, t_temp); + Bs[i].row(ord-1) = y.getCoefficients().transpose(); + } + } + + + // Solve linear systems + for (int i = 0; i < xs.size(); i++) { + // temp contains solution in decreasing order of bases + temps[i] = As[i].colPivHouseholderQr().solve(Bs[i]); + + for (int j = 0; j < order() - 1; j++) { + // TODO Add vectorized implementation + // as[i][order() - j - 1] += temp(j); + for (int k = 0; k < xs[0].dimension(); k++) { + as[i][order() - j - 1].set_coord(k, temps[i](j, k)); + } + } + } + + } + + + } + + + // Compute next point + for (unsigned int i = 0; i < xs.size(); i++) { + if (Ks[i] == NULL) { + // xs[i] = Point(xs[i].dimension()); + // for (unsigned int ord = 0; ord < order(); ord++) { + // xs[i] += as[i][ord] * phi(t_prev + eta, t_prev, ord, order()); + // } + + if (prev_facet != -1 && i > 0) Ks[i-1]->compute_reflection(xs[i], prev_point, prev_facet); + prev_facet = -1; + + + } else { + std::tuple result = Ks[i]->curve_intersect(t_prev, t_prev, + eta, as[i], phi, grad_phi, oracle); + + // Point is inside polytope + if (std::get<2>(result) == -1 && Ks[i]->is_in(std::get<1>(result))) { + // std::cout << "Inside" << std::endl; + xs[i] = Point(xs[i].dimension()); + for (unsigned int ord = 0; ord < order(); ord++) { + xs[i] += as[i][ord] * phi(t_prev + eta, t_prev, ord, order()); + } + + prev_facet = -1; + + } + else { + // std::cout << "outside" << std::endl; + // Stick to the boundary + xs[i] = 0.99 * std::get<1>(result); + prev_point = xs[i]; + prev_facet = std::get<2>(result); + + + } + } + } + + t += eta; + + precompute_flag = true; + + } + + + void print_state() { + for (int j = 0; j < xs.size(); j++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps) { + for (int i = 0; i < num_steps; i++) step(); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } +}; + +#endif diff --git a/src/volesti/include/ode_solvers/euler.hpp b/src/volesti/include/ode_solvers/euler.hpp new file mode 100644 index 00000000..2e240742 --- /dev/null +++ b/src/volesti/include/ode_solvers/euler.hpp @@ -0,0 +1,130 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_EULER_HPP +#define ODE_SOLVERS_EULER_HPP + +template < + typename Point, + typename NT, + typename Polytope, + typename func +> +struct EulerODESolver { + + typedef std::vector pts; + typedef std::vector bounds; + typedef typename Polytope::VT VT; + + unsigned int dim; + + NT eta; + NT t; + VT Ar, Av; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + pts xs_prev; + + // Previous state boundary point + Point x_prev_bound; + + // Previous state boundary facet + int prev_facet = -1; + + EulerODESolver(NT initial_time, NT step, pts initial_state, func oracle, + bounds boundaries) : + eta(step), t(initial_time), F(oracle), Ks(boundaries), xs(initial_state) { + dim = xs[0].dimension(); + }; + + + void step(int k, bool accepted) { + xs_prev = xs; + t += eta; + for (unsigned int i = 0; i < xs.size(); i++) { + Point y = F(i, xs_prev, t); + y = eta * y; + + if (Ks[i] == NULL) { + xs[i] = xs_prev[i] + y; + if (prev_facet != -1 && i > 0) { + Ks[i-1]->compute_reflection(xs[i], x_prev_bound, prev_facet); + } + + } + else { + + // Find intersection (assuming a line trajectory) between x and y + do { + // Find line intersection between xs[i] (new position) and y + std::pair pbpair = Ks[i]->line_positive_intersect(xs_prev[i], y, Ar, Av); + + if (pbpair.first >= 0 && pbpair.first <= 1) { + // Advance to point on the boundary + xs_prev[i] += (pbpair.first * 0.95) * y; + + // Update facet for reflection of derivative + prev_facet = pbpair.second; + x_prev_bound = xs_prev[i]; + + // Reflect ray y on the boundary point y now is the reflected ray + Ks[i]->compute_reflection(y, xs_prev[i], pbpair.second); + // Add it to the existing (boundary) point and repeat + xs[i] = xs_prev[i] + y; + + } + else { + prev_facet = -1; + xs[i] = xs_prev[i] + y; + } + } while (!Ks[i]->is_in(xs[i])); + + } + + } + + } + + void print_state() { + for (int j = 0; j < xs.size(); j++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) step(i, accepted); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + + void disable_adaptive() { + // TODO Implement + } + + void enable_adaptive() { + // TODO Implement + } + +}; + +#endif diff --git a/src/volesti/include/ode_solvers/generalized_leapfrog.hpp b/src/volesti/include/ode_solvers/generalized_leapfrog.hpp new file mode 100644 index 00000000..098ea035 --- /dev/null +++ b/src/volesti/include/ode_solvers/generalized_leapfrog.hpp @@ -0,0 +1,172 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_GENERALIZED_LEAPFROG_HPP +#define ODE_SOLVERS_GENERALIZED_LEAPFROG_HPP + +template < +typename Point, +typename NT, +typename ConvexBody, +typename func +> +struct GeneralizedLeapfrogODESolver { + + typedef std::vector pts; + + typedef Eigen::Matrix MT; + + typedef std::vector bounds; + typedef typename ConvexBody::VT VT; + + unsigned int dim; + std::vector lambda_prev; + + + NT eta; + NT eta0; + NT t; + NT dl = 0.995; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + pts xs_prev; + + MT _AA; + + std::pair pbpair; + + unsigned long long num_reflections = 0; + unsigned long long num_steps = 0; + + bool adaptive = true; + + GeneralizedLeapfrogODESolver(NT initial_time, NT step, pts initial_state, func oracle, bounds boundaries, bool adaptive_=true) : + eta(step), eta0(step), t(initial_time), F(oracle), Ks(boundaries), xs(initial_state), adaptive(adaptive_) { + dim = xs[0].dimension(); + initialize(); + }; + + void initialize() { + for (unsigned int i = 0; i < xs.size(); i++) { + lambda_prev.push_back(NT(0)); + } + } + + void disable_adaptive() { + adaptive = false; + } + + void enable_adaptive() { + adaptive = true; + } + + void step(int k, bool accepted) { + num_steps++; + + if (adaptive) eta = (eta0 * num_steps) / (num_steps + num_reflections); + + xs_prev = xs; + unsigned int x_index, v_index, it; + t += eta; + for (unsigned int i = 1; i < xs.size(); i += 2) { + //pbpair.second = -1; + x_index = i - 1; + v_index = i; + + // v' <- v + eta / 2 F(x) + Point z = F(v_index, xs_prev, t); + z = (eta / 2) * z; + xs[v_index] = xs[v_index] + z; + + // x <- x + eta v' + Point y = xs[v_index]; + + if (Ks[x_index] == NULL) { + xs[x_index] = xs_prev[x_index] + eta*y; + } + else { + // Find intersection (assuming a line trajectory) between x and y + bool step_completed = false; + NT T = eta; + if (k == 0 && !accepted) { + lambda_prev[x_index] = 0.0; + } + + pbpair = Ks[x_index]->line_positive_intersect(xs_prev[x_index], y); + if (T <= pbpair.first) { + xs[x_index] = xs_prev[x_index] + T * y; + xs[v_index] = y; + lambda_prev[x_index] = T; + step_completed = true; + } + + if (!step_completed) { + lambda_prev[x_index] = dl * pbpair.first; + xs_prev[x_index] = xs_prev[x_index] + lambda_prev[x_index] * y; + T -= lambda_prev[x_index]; + Ks[x_index]->compute_reflection(y, xs_prev[x_index], pbpair.second); + num_reflections++; + + while (true) + { + pbpair = Ks[x_index]->line_positive_intersect(xs_prev[x_index], y); + if (T <= pbpair.first) { + xs[x_index] = xs_prev[x_index] + T * y; + xs[v_index] = y; + lambda_prev[x_index] = T; + break; + } + lambda_prev[x_index] = dl * pbpair.first; + xs_prev[x_index] = xs_prev[x_index] + lambda_prev[x_index] * y; + T -= lambda_prev[x_index]; + Ks[x_index]->compute_reflection(y, xs_prev[x_index], pbpair.second); + num_reflections++; + } + } + } + + // tilde v <- v + eta / 2 F(tilde x) + z = F(v_index, xs, t); + z = (eta / 2) * z; + xs[v_index] = xs[v_index] + z; + + } + + } + + void print_state() { + for (int j = 0; j < xs.size(); j ++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) step(i, accepted); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + +}; + + +#endif diff --git a/src/volesti/include/ode_solvers/implicit_midpoint.hpp b/src/volesti/include/ode_solvers/implicit_midpoint.hpp new file mode 100644 index 00000000..786734d8 --- /dev/null +++ b/src/volesti/include/ode_solvers/implicit_midpoint.hpp @@ -0,0 +1,177 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" + +#ifndef IMPLICIT_MIDPOINT_HPP +#define IMPLICIT_MIDPOINT_HPP +#include "preprocess/crhmc/opts.h" +#include "random_walks/crhmc/hamiltonian.hpp" +#ifdef TIME_KEEPING +#include +#endif + +template +inline std::vector operator+(const std::vector &v1, + const std::vector &v2) +{ + std::vector result(v1.size()); + for (int i = 0; i < v1.size(); i++) { + result[i] = v1[i] + v2[i]; + } + return result; +} +template +inline std::vector operator*(const std::vector &v, const Type alpha) +{ + std::vector result(v.size()); + for (int i = 0; i < v.size(); i++) { + result[i] = v[i] * alpha; + } + return result; +} +template +inline std::vector operator/(const std::vector &v, const Type alpha) +{ + return v * (1 / alpha); +} +template +inline std::vector operator-(const std::vector &v1, + const std::vector &v2) +{ + + return v1 + v2 * (-1.0); +} +template +struct ImplicitMidpointODESolver { + using VT = typename Polytope::VT; + using MT = typename Polytope::MT; + using pts = std::vector; + using hamiltonian = Hamiltonian; + using Opts = opts; + + unsigned int dim; + NT eta; + int num_steps = 0; + NT t; + + // Contains the sub-states + pts xs; + pts xs_prev; + + // Function oracle + func F; + Polytope &P; + Opts &options; + MT nu; + int num_runs = 0; + hamiltonian ham; + bool done; +#ifdef TIME_KEEPING + std::chrono::time_point start, end; + std::chrono::duration DU_duration = std::chrono::duration::zero(); + std::chrono::duration approxDK_duration = std::chrono::duration::zero(); +#endif + ImplicitMidpointODESolver(NT initial_time, + NT step, + pts initial_state, + func oracle, + Polytope &boundaries, + Opts &user_options) : + eta(step), + t(initial_time), + xs(initial_state), + F(oracle), + P(boundaries), + options(user_options), + ham(hamiltonian(boundaries)) + { + dim = xs[0].rows(); + }; + + void step(int k, bool accepted) { + num_runs++; + pts partialDerivatives; +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + partialDerivatives = ham.DU(xs); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + DU_duration += end - start; +#endif + xs = xs + partialDerivatives * (eta / 2); + xs_prev = xs; + done = false; + nu = MT::Zero(P.equations(), simdLen); + for (int i = 0; i < options.maxODEStep; i++) { + pts xs_old = xs; + pts xmid = (xs_prev + xs) / 2.0; +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + partialDerivatives = ham.approxDK(xmid, nu); + +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + approxDK_duration += end - start; +#endif + xs = xs_prev + partialDerivatives * (eta); + VT dist = ham.x_norm(xmid, xs - xs_old) / eta; + NT maxdist = dist.maxCoeff(); + //If the estimate does not change terminate + if (maxdist < options.implicitTol) { + done = true; + num_steps = i; + break; + //If the estimate is very bad, sample another velocity + } else if (maxdist > options.convergence_bound) { + xs = xs * std::nan("1"); + done = true; + num_steps = i; + break; + } + } +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + partialDerivatives = ham.DU(xs); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + DU_duration += end - start; +#endif + xs = xs + partialDerivatives * (eta / 2); + ham.project(xs); + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) { + step(i, accepted); + } + } + + MT get_state(int index) { return xs[index]; } + + void set_state(int index, MT p) { xs[index] = p; } + template + void print_state(StreamType &stream) { + for (int j = 0; j < xs.size(); j++) { + stream << "state " << j << ": \n"; + stream << xs[j]; + stream << '\n'; + } + } +}; + +#endif diff --git a/src/volesti/include/ode_solvers/integral_collocation.hpp b/src/volesti/include/ode_solvers/integral_collocation.hpp new file mode 100644 index 00000000..10db9be9 --- /dev/null +++ b/src/volesti/include/ode_solvers/integral_collocation.hpp @@ -0,0 +1,289 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// Refers to the integral collocation method with Lagrange Polynomials +// from Lee, Yin Tat, Zhao Song, and Santosh S. Vempala. +//"Algorithmic theory of ODEs and sampling from well-conditioned +// logconcave densities." arXiv preprint arXiv:1812.06243 (2018). + + +#ifndef ODE_SOLVERS_INTEGRAL_COLLOCATION_HPP +#define ODE_SOLVERS_INTEGRAL_COLLOCATION_HPP + +#include "nlp_oracles/nlp_hpolyoracles.hpp" +#include "nlp_oracles/nlp_vpolyoracles.hpp" +#include "boost/numeric/ublas/vector.hpp" +#include "boost/numeric/ublas/io.hpp" +#include "boost/math/special_functions/chebyshev.hpp" +#include "boost/math/special_functions/chebyshev_transform.hpp" + +template < + typename Point, + typename NT, + typename Polytope, + typename func +> +struct IntegralCollocationODESolver { + + + // Vectors of points + typedef std::vector pts; + typedef std::vector ptsv; + + // typedef from existing templates + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef std::vector MTs; + + typedef std::vector bounds; + typedef std::vector coeffs; + typedef boost::numeric::ublas::vector boost_vector; + typedef boost::math::chebyshev_transform chebyshev_transform_boost; + + unsigned int dim; + + NT eta; + NT t, t_prev, dt, temp_node, a, b; + const NT tol = 1e-6; + + // Function oracles x'(t) = F(x, t) + func F; + bounds Ks; + + // Contains the sub-states + pts xs, xs_prev, X_temp; + Point y; + + // Temporal coefficients + coeffs cs; + + VT Ar, Av, X_op, nodes; + + MT A_phi, X0, X, X_prev, F_op; + + unsigned int _order; + + LagrangePolynomial lagrange_poly; + + int prev_facet = -1; + Point prev_point; + + IntegralCollocationODESolver(NT initial_time, NT step, pts initial_state, + func oracle, bounds boundaries, unsigned int order_=4) : + t(initial_time), xs(initial_state), X_temp(initial_state), F(oracle), eta(step), Ks(boundaries), + _order(order_) { + dim = xs[0].dimension(); + initialize_matrices(); + }; + + unsigned int order() const { + return _order; + } + + void initialize_matrices() { + + A_phi.resize(order(), order()); + nodes.resize(order()); + + std::vector temp; + + for (unsigned int j = 0; j < order(); j++) { + nodes(j) = cos((j+0.5) * M_PI / order()); + } + + lagrange_poly.set_nodes(nodes); + + // Calculate integrals of basis functions based on the Discrete Chebyshev Transform + for (unsigned int i = 0; i < order(); i++) { + + lagrange_poly.set_basis((int) i); + + for (unsigned int j = 0; j <= i; j++) { + if (nodes(j) < NT(0)) { + a = nodes(j); + b = NT(0); + } else { + a = NT(0); + b = nodes(j); + } + + chebyshev_transform_boost transform(lagrange_poly, a, b); + A_phi(i, j) = NT(transform.integrate()); + A_phi(j, i) = A_phi(i, j); + } + } + + #ifdef VOLESTI_DEBUG + std::cout << "A_phi" << std::endl; + std::cout << A_phi << std::endl; + #endif + + X.resize(xs.size() * dim, order()); + X0.resize(xs.size() * dim, order()); + X_prev.resize(xs.size() * dim, order()); + X_op.resize(xs.size() * dim); + + F_op.resize(xs.size() * dim, order()); + + lagrange_poly.set_basis(-1); + } + + void initialize_fixed_point() { + for (unsigned int ord = 0; ord < order(); ord++) { + for (unsigned int i = 0; i < xs.size(); i++) { + for (unsigned int j = i * dim; j < (i + 1) * dim; j++) { + X0(j, ord) = xs_prev[i][j % dim]; + } + } + } + } + + void step() { + xs_prev = xs; + initialize_fixed_point(); + + std::vector transforms; + + X = X0; + X_prev = 100 * X0; + NT err; + + do { + for (unsigned int ord = 0; ord < order(); ord++) { + for (unsigned int i = 0; i < xs.size(); i++) { + for (unsigned int j = i * dim; j < (i + 1) * dim; j++) { + X_temp[i].set_coord(j % dim, X(j, ord)); + } + } + + for (unsigned int i = 0; i < xs.size(); i++) { + // std::cout << "pre y" << std::endl; + temp_node = nodes(ord) * eta; + y = F(i, X_temp, temp_node); + + for (int j = i * dim; j < (i + 1) * dim; j++) { + F_op(j, ord) = y[j % dim]; + } + + } + } + + X = X0 + F_op * A_phi; + + X_prev = X; + + err = sqrt((X - X_prev).squaredNorm()); + + } while (err > 1e-10); + + X_op = X0.col(0); + + unsigned int max_transform_coeffs_length = 0; + + + for (unsigned int i = 0; i < xs.size(); i++) { + for (unsigned int j = i * dim; j < (i + 1) * dim; j++) { + lagrange_poly.set_coeffs(F_op.row(j).transpose()); + chebyshev_transform_boost transform(lagrange_poly, 0, eta, 1e-5, 5); + transforms.push_back(transform); + // Keep max transform length for zero-padding + if (max_transform_coeffs_length < transform.coefficients().size()) { + max_transform_coeffs_length = transform.coefficients().size(); + } + + X_op(j) += NT(transform.integrate()); + } + } + + + for (unsigned int i = 0; i < xs.size(); i++) { + if (Ks[i] == NULL) { + for (unsigned int j = i * dim; j < (i + 1) * dim; j++) { + xs[i].set_coord(j % dim, X_op(j)); + } + } + else { + throw true; + // TODO Implement oracle. It requires doing chebyshev transforms + // with the same #of points at each dimension + // 1. Temporarily store coefficients as points + // 2. Initialize everything to Points at origin (for zero-padding) + // pts temp_pts(max_transform_coeffs_length, Point(dim)); + + // 3. Store transformation to polynomial of twice the degree + // pts transformed_pts(2 * max_transform_coeffs_length, Point(dim)); + // std::vector temp_coeffs; + + // 4. Invoke chebyshev transform coefficients + // Precomputed from the final step of the fixed point iterator + // for (unsigned int j = i * dim; j < (i + 1) * dim; j++) { + // temp_coeffs = transforms[j].coefficients(); + // for (unsigned int k = 0; k < temp_coeffs.size(); k++) { + // temp_pts[k].set_coord(j % dim, temp_coeffs[k]); + // } + // } + + // 5. Apply degree-doubling transformation + // The transformation takes the n Chebyshev transform coefficients c[i] + // and creates a complex polynomial of order 2n with coefficients a[i] + // such that a[n] = 2 * c[0], a[i] = c[i - n] for i > n and a[i] = c[n - i] for i < n + // This polynomial h(z) is defined on the complex plane. + // Its roots are related to the chebyshev transform as: z is a root of + // the degree-doubled polynomial h(z) then Re(z) is a + // root for the chebyshev transfrom. This transformation is domain_name + // in order to be able to use MPSolve to compute the boundary oracle + // degree_doubling_chebyshev(temp_pts, transformed_pts); + + // 6. Integrate coefficients + // We want the integral of the function instead of the function itself. + // Integrating the polynomial is fairly easy. + // transformed_pts.insert(transformed_pts.begin(), Point(dim)); + + // for (unsigned int k = 1; k < transformed_pts.size(); k++) { + // transformed_pts[k] = (1.0 / k) * transformed_pts[k]; + // } + + // 7. Find roots using MPSolve + // Project the computed coefficients to each of the polytopes normals and + // calculate the complex roots of the resulting equations. + // Then keep the smallest positive rational part. + } + + } + + + + } + + + void print_state() { + for (int j = 0; j < xs.size(); j++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps) { + for (int i = 0; i < num_steps; i++) step(); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } +}; + + +#endif diff --git a/src/volesti/include/ode_solvers/leapfrog.hpp b/src/volesti/include/ode_solvers/leapfrog.hpp new file mode 100644 index 00000000..b7f2724b --- /dev/null +++ b/src/volesti/include/ode_solvers/leapfrog.hpp @@ -0,0 +1,221 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_LEAPFROG_HPP +#define ODE_SOLVERS_LEAPFROG_HPP + +template < +typename Point, +typename NT, +typename Polytope, +typename func +> +struct LeapfrogODESolver { + + struct update_parameters +{ + update_parameters() + : facet_prev(0), hit_ball(false), inner_vi_ak(0.0), ball_inner_norm(0.0) + {} + int facet_prev; + bool hit_ball; + double inner_vi_ak; + double ball_inner_norm; +}; + + update_parameters _update_parameters; + + typedef std::vector pts; + + typedef Eigen::Matrix MT; + + typedef std::vector bounds; + typedef typename Polytope::VT VT; + + unsigned int dim; + + std::vector Ar, Av; + std::vector lambda_prev; + + NT eta; + NT eta0; + NT t; + NT dl = 0.995; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + pts xs_prev; + + Point grad_x; + + MT _AA; + + std::pair pbpair; + + unsigned long long num_reflections = 0; + unsigned long long num_steps = 0; + + bool adaptive = true; + + LeapfrogODESolver(NT initial_time, NT step, pts initial_state, func oracle, bounds boundaries, bool adaptive_=true) : + eta(step), eta0(step), t(initial_time), F(oracle), Ks(boundaries), xs(initial_state), adaptive(adaptive_) { + dim = xs[0].dimension(); + _update_parameters = update_parameters(); + grad_x.set_dimension(dim); + initialize(); + }; + + + void initialize() { + for (unsigned int i = 0; i < xs.size(); i++) { + VT ar, av; + if (Ks[i] != NULL) { + ar.setZero(Ks[i]->num_of_hyperplanes()); + ar = Ks[i]->get_mat() * xs[i].getCoefficients(); + av.setZero(Ks[i]->num_of_hyperplanes()); + _AA.noalias() = Ks[i]->get_mat() * Ks[i]->get_mat().transpose(); + Ks[i]->resetFlags(); + } + Ar.push_back(ar); + Av.push_back(av); + lambda_prev.push_back(NT(0)); + } + } + + void disable_adaptive() { + adaptive = false; + } + + void enable_adaptive() { + adaptive = true; + } + + void step(int k, bool accepted) { + num_steps++; + if (adaptive) eta = (eta0 * num_steps) / (num_steps + num_reflections); + + xs_prev = xs; + unsigned int x_index, v_index, it; + t += eta; + Point y; + for (unsigned int i = 1; i < xs.size(); i += 2) { + + x_index = i - 1; + v_index = i; + + // v' <- v + eta / 2 F(x) + if (k == 0 && !accepted) { + grad_x = F(v_index, xs_prev, t); + } + xs[v_index] += (eta / 2) * grad_x; + + // x <- x + eta v' + y = xs[v_index]; + + if (Ks[x_index] == NULL) { + xs[x_index] += eta*y; + } + else { + // Find intersection (assuming a line trajectory) between x and y + bool step_completed = false; + NT T = eta; + if (k == 0 && !accepted) { + Ar[x_index] = Ks[x_index]->get_mat() * xs_prev[x_index].getCoefficients(); + lambda_prev[x_index] = 0.0; + Ks[x_index]->resetFlags(); + } + + pbpair = Ks[x_index]->line_positive_intersect(xs_prev[x_index], y, Ar[x_index], Av[x_index], + lambda_prev[x_index], _update_parameters); + if (T <= pbpair.first) { + xs[x_index] = xs_prev[x_index] + T * y; + xs[v_index] = y; + lambda_prev[x_index] = T; + Ks[x_index]->update_position_internal(T); + step_completed = true; + } + + if (!step_completed) { + lambda_prev[x_index] = dl * pbpair.first; + xs_prev[x_index] = xs_prev[x_index] + lambda_prev[x_index] * y; + T -= lambda_prev[x_index]; + Ks[x_index]->update_position_internal(lambda_prev[x_index]); + Ks[x_index]->compute_reflection(y, xs_prev[x_index], _update_parameters); + num_reflections++; + + while (true) + { + pbpair = Ks[x_index]->line_positive_intersect(xs_prev[x_index], y, Ar[x_index], Av[x_index], + lambda_prev[x_index], _AA, _update_parameters); + if (T <= pbpair.first) { + xs[x_index] = xs_prev[x_index] + T * y; + xs[v_index] = y; + lambda_prev[x_index] = T; + Ks[x_index]->update_position_internal(T); + break; + } + lambda_prev[x_index] = dl * pbpair.first; + xs_prev[x_index] = xs_prev[x_index] + lambda_prev[x_index] * y; + T -= lambda_prev[x_index]; + Ks[x_index]->update_position_internal(lambda_prev[x_index]); + Ks[x_index]->compute_reflection(y, xs_prev[x_index], _update_parameters); + num_reflections++; + } + } + } + + // tilde v <- v + eta / 2 F(tilde x) + grad_x = F(v_index, xs, t); + xs[v_index] += (eta / 2) * grad_x; + } + + } + + void print_state() { + for (int j = 0; j < xs.size(); j ++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) step(i, accepted); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + + NT get_eta() { + return eta; + } + + void set_eta(NT &eta_) { + eta = eta_; + eta0 = eta_; + } + + bounds get_bounds() { + return Ks; + } + +}; + + +#endif diff --git a/src/volesti/include/ode_solvers/ode_solvers.hpp b/src/volesti/include/ode_solvers/ode_solvers.hpp new file mode 100644 index 00000000..68e730fc --- /dev/null +++ b/src/volesti/include/ode_solvers/ode_solvers.hpp @@ -0,0 +1,64 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +/* + Each solver solves an ODE of the form d^k x/dt^k = F(x, t) + The vector x can be written as x = (x_1, ...., x_n) where each + x_i is conditioned on a convex polytope K_i with bounary reflections + on the boundary of K_i for each 1 <= i <= n. Each sub-state is determined + by the oracle function F_i(x, t) which has range over x = (x_1, ..., x_n). + + Some general parameter notations for the solvers + + Templates + 1. Polytope: The polytope type (H-Polytope, V-Polytope, Z-Polytope) K_i + 2. NT: Number type (double, float) + 3. Point: Point type + 4. func: Function type for the oracle collection K_i + 5. bfunc: Basis function type (e.g. polynomial) for non-linear trajectory + methods (such as collocation) + + Variables + 1. Ks: A vector of domains K_i for 1 <= i <= n + 2. xs: A vector of substates x_i for 1 <= i <= n + 3. xs_prev: The previous state of the solver + 4. F: A functor of oracles F_i for 1 <= i <= n + 5. eta: Step size + 6. t: Temporal variable + + TODO: + 1. Change datastructure of boundaries (std::vector) + +*/ + +#include +#include +#include +#include + +#include "ode_solvers/euler.hpp" +#include "ode_solvers/implicit_midpoint.hpp" +#include "ode_solvers/runge_kutta.hpp" +#include "ode_solvers/leapfrog.hpp" +#include "ode_solvers/richardson_extrapolation.hpp" +#include "ode_solvers/oracle_functors.hpp" +#include "ode_solvers/randomized_midpoint.hpp" +#include "ode_solvers/generalized_leapfrog.hpp" + +#ifndef DISABLE_NLP_ORACLES +#include "ode_solvers/collocation.hpp" +#include "ode_solvers/basis.hpp" +#include "ode_solvers/integral_collocation.hpp" +#endif + +#ifndef ODE_SOLVERS_ODE_SOLVERS_HPP +#define ODE_SOLVERS_ODE_SOLVERS_HPP + +#endif diff --git a/src/volesti/include/ode_solvers/oracle_functors.hpp b/src/volesti/include/ode_solvers/oracle_functors.hpp new file mode 100644 index 00000000..35fc8887 --- /dev/null +++ b/src/volesti/include/ode_solvers/oracle_functors.hpp @@ -0,0 +1,397 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_ORACLE_FUNCTORS_HPP +#define ODE_SOLVERS_ORACLE_FUNCTORS_HPP + +struct OptimizationFunctor { + template < + typename NT, + typename Functor, + typename GradFunctor + > + struct parameters { + NT T; // Temperature + unsigned int dim; // Dimension + Functor f; + GradFunctor neg_grad_f; + NT L; + NT m; + NT kappa; + unsigned int order; + + parameters( + NT T_, + unsigned int dim_, + Functor f_, + GradFunctor neg_grad_f_) : + T(T_), + dim(dim_), + f(f_), + neg_grad_f(neg_grad_f_), + L(1), + m(1), + kappa(1), + order(2) + {}; + + void update_temperature(NT k, NT l) { + T = T * pow(1.0 + 1.0 / pow(dim, k), l); + } + }; + + template + < + typename Point, + typename Functor, + typename GradFunctor + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef std::vector pts; + + parameters ¶ms; + + GradientFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + return params.neg_grad_f(i, xs, t) * params.T; // returns - a*x + } else { + return xs[i + 1]; // returns derivative + } + } + }; + + template + < + typename Point, + typename Functor, + typename GradFunctor + > + struct FunctionFunctor { + typedef typename Point::FT NT; + parameters ¶ms; + + FunctionFunctor(parameters ¶ms_) : params(params_) {}; + + NT operator() (Point const& x) const { + return params.f(x) * params.T; + } + }; +}; + +struct IsotropicQuadraticFunctor { + + // Holds function oracle and gradient oracle for the function 1/2 a ||x||^2 + template < + typename NT + > + struct parameters { + NT alpha; + unsigned int order; + NT L; // Lipschitz constant of gradient + NT m; // Strong-convexity parameter + NT kappa; // Condition number + + parameters() : + alpha(NT(1)), + order(2), + L(NT(1)), + m(NT(1)), + kappa(1) + {}; + + parameters( + NT alpha_, + unsigned int order_) : + alpha(alpha_), + order(order_), + L(alpha_), + m(alpha_), + kappa(1) + {}; + }; + + + template + < + typename Point + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef std::vector pts; + + parameters ¶ms; + + GradientFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + return (-params.alpha) * xs[0]; // returns - a*x + } else { + return xs[i + 1]; // returns derivative + } + } + + Point operator()(Point const &x){ + Point y = (-params.alpha) * x; + return y; + } + }; + + + template + < + typename Point + > + struct FunctionFunctor { + typedef typename Point::FT NT; + + parameters ¶ms; + + FunctionFunctor(parameters ¶ms_) : params(params_) {}; + + NT operator() (Point const& x) const { + return 0.5 * params.alpha * x.dot(x); + } + }; + +}; + +struct IsotropicLinearFunctor { + + // Exponential Density + template < + typename NT + > + struct parameters { + NT alpha; + unsigned int order; + NT L; // Lipschitz constant of gradient + NT m; // Strong-convexity constant + NT kappa; // Condition number + + parameters() : + alpha(NT(1)), + order(1), + L(0), + m(0), + kappa(1) + {}; + + parameters(NT alpha_, unsigned int order_) : + alpha(alpha_), + order(order), + L(0), + m(0), + kappa(1) + {} + }; + + template + < + typename Point + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef std::vector pts; + + parameters ¶ms; + + GradientFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + Point y = Point::all_ones(xs[0].dimension()); + y = (- params.alpha) * y; + return y; + } else { + return xs[i + 1]; // returns derivative + } + } + + }; + + template + < + typename Point + > + struct FunctionFunctor { + typedef typename Point::FT NT; + + parameters ¶ms; + + FunctionFunctor(parameters ¶ms_) : params(params_) {}; + + NT operator() (Point const& x) const { + return params.alpha * x.sum(); + } + + }; + +}; + + +struct ExponentialFunctor { + + // Sample from linear program c^T x (exponential density) + template < + typename NT, + typename Point + > + struct parameters { + unsigned int order; + NT L; // Lipschitz constant for gradient + NT m; // Strong convexity constant + NT kappa; // Condition number + Point c; // Coefficients of LP objective + NT a; // Inverse variance + + parameters(Point c_) : order(2), L(1), m(1), kappa(1), c(c_), a(1.0) {}; + parameters(Point c_, NT a_) : order(2), L(1), m(1), kappa(1), c(c_), a(a_) {}; + + }; + + template + < + typename Point + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef std::vector pts; + + parameters ¶ms; + + GradientFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + Point y(params.c); + return (-params.a) * y; + } else { + return xs[i + 1]; // returns derivative + } + } + + }; + + template + < + typename Point + > + struct FunctionFunctor { + typedef typename Point::FT NT; + + parameters ¶ms; + + FunctionFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + NT operator() (Point const& x) const { + return params.a * x.dot(params.c); + } + + }; + +}; + + +struct GaussianFunctor { + + template < + typename NT, + typename Point + > + struct parameters { + Point x0; + NT a; + NT eta; + unsigned int order; + NT L; // Lipschitz constant for gradient + NT m; // Strong convexity constant + NT kappa; // Condition number + + parameters(Point x0_, NT a_, NT eta_) : + x0(x0_), a(a_), eta(eta_), order(2), L(2 * a_), m(2 * a_), kappa(1) {}; + + }; + + template + < + typename Point + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef std::vector pts; + + parameters ¶ms; + + GradientFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + Point y = (-2.0 * params.a) * (xs[0] - params.x0); + return y; + } else { + return xs[i + 1]; // returns derivative + } + } + Point operator()(Point const&x){ + Point y = (-2.0 * params.a) * (x - params.x0); + return y; + } + }; + + template + < + typename Point + > + struct FunctionFunctor { + typedef typename Point::FT NT; + + parameters ¶ms; + + FunctionFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + NT operator() (Point const& x) const { + Point y = x - params.x0; + return params.a * y.dot(y); + } + + }; + + template +< + typename Point +> +struct HessianFunctor { + typedef typename Point::FT NT; + + parameters ¶ms; + + HessianFunctor(parameters ¶ms_) : params(params_) {}; + + // The index i represents the state vector index + Point operator() (Point const& x) const { + return (2.0 * params.a) * Point::all_ones(x.dimension()); + } + +}; + +}; + +#endif diff --git a/src/volesti/include/ode_solvers/oracle_functors_rcpp.hpp b/src/volesti/include/ode_solvers/oracle_functors_rcpp.hpp new file mode 100644 index 00000000..853dcb3c --- /dev/null +++ b/src/volesti/include/ode_solvers/oracle_functors_rcpp.hpp @@ -0,0 +1,159 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_ORACLE_FUNCTORS_RCPP_HPP +#define ODE_SOLVERS_ORACLE_FUNCTORS_RCPP_HPP + +enum ode_solvers { + no_solver, + leapfrog, + euler, + runge_kutta, + richardson, + collocation +}; + +// Holds Oracle Functor that wraps an R function via RCpp +// The R function is provided as an Rcpp::Function object +// The functor uses Rcpp::as and Rcpp::wrap to do the conversion, +// call the oracle, and convert the results back to C++ +struct RcppFunctor { + + template < + typename NT + > + struct parameters { + NT L; // Lipschitz constant of gradient + NT m; // Strong-convexity parameter + NT eta; // Step-size (if defined by user) + NT kappa; // Condition number + unsigned int order; // Order of ODE functor + + parameters( + NT L_, + NT m_, + NT eta_, + unsigned int order_=2 + ) : + L(L_), + m(m_), + eta(eta_), + kappa(L_ / m_), + order(order_) + {} + }; + + // Log-probability gradient functor + template + < + typename Point + > + struct GradientFunctor { + typedef typename Point::FT NT; + typedef typename Point::Coeff VT; + typedef std::vector pts; + + parameters params; + Rcpp::Function neg_grad_f; // Negative gradient as an Rcpp::Function + bool negate; + + GradientFunctor( + parameters params_, + Rcpp::Function neg_grad_f_, + bool negate_=true): + params(params_), + neg_grad_f(neg_grad_f_), + negate(negate_) + {}; + + // The index i represents the state vector index + Point operator() (unsigned int const& i, pts const& xs, NT const& t) const { + if (i == params.order - 1) { + // Convert point to Rcpp::NumericMatrix + + VT y = Rcpp::as(neg_grad_f(Rcpp::wrap(xs[0].getCoefficients()))); + + Point z(y); + + if (negate) z = (-1.0) * z; + + // Return result as Point + return z; + } else { + return xs[i + 1]; // returns derivative + } + } + + Point operator() (Point const& x) const { + VT y = Rcpp::as(neg_grad_f(Rcpp::wrap(x.getCoefficients()))); + + Point z(y); + + if (negate) z = (-1.0) * z; + + // Return result as Point + return z; + } + + }; + + // Negative log-probability functor + template + < + typename Point + > + struct FunctionFunctor { + typedef typename Point::FT NT; + typedef typename Point::Coeff VT; + + parameters params; + Rcpp::Function negative_logprob; + + FunctionFunctor( + parameters params_, + Rcpp::Function negative_logprob_) : + params(params_), + negative_logprob(negative_logprob_) + {}; + + NT operator() (Point const& x) const { + return Rcpp::as(negative_logprob(Rcpp::wrap(x.getCoefficients()))); + } + + }; + + // Log-probability hessian functor + template + < + typename Point + > + struct HessianFunctor { + typedef typename Point::FT NT; + typedef typename Point::Coeff VT; + + parameters params; + Rcpp::Function hessian; // Negative hessian as an Rcpp::Function + + HessianFunctor( + parameters params_, + Rcpp::Function hessian_) : + params(params_), + hessian(hessian_) + {}; + + Point operator() (Point const& x) const { + VT y= Rcpp::as(hessian(Rcpp::wrap(x.getCoefficients()))); + return Point(y); + } + + }; +}; + +#endif diff --git a/src/volesti/include/ode_solvers/randomized_midpoint.hpp b/src/volesti/include/ode_solvers/randomized_midpoint.hpp new file mode 100644 index 00000000..d2a58e3c --- /dev/null +++ b/src/volesti/include/ode_solvers/randomized_midpoint.hpp @@ -0,0 +1,226 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// Based on https://papers.nips.cc/paper/8483-the-randomized-midpoint-method-for-log-concave-sampling.pdf + +#ifndef ODE_SOLVERS_RANDOMIZED_MIDPOINT_HPP +#define ODE_SOLVERS_RANDOMIZED_MIDPOINT_HPP + +template < +typename Point, +typename NT, +typename Polytope, +typename func, +typename RandomNumberGenerator +> +struct RandomizedMipointSDESolver { + + typedef std::vector pts; + + typedef std::vector bounds; + typedef typename Polytope::VT VT; + typedef typename Polytope::MT MT; + + unsigned int dim; + + VT Ar, Av; + + NT eta; + NT t; + NT u; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + pts xs_prev; + Point y, w, z; + + Point W1, W2, W3; + VT Y, Z; + MT C, L, D, M; + + std::pair pbpair; + + RandomizedMipointSDESolver(NT initial_time, NT step, pts initial_state, func oracle, bounds boundaries, NT u_=NT(1)) : + eta(step), t(initial_time), u(u_), F(oracle), Ks(boundaries), xs(initial_state) { + dim = xs[0].dimension(); + Y.resize(2 * dim); + Z.resize(2 * dim); + C.resize(2 * dim, 2 * dim); + L.resize(2 * dim, 2 * dim); + D.resize(2 * dim, 2 * dim); + M.resize(2 * dim, 2 * dim); + W1 = Point(dim); + W2 = Point(dim); + W3 = Point(dim); + + for (unsigned int i = 0; i < 2 * dim; i++) { + Y(i) = NT(0); + Z(i) = NT(0); + for (unsigned int j = 0; j < 2 * dim; j++) { + C(i, j) = NT(0); + D(i, j) = NT(0); + } + } + }; + + void step(RandomNumberGenerator &rng) { + xs_prev = xs; + unsigned int x_index, v_index, it; + NT a; + t += eta; + for (unsigned int i = 1; i < xs.size(); i += 2) { + + a = rng.sample_urdist(); + + x_index = i - 1; + v_index = i; + + calculate_Ws(a, rng); + + z = xs_prev[x_index]; + z = z + (0.5 * (1 - exp(- 2 * a * eta))) * xs_prev[v_index]; + z = z - (0.5 * u * (a * eta - 0.5 * (1 - exp(-2 * a * eta)))) * F(v_index, xs_prev, t); + z = z + sqrt(u) * W1; + + w = xs_prev[x_index]; + xs[x_index] = xs_prev[x_index]; + y = (-1.0) * xs_prev[x_index]; + xs_prev[x_index] = z; + xs[x_index] = xs[x_index] + (0.5 * (1 - exp(-2 * eta))) * xs[v_index]; + xs[x_index] = xs[x_index] + (0.5 * u * eta * (1 - exp(-2 * (eta - a * eta)))) * F(v_index, xs_prev, t); + xs[x_index] = xs[x_index] + sqrt(u) * W2; + + xs[v_index] = exp(- 2 * eta) * xs_prev[v_index]; + xs[v_index] = xs[v_index] + u * eta * exp(-2 * (eta - a * eta)) * F(v_index, xs_prev, t); + xs[v_index] = xs[v_index] + (2 * sqrt(u)) * W3; + + xs_prev[x_index] = w; + + y = y + xs[x_index]; + + if (Ks[x_index] == NULL) { + xs[x_index] = xs_prev[x_index] + y; + } + else { + // Find intersection (assuming a line trajectory) between x and y + do { + + pbpair = Ks[x_index]->line_positive_intersect(xs_prev[x_index], y, Ar, Av); + + if (pbpair.first >= 0 && pbpair.first <= 1) { + xs_prev[x_index] += (pbpair.first * 0.95) * y; + Ks[x_index]->compute_reflection(y, xs_prev[x_index], pbpair.second); + xs[x_index] = xs_prev[x_index] + y; + + // Reflect velocity + Ks[x_index]->compute_reflection(xs[v_index], xs[x_index], pbpair.second); + } + else { + xs[x_index] = xs_prev[x_index] + y; + } + } while (!Ks[x_index]->is_in(xs[x_index])); + } + + } + } + + void print_state() { + for (int j = 0; j < xs.size(); j ++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, RandomNumberGenerator &rng) { + for (int i = 0; i < num_steps; i++) step(rng); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + + void calculate_Ws(NT &a, RandomNumberGenerator &rng) { + // Initialize matrices to zero + Y = 0 * Y; + Z = 0 * Z; + C = 0 * C; + D = 0 * D; + + // Helper variables + NT temp_y, temp_z, h1, h2, g1, g2; + + // Variance of variable G1, G2 + temp_y = 0.25 * (exp(4 * a * eta) - 1); + temp_z = 0.25 * (exp(4 * eta) - exp(4 * a * eta)); + for (unsigned int i = 0; i < dim; i++) { + C(i, i) = temp_y; + D(i, i) = temp_z; + } + + // Variance of H1, H2 + temp_y = a * eta; + temp_z = (eta - a * eta); + for (unsigned int i = dim; i < 2 * dim; i++) { + C(i, i) = temp_y; + D(i, i) = temp_z; + } + + // Covariances of Hi, Gi + temp_y = 0.5 * (exp(2 * a * eta) - 1); + temp_z = 0.5 * (exp(2 * eta) - exp(2 * a * eta)); + for (unsigned int i = 0; i < dim; i++) { + C(i, i + dim) = temp_y; + C(i + dim, i) = temp_y; + D(i, i + dim) = temp_z; + D(i + dim, i) = temp_z; + } + + // Cholesky Decomposition + Eigen::LLT lltofC(C); + L = lltofC.matrixL(); + Eigen::LLT lltofD(D); + M = lltofD.matrixL(); + + // Normal Vectors + for (unsigned int i = 0; i < 2 * dim; i++) { + Y(i) = rng.sample_ndist(); + Z(i) = rng.sample_ndist(); + } + + // Transformed vectors + Y = L * Y; + Z = M * Z; + + // Calculate Brownian Integrals W1, W2, W3 (Appendix A) + for (int i = 0; i < dim; i++) { + h1 = Y(i + dim); + g1 = Y(i); + h2 = Z(i + dim); + g2 = Z(i); + W1.set_coord(i, h1 - exp(- 2 * a * eta) * g1); + W2.set_coord(i, h1 + h2 - exp(- 2 * a * eta) * (g1 + g2)); + W3.set_coord(i, exp(- 2 * eta) * (g1 + g2)); + } + + } + +}; + + +#endif diff --git a/src/volesti/include/ode_solvers/richardson_extrapolation.hpp b/src/volesti/include/ode_solvers/richardson_extrapolation.hpp new file mode 100644 index 00000000..5d6eb887 --- /dev/null +++ b/src/volesti/include/ode_solvers/richardson_extrapolation.hpp @@ -0,0 +1,188 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_RICHARDSON_EXTRAPOLATION_HPP +#define ODE_SOLVERS_RICHARDSON_EXTRAPOLATION_HPP + + +template < + typename Point, + typename NT, + typename Polytope, + typename func +> +struct RichardsonExtrapolationODESolver { + + typedef std::vector pts; + + typedef std::vector bounds; + typedef std::vector coeffs; + typedef std::vector scoeffs; + typedef std::vector ptsv; + typedef std::vector ptsm; + + typedef typename Polytope::VT VT; + + unsigned int dim; + const unsigned int MAX_TRIES = 5; + + NT eta, eta_temp; + NT t, t_prev; + NT tol = 1e-7; + NT error = NT(-1); + NT den; + Point num, y; + VT Ar, Av; + + RKODESolver *solver; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + pts xs_prev; + + ptsm A; + bool flag; + + // Previous state boundary point + Point x_prev_bound; + + // Previous state boundary facet + int prev_facet = -1; + + RichardsonExtrapolationODESolver(NT initial_time, NT step, pts initial_state, + func oracle, bounds boundaries) : + eta(step), t(initial_time), F(oracle), Ks(boundaries), xs(initial_state) { + dim = xs[0].dimension(); + A = ptsm(MAX_TRIES+1, ptsv(MAX_TRIES+1, pts(xs.size()))); + initialize_solver(); + }; + + void initialize_solver() { + solver = new RKODESolver(t, eta, xs, F, bounds{NULL}); + } + + void step(int k, bool accepted) { + xs_prev = xs; + eta_temp = eta; + flag = true; + + // Use RK4 solver + solver->xs = xs_prev; + solver->t = t; + solver->eta = eta_temp; + solver->steps(1, false); + A[1][1] = solver->xs; + + + for (unsigned int j = 1; j <= MAX_TRIES-1; j++) { + // Reduce step size by two + eta_temp /= 2; + + // Find solution with half stepsize and twice the num of steps + solver->xs = xs_prev; + solver->t = t; + solver->eta = eta_temp; + solver->steps(2*j, false); + A[j+1][1] = solver->xs; + + // Perform Richardson extrapolation + for (unsigned int k = 1; k <= j; k++) { + den = 1.0 * ((4 << k) - 1); + for (unsigned int i = 0; i < xs.size(); i++) { + num = (1.0 * (4 << k)) * A[j+1][k][i]; + num = num - A[j][k][i]; + A[j+1][k+1][i] = (1 / den) * num; + } + } + + error = NT(-1); + + for (unsigned int i = 0; i < xs.size(); i++) { + y = A[j+1][j+1][i] - A[j][j][i]; + if (sqrt(y.dot(y)) > error) error = sqrt(y.dot(y)); + } + + if (error < tol || j == MAX_TRIES - 1) { + for (unsigned int i = 0; i < xs.size(); i++) { + y = A[j+1][j+1][i] - xs[i]; + + if (Ks[i] == NULL) { + xs[i] = xs_prev[i] + y; + if (prev_facet != -1 && i > 0) { + Ks[i-1]->compute_reflection(xs[i], x_prev_bound, prev_facet); + } + prev_facet = -1; + } + else { + + // Find intersection (assuming a line trajectory) between x and y + do { + // Find line intersection between xs[i] (new position) and y + std::pair pbpair = Ks[i]->line_positive_intersect(xs_prev[i], y, Ar, Av); + // If point is outside it would yield a negative param + if (pbpair.first >= 0 && pbpair.first <= 1) { + + xs_prev[i] += (pbpair.first * 0.95) * y; + + // Update facet for reflection of derivative + prev_facet = pbpair.second; + x_prev_bound = xs_prev[i]; + + // Reflect ray y on the boundary point y now is the reflected ray + Ks[i]->compute_reflection(y, xs_prev[i], pbpair.second); + // Add it to the existing (boundary) point and repeat + xs[i] = xs_prev[i] + y; + + } + else { + prev_facet = -1; + xs[i] = xs_prev[i] + y; + } + } while (!Ks[i]->is_in(xs[i])); + + } + + } + break; + } + } + + t += eta; + + } + + void print_state() { + for (int j = 0; j < xs.size(); j++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) step(i, accepted); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + +}; + + +#endif diff --git a/src/volesti/include/ode_solvers/runge_kutta.hpp b/src/volesti/include/ode_solvers/runge_kutta.hpp new file mode 100644 index 00000000..5d8793ca --- /dev/null +++ b/src/volesti/include/ode_solvers/runge_kutta.hpp @@ -0,0 +1,172 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ODE_SOLVERS_RUNGE_KUTTA_H +#define ODE_SOLVERS_RUNGE_KUTTA_H + + +template < + typename Point, + typename NT, + typename Polytope, + typename func +> +struct RKODESolver { + + typedef std::vector pts; + typedef std::vector ptsv; + + typedef std::vector bounds; + typedef std::vector coeffs; + typedef std::vector scoeffs; + + typedef typename Polytope::VT VT; + + unsigned int dim; + + NT eta; + NT t, t_prev; + + VT Ar, Av; + + func F; + bounds Ks; + + // Contains the sub-states + pts xs; + ptsv ks; + Point y; + + // Previous state boundary point + Point x_prev_bound; + + // Previous state boundary facet + int prev_facet = -1; + + scoeffs as; + coeffs cs, bs; + + + RKODESolver(NT initial_time, NT step, pts initial_state, func oracle, + bounds boundaries) : + eta(step), t(initial_time), F(oracle), Ks(boundaries), xs(initial_state) { + dim = xs[0].dimension(); + // If no coefficients are given the RK4 method is assumed + cs = coeffs{0, 0.5, 0.5, 1}; + bs = coeffs{1.0/6, 1.0/3, 1.0/3, 1.0/6}; + as = scoeffs{ + coeffs{}, + coeffs{0.5}, + coeffs{0, 0.5}, + coeffs{0, 0, 1.0} + }; + }; + + + // RKODESolver(NT initial_time, NT step, pts initial_state, func oracle, + // bounds boundaries, scoeffs a_coeffs, coeffs b_coeffs, coeffs c_coeffs) : + // t(initial_time), xs(initial_state), F(oracle), eta(step), Ks(boundaries), + // as(a_coeffs), bs(b_coeffs), cs(c_coeffs) { + // dim = xs[0].dimension(); + // }; + + unsigned int order() { + return bs.size(); + } + + void step(int k, bool accepted) { + ks = ptsv(order(), xs); + t_prev = t; + + for (unsigned int ord = 0; ord < order(); ord++) { + // Initialize t to previous + t = t_prev + cs[ord] * eta; + + // Initialize ks to previous solution we use + // Initialize argument + for (int j = 0; j < ord; j++) { + for (int r = 0; r < xs.size(); r++) { + y = ks[j][r]; + y = (eta * as[ord][j]) * y; + ks[ord][r] = ks[ord][r] + y; + } + } + + for (unsigned int i = 0; i < xs.size(); i++) { + // Calculate k_i s + y = F(i,ks[ord], t); + ks[ord][i] = y; + y = (eta * bs[i]) * y; + + if (Ks[i] == NULL) { + xs[i] = xs[i] + y; + if (prev_facet != -1 && i > 0) + Ks[i-1]->compute_reflection(xs[i], x_prev_bound, prev_facet); + prev_facet = -1; + } + else { + + // Find intersection (assuming a line trajectory) between x and y + do { + // Find line intersection between xs[i] (new position) and y + std::pair pbpair = Ks[i]->line_positive_intersect(xs[i], y, Ar, Av); + // If point is outside it would yield a negative param + if (pbpair.first >= 0 && pbpair.first <= 1) { + // Advance to point on the boundary + xs[i] += (pbpair.first * 0.99) * y; + + // Update facet for reflection of derivative + prev_facet = pbpair.second; + x_prev_bound = xs[i]; + + // Reflect ray y on the boundary point y now is the reflected ray + Ks[i]->compute_reflection(y, xs[i], pbpair.second); + // Add it to the existing (boundary) point and repeat + xs[i] += y; + + } + else { + prev_facet = -1; + xs[i] += y; + } + } while (!Ks[i]->is_in(xs[i])); + + } + + } + } + + } + + void print_state() { + for (int j = 0; j < xs.size(); j++) { + for (unsigned int i = 0; i < xs[j].dimension(); i++) { + std::cout << xs[j][i] << " "; + } + } + std::cout << std::endl; + } + + void steps(int num_steps, bool accepted) { + for (int i = 0; i < num_steps; i++) step(i, accepted); + } + + Point get_state(int index) { + return xs[index]; + } + + void set_state(int index, Point p) { + xs[index] = p; + } + +}; + + +#endif diff --git a/src/volesti/include/optimization/simulated_annealing.hpp b/src/volesti/include/optimization/simulated_annealing.hpp new file mode 100644 index 00000000..77e15b12 --- /dev/null +++ b/src/volesti/include/optimization/simulated_annealing.hpp @@ -0,0 +1,152 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_SIMULATED_ANNEALING_HPP +#define VOLESTI_SIMULATED_ANNEALING_HPP + +#include + +#include "optimization/sliding_window.hpp" +#include "random_walks/boltzmann_hmc_walk.hpp" + + +/// A magic number! +/// when estimating the diameter of the spectrahedron, +/// sample 20 + sqrt(dimension) points to estimate it +#define CONSTANT_1 20 + +/// Holds parameters of the algorithm +/// \tparam Point Point Type +template +struct SimulatedAnnealingSettings { + /// The numeric type + typedef typename Point::FT NT; + + /// Desired accuracy (relative error) + NT error; + /// The walk length of the HMC random walk + int walkLength; + /// A bound to the number of steps; if negative it is unbounded + int maxNumSteps; + + /// Starting from an initial temperature, at each step it will decrease by a factor of + /// \[ 1 - 1 / (dimension^k) \]. Default is 0.5 + NT k; + + SimulatedAnnealingSettings(NT const error, int const walkLength = 1, int const maxNumSteps = -1, NT const k = 0.5) : error(error), + walkLength(walkLength), maxNumSteps(maxNumSteps), k(k) {} +}; + + +/// Simulated Annealing algorithm for a semidefinite program +/// Minimize \[ c^T x \], s.t. LMI(x) <= 0 +/// \param[in] spectrahedron A spectrahedron described by a linear matrix inequality +/// \param[in] objectiveFunction The function we minimize +/// \param[in] settings Parameters of the algorithm +/// \param[in] interiorPoint An initial feasible solution to start the algorithm +/// \param[out] solution The vector minimizing the objective function +/// \param[in] verbose True to print messages. Default is false +/// \return The best approximation to the optimal solution +template +double solve_sdp(_Spectrahedron & spectrahedron, Point const & objectiveFunction, _Settings const & settings, + Point const & interiorPoint, Point& solution, bool verbose = false) { + + // fetch the data types we will use + typedef typename _Spectrahedron::NT NT; + typedef typename _Spectrahedron::MT MT; + typedef typename _Spectrahedron::VT VT; + typedef BoostRandomNumberGenerator RNGType; + typedef BoltzmannHMCWalk::Walk<_Spectrahedron, RNGType > HMC; + + // the algorithm requires the objective function to be normalized + // we will need to remember the norm + VT _objectiveFunctionNormed = objectiveFunction.getCoefficients(); + NT objectiveFunctionNorm = _objectiveFunctionNormed.norm(); + _objectiveFunctionNormed.normalize(); + Point objectiveFunctionNormed = Point(_objectiveFunctionNormed); + + RNGType rng(spectrahedron.dimension()); + + // Estimate the diameter of the spectrahedron + // needed for the random walk and for the simulated annealing algorithm + NT diameter = spectrahedron.estimateDiameter(CONSTANT_1 + std::sqrt(spectrahedron.dimension()), interiorPoint, rng); + + /******** initialization *********/ + solution = interiorPoint; + // the minimum till last iteration + NT currentMin = objectiveFunction.dot(solution); + int stepsCount = 0; + // initial temperature must be the diameter of the body + NT temperature = diameter; + // after each iteration, temperature = temperature * tempDecreaseFactor + NT tempDecreaseFactor = 1.0 - static_cast(1.0 / std::pow(spectrahedron.dimension(), settings.k)); + + // initialize random walk; + typename HMC::Settings hmc_settings = typename HMC::Settings(settings.walkLength, rng, objectiveFunction, temperature, diameter); + HMC hmcRandomWalk = HMC(hmc_settings); + NT previous_min = objectiveFunction.dot(solution); + + /******** solve *********/ + // if settings.maxNumSteps is negative there is no + // bound to the number of steps - stop + // when desired relative error is achieved + while (stepsCount < settings.maxNumSteps || settings.maxNumSteps < 0) { + + // sample one point with current temperature + std::list randPoints; + + // get a sample under the Boltzmann distribution + // using the HMC random walk + while (1) { + hmcRandomWalk.apply(spectrahedron, solution, settings.walkLength, randPoints); + + // if the sampled point is not inside the spectrahedron (error in boundary oracle), + // get a new one + if (spectrahedron.isExterior(spectrahedron.get_C())) { + if (verbose) std::cout << "Sampled point outside the spectrahedron.\n"; + randPoints.clear(); + spectrahedron.resetFlags(); + } + else { + // update values; + solution = randPoints.back(); + randPoints.clear(); + break; + } + } + + // update current value + currentMin = objectiveFunction.dot(solution); + ++stepsCount; + + // compute relative error + NT relError = relativeError(previous_min, currentMin); + previous_min = currentMin; + + if (verbose) + std::cout << "Step: " << stepsCount << ", Temperature: " << temperature << ", Min: " << currentMin + << ", Relative error: " << relError << "\n"; + + // check if we reached desired accuracy + if (relError < settings.error) + break; + + // decrease the temperature + temperature *= tempDecreaseFactor; + hmcRandomWalk.setTemperature(temperature); + + } /* while (stepsCount < settings.maxNumSteps || settings.maxNumSteps < 0) { */ + + // return the minimum w.r.t. the original objective function + return currentMin*objectiveFunctionNorm; +} + + + +#endif //VOLESTI_SIMULATED_ANNEALING_HPP diff --git a/src/volesti/include/optimization/sliding_window.hpp b/src/volesti/include/optimization/sliding_window.hpp new file mode 100644 index 00000000..daab137a --- /dev/null +++ b/src/volesti/include/optimization/sliding_window.hpp @@ -0,0 +1,66 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_SLIDING_WINDOW_HPP +#define VOLESTI_SLIDING_WINDOW_HPP + + +/// Computes the relative error +/// \tparam NT Numeric type +/// \param[in] approx The approximated value +/// \param[in] exact The exact value +/// \return The relative error +template +NT relativeError(NT approx, NT exact) { + return std::fabs((exact - approx) / exact); +} + + +/// A sliding window, which allows to get the relative error of approximations +/// \tparam NT Numeric type +template +class SlidingWindow { +public: + + /// The stored approximations + std::list approximations; + /// The size of the window + int windowSize; + /// The number of entries in list approximations + int numEntries; + + /// Constructor + /// \param[in] windowSize The size of the window + SlidingWindow(int windowSize) : windowSize(windowSize) { + numEntries = 0; + } + + /// Adds an approximation in the window + /// \param[in] approximation The new approximation + void push(NT approximation) { + // if window is full, remove the oldest value + if (numEntries >= windowSize) { + approximations.pop_back(); + } + else + numEntries++; + + approximation.push_front(approximation); + } + + /// \return The relative error between the youngest and oldest approximations + double getRelativeError() { + if (numEntries < windowSize) + return 1; + + return relativeError(approximations.back(), approximations.front()); + } +}; + +#endif //VOLESTI_SLIDING_WINDOW_HPP diff --git a/src/volesti/include/preprocess/crhmc/analytic_center.h b/src/volesti/include/preprocess/crhmc/analytic_center.h new file mode 100644 index 00000000..7124626b --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/analytic_center.h @@ -0,0 +1,173 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef ANALYTIC_CENTER_H +#define ANALYTIC_CENTER_H +#include "Eigen/Eigen" +#include "PackedCSparse/PackedChol.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include "preprocess/crhmc/opts.h" +#include +#include +#include +#ifndef SIMD_LEN +#define SIMD_LEN 0 +#endif +const size_t chol_k2 = (SIMD_LEN == 0) ? 1 : SIMD_LEN; + +/*This function computes the analytic center of the polytope*/ +//And detects additional constraint that need to be added +// x - It outputs the minimizer of min f(x) subjects to {Ax=b} +// C - detected constraint matrix +// If the domain ({Ax=b} intersect dom(f)) is not full dimensional in {Ax=b} +// because of the dom(f), the algorithm will detect the collapsed dimension +// and output the detected constraint C x = d +// d - detected constraint vector +template +std::tuple analytic_center(SpMat const &A, VT const &b, Polytope &f, Opts const &options, VT x = VT::Zero(0, 1)) +{ + using CholObj = typename Polytope::CholObj; + using Triple = typename Polytope::Triple; + using Tx = typename Polytope::Tx; + // initial conditions + int n = A.cols(); + int m = A.rows(); + if (x.rows() == 0 || !f.barrier.feasible(x)) + { + x = f.barrier.center; + } + + VT lambda = VT::Zero(n, 1); + int fullStep = 0; + NT tConst = 0; + NT primalErr = std::numeric_limits::max(); + NT dualErr = std::numeric_limits::max(); + NT primalErrMin = std::numeric_limits::max(); + NT primalFactor = 1; + NT dualFactor = 1 + b.norm(); + std::vector idx; + + CholObj solver = CholObj(transform_format(A)); + solver.accuracyThreshold = 0; + for (int iter = 0; iter < options.ipmMaxIter; iter++) + { + std::pair pair_analytic_oracle = f.analytic_center_oracle(x); + VT grad = pair_analytic_oracle.first; + VT hess = pair_analytic_oracle.second; + + // compute the residual + VT rx = lambda - grad; + VT rs = b - A * x; + + // check stagnation + primalErrMin = std::min(primalErr, primalErrMin); + primalErr = rx.norm() / primalFactor; + NT dualErrLast = dualErr; + dualErr = rs.norm() / dualFactor; + bool feasible = f.barrier.feasible(x); + //Compare the dual and primal error to the last and minimum so far + if ((dualErr > (1 - 0.9 * tConst) * dualErrLast) || + (primalErr > 10 * primalErrMin) || !feasible) + { + VT dist = f.barrier.boundary_distance(x); + NT th = options.ipmDistanceTol; + visit_lambda(dist, [&idx, th](double v, int i, int j) + { + if (v < th) + idx.push_back(i); }); + if (idx.size() > 0) + { + break; + } + } + + // compute the step direction + VT Hinv = hess.cwiseInverse(); + solver.decompose((Tx *)Hinv.data()); + VT out(m, 1); + solver.solve((Tx *)rs.data(), (Tx *)out.data()); + VT dr1 = A.transpose() * out; + VT in = A * Hinv.cwiseProduct(rx); + solver.solve((Tx *)in.data(), (Tx *)out.data()); + + VT dr2 = A.transpose() * out; + VT dx1 = Hinv.cwiseProduct(dr1); + VT dx2 = Hinv.cwiseProduct(rx - dr2); + + // compute the step size + VT dx = dx1 + dx2; + NT tGrad = std::min(f.barrier.step_size(x, dx), 1.0); + dx = dx1 + tGrad * dx2; + NT tConst = std::min(0.99 * f.barrier.step_size(x, dx), 1.0); + tGrad = tGrad * tConst; + + // make the step + x = x + tConst * dx; + lambda = lambda - dr2; + + if (!f.barrier.feasible(x)) + { + break; + } + //If we have have converged + if (tGrad == 1) + { + //do some more fullStep + fullStep = fullStep + 1; + if (fullStep > log(dualErr / options.ipmDualTol) && fullStep > options.min_convergence_steps) + { + break; + } + } + else + { + fullStep = 0; + } + } + SpMat C; + VT d; + if (idx.size() == 0) + { + VT dist = f.barrier.boundary_distance(x); + NT th = options.ipmDistanceTol; + visit_lambda(dist, [&idx, th](double v, int i, int j) + { + if (v < th) + idx.push_back(i); }); + } + + if (idx.size() > 0) + { + C.resize(idx.size(), n); + std::pair pboundary = f.barrier.boundary(x); + VT A_ = pboundary.first; + VT b_ = pboundary.second; + std::vector sparseIdx; + for (int i = 0; i < idx.size(); i++) + { + sparseIdx.push_back(Triple(i, idx[i], A_(idx[i]))); + } + C.setFromTriplets(sparseIdx.begin(), sparseIdx.end()); + d.resize(idx.size(), 1); + copy_indicies(d, b_, idx); + } + else + { + C = MT::Zero(0, n).sparseView(); + d = VT::Zero(0, 1); + } + return std::make_tuple(x, C, d); +} +#endif diff --git a/src/volesti/include/preprocess/crhmc/constraint_problem.h b/src/volesti/include/preprocess/crhmc/constraint_problem.h new file mode 100644 index 00000000..82931505 --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/constraint_problem.h @@ -0,0 +1,81 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef CONSTRAINT_PROBLEM_H +#define CONSTRAINT_PROBLEM_H +#include "Eigen/Eigen" +/*Input structure: With this the user can define a polytope sampling problem*/ +template +class constraint_problem { +public: + using Type = typename Point::FT; + using PointType = Point; + using VT = Eigen::Matrix; + using MT = MatrixType; +private: + unsigned int num_vars; // num_vars of the original problem + MatrixType Aeq; // Matrix of coefficients for the equality constraints + VT beq; // Right hand side of the equality constraints + MatrixType Aineq; // Matrix of coefficients for the inequality constraints + VT bineq; // Right hand side of the inequality constraints + VT lb; // lb on the output coordinates preset to -1e9 + VT ub; // ub on the output coordinates preset to +1e9 + Type inf = 1e9; +public: + /*Constructors for different input instances*/ + constraint_problem(const int dim, MT const &Aeq_, VT const &beq_, MT const &Aineq_, + VT const &bineq_, VT const &lb_, VT const &ub_) + : num_vars(dim), Aeq(Aeq_), beq(beq_), Aineq(Aineq_), bineq(bineq_), + lb(lb_), ub(ub_) { + } + + constraint_problem(const int dim) { + num_vars = dim; + init(num_vars); + } + + + void init(int num_vars) { + Aineq.resize(0, num_vars); + Aeq.resize(0, num_vars); + bineq.resize(0, 1); + beq.resize(0, 1); + lb = -VT::Ones(num_vars) * inf; + ub = VT::Ones(num_vars) * inf; + } + void set_equality_constraints(MT const &Aeq_, VT const &beq_){ + Aeq = Aeq_; + beq = beq_; + } + void set_inequality_constraints(MT const &Aineq_, VT const &bineq_){ + Aineq = Aineq_; + bineq = bineq_; + } + void set_bounds(VT const &lb_, VT const &ub_){ + lb = lb_; + ub = ub_; + } + std::pair get_equations(){ + return std::make_pair(Aeq,beq); + } + std::pair get_inequalities(){ + return std::make_pair(Aineq,bineq); + } + std::pair get_bounds(){ + return std::make_pair(lb,ub); + } + unsigned int dimension(){ + return num_vars; + } + +}; + +#endif diff --git a/src/volesti/include/preprocess/crhmc/crhmc_input.h b/src/volesti/include/preprocess/crhmc/crhmc_input.h new file mode 100644 index 00000000..05f61c4b --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/crhmc_input.h @@ -0,0 +1,172 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef CRHMC_INPUT_H +#define CRHMC_INPUT_H +#include "Eigen/Eigen" +#include "opts.h" +#include "convex_bodies/hpolytope.h" +#include "preprocess/crhmc/constraint_problem.h" +/*0 funciton handles are given as a reference in case the user gives no +function. Then the uniform function is implied*/ +template +struct ZeroFunctor +{ + using Type = typename Point::FT; + Point operator()(Point const &x) const { return Point(x.dimension()); } + struct parameters { + Type L=1; + Type eta=1; + }; + struct parameters params; +}; +template +struct ZeroScalarFunctor +{ + using Type = typename Point::FT; + Type operator()(Point const &x) const { return 0; } +}; + +/// +/// Input structure: With this the user can define the input for a crhmc polytope sampling problem +template , + typename grad = ZeroFunctor, + typename hess = ZeroFunctor> +class crhmc_input +{ + using Type = typename Point::FT; + using VT = Eigen::Matrix; + ZeroFunctor zerof; + ZeroScalarFunctor zerosf; + +public: + using MT = MatrixType; + using Func = func; + using Grad = grad; + using Hess = hess; + using point= Point; + MatrixType Aineq; // Matrix of coefficients for the inequality constraints + VT bineq; // Right hand side of the inequality constraints + MatrixType Aeq; // Matrix of coefficients for the equality constraints + VT beq; // Right hand side of the equality constraints + opts options; // structure of the parameters of the problem + VT lb; // lb on the output coordinates preset to -1e7 + VT ub; // ub on the output coordinates preset to +1e7 + func &f; // Negative log density function handle + grad &df; // Negative log density gradient function handle + hess &ddf; // Negative log density hessian function handle + bool fZero; // whether f is completely zero + bool fHandle; // whether f is handle or not + bool dfHandle; // whether df is handle or not + bool ddfHandle; // whether ddf is handle or not + unsigned int dimension; // dimension of the original problem + const Type inf = options.max_coord + 1; // helper for barrier handling + /*Constructors for different input instances*/ + crhmc_input(int dim, func &function, grad &g, hess &h) + : f(function), df(g), ddf(h) + { dimension=dim; + fZero = false; + fHandle = true; + dfHandle = true; + ddfHandle = true; + init(dimension); + } + crhmc_input(int dim, func &function) + : f(function), df(zerof), ddf(zerof) + { dimension=dim; + fZero = false; + fHandle = true; + dfHandle = false; + ddfHandle = false; + init(dimension); + } + crhmc_input(int dim, func &function, grad &g) + : f(function), df(g), ddf(zerof) + { dimension=dim; + fZero = false; + fHandle = true; + dfHandle = true; + ddfHandle = false; + init(dimension); + } + crhmc_input(int dim) : f(zerosf), df(zerof), ddf(zerof) + { dimension=dim; + fZero = true; + fHandle = false; + dfHandle = false; + ddfHandle = false; + init(dimension); + } + + void init(int dimension) + { + Aineq.resize(0, dimension); + Aeq.resize(0, dimension); + bineq.resize(0, 1); + beq.resize(0, 1); + lb = -VT::Ones(dimension) * inf; + ub = VT::Ones(dimension) * inf; + } +}; +#include + +template < + typename Input, typename Polytope, typename Func, typename Grad, typename Hess, + typename std::enable_if>::value>::type * = nullptr> +inline Input convert2crhmc_input(Polytope &P, Func &f, Grad &g, Hess &h) { + int dimension = P.dimension(); + Input input = Input(dimension, f, g, h); + if (std::is_same< + Hess, ZeroFunctor>::value){ + input.ddfHandle=false; + } + input.Aineq = P.get_mat(); + input.bineq = P.get_vec(); + return input; +} + +template >::value>::type + * = nullptr> +inline Input convert2crhmc_input(Polytope &P, Func &f, Grad &g, Hess &h) { + int dimension = P.dimension(); + Input input = Input(dimension, f, g, h); + if (std::is_same< + Hess, ZeroFunctor>::value){ + input.ddfHandle=false; + } + std::tie(input.Aineq, input.bineq) = P.get_inequalities(); + std::tie(input.Aeq, input.beq) = P.get_equations(); + std::tie(input.lb, input.ub) = P.get_bounds(); + return input; +} +template >::value && + !std::is_same>::value>:: + type * = nullptr> +inline Input convert2crhmc_input(Polytope &P, Func &f, Grad &g, Hess &h) { + /*CRHMC works only for H-polytopes and constraint_problems for now*/ + int dimension = 0; + Input input = Input(dimension, f, g, h); + return input; +} +#endif diff --git a/src/volesti/include/preprocess/crhmc/crhmc_problem.h b/src/volesti/include/preprocess/crhmc/crhmc_problem.h new file mode 100644 index 00000000..c501bce6 --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/crhmc_problem.h @@ -0,0 +1,703 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef CRHMC_PROBLEM_H +#define CRHMC_PROBLEM_H +#include "Eigen/Eigen" +#include "PackedCSparse/PackedChol.h" +#include "cartesian_geom/cartesian_kernel.h" +#include "convex_bodies/hpolytope.h" +#include "preprocess/crhmc/analytic_center.h" +#include "preprocess/crhmc/crhmc_input.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include "preprocess/crhmc/lewis_center.h" +#include "preprocess/crhmc/opts.h" +#include "preprocess/crhmc/two_sided_barrier.h" +#include +#include +#include +#include +#include + +#ifndef SIMD_LEN +#define SIMD_LEN 0 +#endif +const size_t chol_k = (SIMD_LEN == 0) ? 1 : SIMD_LEN; + +/// +/// Crhmc sampling problem: With this the user can define a crhmc polytope sampling problem +/// @tparam Point Point type +/// @tparam Input Input format +template +class crhmc_problem { +public: + using NT = double; + using PolytopeType = HPolytope; + using MT = Eigen::Matrix; + using VT = Eigen::Matrix; + using IVT = Eigen::Matrix; + using SpMat = Eigen::SparseMatrix; + using PM = Eigen::PermutationMatrix; + using IndexVector = Eigen::Matrix; + using CholObj = PackedChol; + using Triple = Eigen::Triplet; + using Barrier = two_sided_barrier; + using Tx = FloatArray; + using Opts = opts; + using Diagonal_MT = Eigen::DiagonalMatrix; + using Func = typename Input::Func; + using Grad = typename Input::Grad; + using Hess = typename Input::Hess; + using Crhmc_problem=crhmc_problem; + + unsigned int _d; // dimension + // Problem variables Ax=b st lb<=x<=ub + MT A; // matrix A input matrix + SpMat Asp; // problem matrix A in Sparse form + VT b; // vector b, s.t.: Ax=b + VT lb; // Lower bound for output coordinates + VT ub; // Upper bound for output coordinates + Barrier barrier; // Class that holds functions that handle the log barrier for + // lb and ub + Opts options; // problem parameters + // Transformation (T,y) such that the new variables x + // can be tranform to the original z (z= Tx+y) + SpMat T; + VT y; + // Non zero indices and values for fast tranform + std::vector Tidx; // T x = x(Tidx) .* Ta + VT Ta; // T x = x(Tidx) .* Ta + bool isempty_center = true; + VT center = VT::Zero(0, 1); // Resulting polytope Lewis or Analytic center + VT analytic_ctr; //analytic center vector (for testing) + VT w_center;// weights of the lewis center + + VT width; // width of the varibles + int nP;//input dimension + + Func &func; // function handle + Grad &df; // gradient handle + Hess &ddf; // hessian handle + bool fZero; // whether f is completely zero + bool fHandle; // whether f is handle or not + bool dfHandle; // whether df is handle or not + bool ddfHandle; // whether ddf is handle or not + /*Invalid polytope variables*/ + bool terminate=false; + std::string terminate_message; +#ifdef TIME_KEEPING +//Timing information + std::chrono::duration rescale_duration, sparsify_duration, + reordering_duration, rm_rows_duration, rm_fixed_vars_duration, + ex_collapsed_vars_duration, shift_barrier_duration, lewis_center_duration; +#endif + const NT inf = options.max_coord; // helper for barrier handling + const NT barrier_bound = 1e7; + int equations() const { return Asp.rows(); } + int dimension() const { return Asp.cols(); } + int nnz() const { return Asp.nonZeros(); } + + // Remove varibles that have width under some tolerance + int remove_fixed_variables(const NT tol = 1e-12) { + int m = Asp.rows(); + int n = Asp.cols(); + VT d = estimate_width(); + CholObj solver = CholObj(transform_format(Asp)); + solver.accuracyThreshold = 0; + VT w = VT::Ones(n, 1); + solver.decompose((Tx *)w.data()); + VT out_vector = VT(m, 1); + solver.solve((Tx *)b.data(), (Tx *)out_vector.data()); + VT x = Asp.transpose() * out_vector; + + x = ((x.array()).abs() < tol).select(0., x); + std::vector freeIndices; + std::vector indices; + int nFreeVars = 0; + for (int i = 0; i < n; i++) { + if (d(i) < tol * (1 + abs(x(i)))) { + } else { + freeIndices.push_back(Triple(i, nFreeVars, 1)); + nFreeVars++; + indices.push_back(i); + x(i) = 0.0; + } + } + + if (freeIndices.size() != n) { + SpMat S = SpMat(n, freeIndices.size()); + S.setFromTriplets(freeIndices.begin(), freeIndices.end()); + append_map(S, x); + copy_indicies(barrier.lb, barrier.lb, indices); + copy_indicies(barrier.ub, barrier.ub, indices); + barrier.lb.conservativeResize(indices.size()); + barrier.ub.conservativeResize(indices.size()); + + barrier.set_bound(barrier.lb, barrier.ub); + if (!isempty_center) { + copy_indicies(center, center, indices); + center.conservativeResize(indices.size()); + } + return 1; + } + return 0; + } + + int extract_collapsed_variables() { + SpMat Ac; + VT bc; + if (isempty_center) { + std::tie(center, Ac, bc) = analytic_center(Asp, b, *this, options); + isempty_center = false; + } else { + std::tie(center, Ac, bc) = + analytic_center(Asp, b, *this, options, center); + analytic_ctr=center; + } + if (Ac.rows() == 0) { + return 0; + } + SpMat _A = SpMat(Asp); + sparse_stack_v(_A, Ac, Asp); + b.conservativeResize(b.rows() + bc.rows(), 1); + b.bottomRows(bc.rows()) = bc; + return 1; + } + // Rescale the polytope for numerical stability + void rescale(const VT x = VT::Zero(0, 1)) { + if (std::min(equations(), dimension()) <= 1) { + return; + } + VT hess; + if (x.rows() == 0) { + hess = VT::Ones(dimension(), 1); + } else { + std::tie(std::ignore, hess) = analytic_center_oracle(x); + hess = hess + (width.cwiseProduct(width)).cwiseInverse(); + } + VT scale = (hess.cwiseSqrt()).cwiseInverse(); + SpMat Ain = Asp * scale.asDiagonal(); + VT cscale; + VT rscale; + + std::tie(cscale, rscale) = gmscale(Ain, 0.9); + Asp = (rscale.cwiseInverse()).asDiagonal() * Asp; + b = b.cwiseQuotient(rscale); + barrier.set_bound(barrier.lb.cwiseProduct(cscale), + barrier.ub.cwiseProduct(cscale)); + append_map((cscale.cwiseInverse()).asDiagonal(), VT::Zero(dimension(), 1)); + if (!isempty_center) { + center = center.cwiseProduct(cscale); + } + } + + // Rewrite P so that each cols has no more than maxNZ non-zeros + void splitDenseCols(const int maxnz) { + int m = Asp.rows(); + int n = Asp.cols(); + if (m <= maxnz) { + return; + } + if (Asp.nonZeros() > maxnz * n) { + return; + } + int numBadCols = 1; + lb = barrier.lb; + ub = barrier.ub; + //until there are columns with more than maxNZ elements + while (numBadCols > 0) { + m = Asp.rows(); + n = Asp.cols(); + std::vector colCounts(n); + std::vector badCols; + numBadCols = 0; + //find the columns with count larger than maxNZ + std::tie(colCounts, badCols) = nnzPerColumn(Asp, maxnz); + numBadCols = badCols.size(); + if (numBadCols == 0) { + break; + } + //create a new variable for each one and update Asp, b, lb, ub, T, y accordingly + SpMat A_; + SpMat Aj(m, numBadCols); + SpMat Ai(numBadCols, n + numBadCols); + std::vector newColumns; + std::vector newRows; + b.conservativeResize(m + numBadCols, 1); + lb.conservativeResize(n + numBadCols, 1); + ub.conservativeResize(n + numBadCols, 1); + + for (int j = 0; j < numBadCols; j++) { + int i = badCols[j]; + int k = 0; + for (SpMat::InnerIterator it(Asp, i); it; ++it) { + if (k >= colCounts[i] / 2) { + newColumns.push_back(Triple(it.row(), j, it.value())); + it.valueRef() = 0; + } + k++; + } + newRows.push_back(Triple(j, i, 1)); + newRows.push_back(Triple(j, j + n, -1)); + lb(n + j) = lb(i); + ub(n + j) = ub(i); + b(m + j) = 0; + } + Ai.setFromTriplets(newRows.begin(), newRows.end()); + Aj.setFromTriplets(newColumns.begin(), newColumns.end()); + Asp.prune(0, 0); + sparse_stack_h_inplace(Asp, Aj); + sparse_stack_v(Asp, Ai, A_); + Asp = A_; + } + SpMat _T = MT::Zero(T.rows(), ub.rows() - T.cols()).sparseView(); + sparse_stack_h_inplace(T, _T); + updateT(); + barrier.set_bound(lb, ub); + } + // Change A and the correpsonding Transformation + template + void append_map(MatrixType const &S, VT const &z) { + b = b - Asp * z; + Asp = Asp * S; + y = y + T * z; + T = T * S; + updateT(); + } + // Shift the problem with respect to x + void shift_barrier(VT const &x) { + int size = x.rows(); + b = b - Asp * x; + y = y + T * x; + barrier.set_bound(barrier.lb - x, barrier.ub - x); + if (!isempty_center) { + center = center - x; + } + } + // Reorder the polytope accordint to the AMD Reordering for better sparsity + // pattern in the Cholesky decomposition + void reorder() { + if (!options.EnableReordering || Asp.rows()*Asp.cols() < options.maxNZ) { + return; + } + fillin_reduce(Asp,b); + } +//Using the Cholesky decomposition remove dependent rows in the systen Asp*x=b + int remove_dependent_rows(NT tolerance = 1e-12, NT infinity = 1e+64) { + //this approach does not work with 0 rows + remove_zero_rows(Asp, b); + int m = Asp.rows(); + int n = Asp.cols(); + VT v = VT(m); + VT w = VT::Ones(n, 1); + CholObj solver = CholObj(transform_format(Asp)); + solver.accuracyThreshold = 0; + solver.decompose((Tx *)w.data()); + solver.diagL((Tx *)v.data()); + std::vector indices(m, false); + std::vector idx; + bool changed = false; + for (int i = 0; i < m; i++) { + if ((v(i) > tolerance) && (v(i) < infinity)) { + indices[i] = true; + idx.push_back(i); + }else{ + changed=true; + } + } + if (!changed) { + return 0; + } + + remove_rows(Asp, indices); + copy_indicies(b, b, idx); + b.conservativeResize(idx.size(), 1); + return 1; + } +//Apply number of operations that simplify the problem + void simplify() { +#ifdef TIME_KEEPING + std::chrono::time_point start, end; + start = std::chrono::system_clock::now(); +#endif + rescale(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + rescale_duration += end - start; + start = std::chrono::system_clock::now(); +#endif + splitDenseCols(options.maxNZ); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + sparsify_duration += end - start; + start = std::chrono::system_clock::now(); +#endif + reorder(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + reordering_duration += end - start; +#endif + int changed = 1; + while (changed) { + while (changed) { + changed = 0; +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + changed += remove_dependent_rows(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + rm_rows_duration += end - start; + start = std::chrono::system_clock::now(); +#endif + changed += remove_fixed_variables(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + rm_fixed_vars_duration += end - start; + start = std::chrono::system_clock::now(); +#endif + reorder(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + reordering_duration += end - start; +#endif + } +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + changed += extract_collapsed_variables(); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + ex_collapsed_vars_duration += end - start; +#endif + } + } + + VT estimate_width(bool use_center=false) { + int n = Asp.cols(); + VT hess = VT::Ones(n, 1); + if(use_center){ + std::tie(std::ignore, hess)=analytic_center_oracle(center); + } + CholObj solver = CholObj(transform_format(Asp)); + solver.accuracyThreshold = 0; + solver.decompose((Tx *)hess.data()); + VT w_vector(n, 1); + solver.leverageScoreComplement((Tx *)w_vector.data()); + w_vector = (w_vector.cwiseMax(0)).cwiseProduct(hess.cwiseInverse()); + VT tau = w_vector.cwiseSqrt(); + + return tau; + } + + template + void print(StreamType &stream, std::string const message = "Printing Sparse problem") { + stream << "----------------" << message << "--------------" << '\n'; + stream << "(m,n) = " << equations() << " , " << dimension() + << " nnz= " << Asp.nonZeros() << "\n"; + if (equations() > 20 || dimension() > 20) { + stream << "too big for complete visulization\n"; + return; + } + stream << "A=\n"; + + stream << MT(Asp); + stream << "\n"; + + stream << "b=\n"; + stream << b; + stream << "\n"; + + stream << "lb=\n"; + stream << barrier.lb; + stream << "\n"; + + stream << "ub=\n"; + stream << barrier.ub; + stream << "\n"; + + stream << "T=\n"; + stream << MT(T); + stream << "\n"; + + stream << "y=\n"; + stream << y; + stream << "\n"; + + stream << "center=\n"; + stream << center; + stream << "\n"; + } + + void make_format(Input const &input, MT const &S) { + nP = input.Aeq.cols(); + int nIneq = input.Aineq.rows(); + int nEq = input.Aeq.rows(); + A.resize(nEq + nIneq, nP + nIneq); + A << input.Aeq, MT::Zero(nEq, nIneq), input.Aineq, MT::Identity(nIneq, nIneq); + b.resize(nEq + nIneq, 1); + b << input.beq, input.bineq; + lb.resize(nP + nIneq, 1); + ub.resize(nP + nIneq, 1); + lb << input.lb, MT::Zero(nIneq, 1); + ub << input.ub, MT::Ones(nIneq, 1) * inf; + Asp.resize(nEq + nIneq, nP + nIneq); + int n = dimension(); + /*Move lb=ub to Ax=b*/ + for (int i = 0; i < n; i++) { + if (doubleVectorEqualityComparison(lb(i), ub(i))) { + MT temp = MT::Zero(1, n); + temp(i) = 1; + A.conservativeResize(A.rows() + 1, A.cols()); + A.row(A.rows() - 1) = temp; + b.conservativeResize(b.rows() + 1); + b(b.rows() - 1) = (lb(i) + ub(i)) / 2; + lb(i) = -inf; + ub(i) = inf; + } + } + Asp = A.sparseView(); + } + void make_format(Input const &input, SpMat const &S) { + nP = input.Aeq.cols(); + int nIneq = input.Aineq.rows(); + int nEq = input.Aeq.rows(); + Asp.resize(nEq + nIneq, nP + nIneq); + SpMat B = SpMat(input.Aeq); + SpMat C = SpMat(input.Aineq); + B.conservativeResize(nEq, nIneq + nP); + SpMat temp = SpMat(nIneq, nIneq); + temp.setIdentity(); + sparse_stack_h_inplace(C, temp); + sparse_stack_v(B, C, Asp); + b.resize(nEq + nIneq, 1); + b << input.beq, input.bineq; + lb.resize(nP + nIneq, 1); + ub.resize(nP + nIneq, 1); + lb << input.lb, MT::Zero(nIneq, 1); + ub << input.ub, MT::Ones(nIneq, 1) * inf; + int n = dimension(); + /*Move lb=ub to Ax=b*/ + for (int i = 0; i < n; i++) { + if (doubleVectorEqualityComparison(lb(i), ub(i))) { + B.resize(Asp.rows(), Asp.cols()); + B = SpMat(Asp); + MT temp = MT::Zero(1, n); + temp(i) = 1; + C = temp.sparseView(); + sparse_stack_v(B, C, Asp); + b.conservativeResize(b.rows() + 1); + b(b.rows() - 1) = (lb(i) + ub(i)) / 2; + lb(i) = -inf; + ub(i) = inf; + } + } + Asp.makeCompressed(); + } +//Class constructor + crhmc_problem(Input const &input, Opts _options = Opts()) + : options(_options), func(input.f), df(input.df), ddf(input.ddf), + fZero(input.fZero), fHandle(input.fHandle), dfHandle(input.dfHandle), + ddfHandle(input.ddfHandle) { +#ifdef TIME_KEEPING + rescale_duration = sparsify_duration = reordering_duration = + rm_rows_duration = rm_fixed_vars_duration = ex_collapsed_vars_duration = + shift_barrier_duration = lewis_center_duration = + std::chrono::duration::zero(); +#endif + + make_format(input, input.Aeq); + PreproccessProblem(); + } + // Initialization funciton + void PreproccessProblem() { + int n = dimension(); + barrier.set_bound(lb.cwiseMax(-barrier_bound), ub.cwiseMin(barrier_bound)); + NT tol = std::numeric_limits::epsilon(); + Asp.prune(tol, tol); + /*Update the transformation Tx + y*/ + T = SpMat(nP, n); + std::vector indices; + for (int i = 0; i < nP; i++) { + indices.push_back(Triple(i, i, 1)); + } + T.setFromTriplets(indices.begin(), indices.end()); + Tidx = std::vector(T.rows()); + updateT(); + y = VT::Zero(nP, 1); + /*Simplify*/ + if (!fZero) { + fZero = true; + simplify(); + fZero = false; + } + simplify(); +#ifdef TIME_KEEPING + std::chrono::time_point start, end; + start = std::chrono::system_clock::now(); +#endif + if (isempty_center) { + std::tie(center, std::ignore, std::ignore) = + analytic_center(Asp, b, *this, options); + isempty_center = false; + } + shift_barrier(center); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + shift_barrier_duration += end - start; +#endif + reorder(); + + width = estimate_width(true); + // Recenter again and make sure it is feasible + VT hess; +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + std::tie(center, std::ignore, std::ignore, w_center) = + lewis_center(Asp, b, *this, options, center); + std::tie(std::ignore, hess) = lewis_center_oracle(center, w_center); + CholObj solver = CholObj(transform_format(Asp)); + solver.accuracyThreshold = 0; + VT Hinv = hess.cwiseInverse(); + solver.decompose((Tx *)Hinv.data()); + VT out(equations(), 1); + VT input = (b - Asp * center); + solver.solve((Tx *)input.data(), (Tx *)out.data()); + center = center + (Asp.transpose() * out).cwiseProduct(Hinv); +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + lewis_center_duration += end - start; +#endif + if ((center.array() > barrier.ub.array()).any() || + (center.array() < barrier.lb.array()).any()) { + terminate = true; + terminate_message = "Polytope:Infeasible. The algorithm cannot find a feasible point.\n"; + return; + } + } +#ifdef TIME_KEEPING +template +void print_preparation_time(StreamType& stream){ + stream << "---Preparation Timing Information"<< std::endl; + stream << "Rescale completed in time, "; + stream << rescale_duration.count() << " secs " << std::endl; + stream << "Split dense columns completed in time, "; + stream << sparsify_duration.count() << " secs " << std::endl; + stream << "Reordering completed in time, "; + stream << reordering_duration.count() << " secs " << std::endl; + stream << "Removing dependent rows completed in time, "; + stream << rm_rows_duration.count() << " secs " << std::endl; + stream << "Removing fixed variables completed in time, "; + stream << rm_fixed_vars_duration.count() << " secs " << std::endl; + stream << "Extracting collapsed variables completed in time, "; + stream << ex_collapsed_vars_duration.count() << " secs " << std::endl; + stream << "Shift_barrier completed in time, "; + stream << shift_barrier_duration.count() << " secs " << std::endl; + stream << "Finding Center completed in time, "; + stream << lewis_center_duration.count() << " secs " << std::endl; +} +#endif + // Gradient and hessian of for the analytic center + std::pair analytic_center_oracle(VT const &x) { + MT g, h; + std::tie(std::ignore, g, h) = f_oracle(x); + return std::make_pair(g + barrier.gradient(x), h + barrier.hessian(x)); + } + // Gradient and hessian of for the lewis center + std::pair lewis_center_oracle(VT const &x, VT const &w) { + MT g, h; + std::tie(std::ignore, g, h) = f_oracle(x); + return std::make_pair(g + w.cwiseProduct(barrier.gradient(x)), + h + w.cwiseProduct(barrier.hessian(x))); + } + // Function that uses the transformation (T,y) to apply the function to the + // original variables + std::tuple f_oracle(MT const &x) { + int n = x.rows(); + int m=x.cols(); + VT f=VT(m); + MT g=MT(n,m); + MT h=MT(n,m); + if (fZero) { + f = VT::Zero(m); + g = MT::Zero(n,m); + h = MT::Zero(n,m); + return std::make_tuple(f, g, h); + } + // Take the correpsonding point in the original space + MT z = MT::Zero(y.rows(), m); + if (fHandle || dfHandle || ddfHandle) { + for(int k=0;k +#include "PackedCSparse/SparseMatrix.h" +#include +#include + +template +struct lambda_as_visitor_wrapper : Func +{ + lambda_as_visitor_wrapper(const Func &f) : Func(f) {} + template + void init(const S &v, I i, I j) + { + return Func::operator()(v, i, j); + } +}; + +template +void visit_lambda(const Mat &m, const Func &f) +{ + lambda_as_visitor_wrapper visitor(f); + m.visit(visitor); +} + +template +void sparse_stack_v(const SparseMatrixType &top, const SparseMatrixType &bottom, + SparseMatrixType &stacked) +{ + assert(top.cols() == bottom.cols()); + stacked.resize(top.rows() + bottom.rows(), top.cols()); + stacked.resizeNonZeros(top.nonZeros() + bottom.nonZeros()); + + int i = 0; + + for (int col = 0; col < top.cols(); col++) + { + stacked.outerIndexPtr()[col] = i; + + for (int j = top.outerIndexPtr()[col]; j < top.outerIndexPtr()[col + 1]; + j++, i++) + { + stacked.innerIndexPtr()[i] = top.innerIndexPtr()[j]; + stacked.valuePtr()[i] = top.valuePtr()[j]; + } + + for (int j = bottom.outerIndexPtr()[col]; + j < bottom.outerIndexPtr()[col + 1]; j++, i++) + { + stacked.innerIndexPtr()[i] = (int)top.rows() + bottom.innerIndexPtr()[j]; + stacked.valuePtr()[i] = bottom.valuePtr()[j]; + } + } + stacked.outerIndexPtr()[top.cols()] = i; +} + +template +void sparse_stack_h(const SparseMatrixType &left, const SparseMatrixType &right, + SparseMatrixType &stacked) +{ + assert(left.rows() == right.rows()); + + stacked.resize(left.rows(), left.cols() + right.cols()); + stacked.resizeNonZeros(left.nonZeros() + right.nonZeros()); + + std::copy(left.innerIndexPtr(), left.innerIndexPtr() + left.nonZeros(), + stacked.innerIndexPtr()); + std::copy(right.innerIndexPtr(), right.innerIndexPtr() + right.nonZeros(), + stacked.innerIndexPtr() + left.nonZeros()); + + std::copy(left.valuePtr(), left.valuePtr() + left.nonZeros(), + stacked.valuePtr()); + std::copy(right.valuePtr(), right.valuePtr() + right.nonZeros(), + stacked.valuePtr() + left.nonZeros()); + + std::copy(left.outerIndexPtr(), left.outerIndexPtr() + left.cols(), + stacked.outerIndexPtr()); // dont need the last entry of + // A.outerIndexPtr() -- total length is + // AB.cols() + 1 = A.cols() + B.cols() + 1 + std::transform(right.outerIndexPtr(), + right.outerIndexPtr() + right.cols() + 1, + stacked.outerIndexPtr() + left.cols(), + [&](int i) + { return i + left.nonZeros(); }); +} +#include + +template +void sparse_stack_h_inplace(SparseMatrixType &left, + const SparseMatrixType &right) +{ + assert(left.rows() == right.rows()); + + const int leftcol = (int)left.cols(); + const int leftnz = (int)left.nonZeros(); + + left.conservativeResize(left.rows(), left.cols() + right.cols()); + left.resizeNonZeros(left.nonZeros() + right.nonZeros()); + + std::copy(right.innerIndexPtr(), right.innerIndexPtr() + right.nonZeros(), + left.innerIndexPtr() + leftnz); + std::copy(right.valuePtr(), right.valuePtr() + right.nonZeros(), + left.valuePtr() + leftnz); + std::transform( + right.outerIndexPtr(), right.outerIndexPtr() + right.cols() + 1, + left.outerIndexPtr() + leftcol, [&](int i) + { return i + leftnz; }); +} + +template +void remove_zero_rows(SparseMatrixType &A, VectorType &b) { + std::vector> tripletList; + unsigned Ndata = A.cols(); + unsigned Nbins = A.rows(); + for (int k = 0; k < A.outerSize(); ++k) { + for (typename SparseMatrixType::InnerIterator it(A, k); it; ++it) { + tripletList.push_back( + Eigen::Triplet(it.row(), it.col(), it.value())); + } + } + // get which rows are empty + std::vector has_value(Nbins, false); + for (auto tr : tripletList) + has_value[tr.row()] = true; + if (std::all_of(has_value.begin(), has_value.end(), + [](bool v) { return v; })) { + return; + } + // create map from old to new indices + std::map row_map; + unsigned new_idx = 0; + for (unsigned old_idx = 0; old_idx < Nbins; old_idx++) + if (has_value[old_idx]) { + row_map[old_idx] = new_idx; + b(new_idx) = b(old_idx); + new_idx++; + } + // make new triplet list, dropping empty rows + std::vector> newTripletList; + newTripletList.reserve(Ndata); + for (auto tr : tripletList) + newTripletList.push_back( + Eigen::Triplet(row_map[tr.row()], tr.col(), tr.value())); + + // form new matrix and return + A.resize(new_idx, Ndata); + A.setFromTriplets(newTripletList.begin(), newTripletList.end()); + b.conservativeResize(new_idx); +} + +template +void remove_rows(SparseMatrixType &A, std::vector ¬Removed) { + unsigned Ndata = A.cols(); + unsigned Nbins = A.rows(); + // create map from old to new indices + std::map row_map; + unsigned new_idx = 0; + for (unsigned old_idx = 0; old_idx < Nbins; old_idx++) + if (notRemoved[old_idx]) + row_map[old_idx] = new_idx++; + + std::vector> tripletList; + for (int k = 0; k < A.outerSize(); ++k) { + for (typename SparseMatrixType::InnerIterator it(A, k); it; ++it) { + if (notRemoved[it.row()]) { + tripletList.push_back( + Eigen::Triplet(row_map[it.row()], it.col(), it.value())); + } + } + } + // form new matrix and return + A.resize(new_idx, Ndata); + A.setFromTriplets(tripletList.begin(), tripletList.end()); +} + +template +std::pair colwiseMinMax(SparseMatrixType const &A) +{ + int n = A.cols(); + VectorType cmax(n); + VectorType cmin(n); + for (int k = 0; k < A.outerSize(); ++k) + { + Type minv = +std::numeric_limits::max(); + Type maxv = std::numeric_limits::lowest(); + for (typename SparseMatrixType::InnerIterator it(A, k); it; ++it) + { + minv = std::min(minv, it.value()); + maxv = std::max(maxv, it.value()); + } + cmin(k) = minv; + cmax(k) = maxv; + } + return std::make_pair(cmin, cmax); +} +template +void nextpow2(VectorType &a) +{ + a = (a.array() == 0).select(1, a); + a = (((a.array().log()) / std::log(2)).ceil()).matrix(); + a = pow(2, a.array()).matrix(); +} +template +std::pair gmscale(SparseMatrixType &Asp, + const Type scltol) +{ + using Diagonal_MT = Eigen::DiagonalMatrix; + int m = Asp.rows(); + int n = Asp.cols(); + SparseMatrixType A = Asp.cwiseAbs(); + A.makeCompressed(); + int maxpass = 10; + Type aratio = 1e+50; + Type sratio; + Type damp = 1e-4; + Type small = 1e-8; + VectorType rscale = VectorType ::Ones(m, 1); + VectorType cscale = VectorType ::Ones(n, 1); + VectorType cmax; + VectorType cmin; + VectorType rmax; + VectorType rmin; + VectorType eps = VectorType ::Ones(n, 1) * 1e-12; + SparseMatrixType SA; + for (int npass = 0; npass < maxpass; npass++) + { + + rscale = (rscale.array() == 0).select(1, rscale); + Diagonal_MT Rinv = (rscale.cwiseInverse()).asDiagonal(); + SA = Rinv * A; + std::tie(cmin, cmax) = + colwiseMinMax(SA); + + // cmin = (cmin + eps).cwiseInverse(); + sratio = (cmax.cwiseQuotient(cmin)).maxCoeff(); + + if (npass > 0) + { + cscale = ((cmin.cwiseMax(damp * cmax)).cwiseProduct(cmax)).cwiseSqrt(); + } + + if (npass >= 2 && sratio >= aratio * scltol) + { + break; + } + aratio = sratio; + nextpow2(cscale); + Diagonal_MT Cinv = (cscale.cwiseInverse()).asDiagonal(); + SA = A * Cinv; + std::tie(rmin, rmax) = + colwiseMinMax(SA.transpose()); + // rmin = (rmin + eps).cwiseInverse(); + rscale = ((rmin.cwiseMax(damp * rmax)).cwiseProduct(rmax)).cwiseSqrt(); + nextpow2(rscale); + } + rscale = (rscale.array() == 0).select(1, rscale); + Diagonal_MT Rinv = (rscale.cwiseInverse()).asDiagonal(); + SA = Rinv * A; + std::tie(std::ignore, cscale) = + colwiseMinMax(SA); + nextpow2(cscale); + return std::make_pair(cscale, rscale); +} +template +int doubleVectorEqualityComparison( + const Type a, const Type b, + const Type tol = std::numeric_limits::epsilon()) +{ + return (abs(a - b) < tol * (1 + abs(a) + abs(b))); +} + +template +std::pair, std::vector> +nnzPerColumn(SparseMatrixType const &A, const int threashold) +{ + int n = A.cols(); + std::vector colCounts(n); + std::vector badCols; + for (int k = 0; k < A.outerSize(); ++k) + { + int nnz = 0; + for (typename SparseMatrixType::InnerIterator it(A, k); it; ++it) + { + if (it.value() != 0) + { + nnz++; + } + } + colCounts[k] = nnz; + if (nnz > threashold) + { + badCols.push_back(k); + } + } + return std::make_pair(colCounts, badCols); +} +using PM = Eigen::PermutationMatrix; +template +PM permuteMatAMD(SparseMatrixType const &A) +{ + Eigen::AMDOrdering ordering; + PM perm; + ordering(A, perm); + return perm; +} +template +PM postOrderPerm(SparseMatrixType const &A) +{ + using IndexVector = Eigen::Matrix; + int n = A.rows(); + IndexVector m_etree; + IndexVector firstRowElt; + Eigen::internal::coletree(A, m_etree, firstRowElt); + IndexVector post; + Eigen::internal::treePostorder(int(A.cols()), m_etree, post); + PM post_perm(n); + for (int i = 0; i < n; i++) + post_perm.indices()(i) = post(i); + return post_perm; +} + +template +void fillin_reduce(SparseMatrixType &X,VectorType& b){ + SparseMatrixType I = SparseMatrixType(Eigen::VectorXd::Ones(X.rows()).asDiagonal()); + SparseMatrixType XX = X * X.transpose() + I; + XX.makeCompressed(); + Eigen::SimplicialLDLT> cholesky; + cholesky.analyzePattern(XX); + X = cholesky.permutationP() * X; + b = cholesky.permutationP() *b; +} +template +PackedCSparse::SparseMatrix transform_format(SparseMatrixType const &mat) { + PackedCSparse::SparseMatrix A = PackedCSparse::SparseMatrix(mat.rows(), mat.cols(), mat.nonZeros()); + IndexType nnz = 0; + for (IndexType outeindex = 0; outeindex < mat.outerSize(); ++outeindex) { + A.p[outeindex] = nnz; + for (typename SparseMatrixType::InnerIterator it(mat, outeindex); it; ++it) { + A.i[nnz] = it.row(); + A.x[nnz] = it.value(); + nnz++; + } + } + A.p[A.n] = nnz; + return A; +} +template +void copy_indicies(MatrixType& a, MatrixType& b, std::vectorconst & a_idx, std::vectorconst & b_idx){ +for(int i=0;i +void copy_indicies(MatrixType& a, MatrixType b, std::vectorconst & b_idx){ +for(int i=0;i +void set(MatrixType &a, std::vectorconst & idx, const Type c){ + for(int i=0;i +void saxpy(MatrixType &a,MatrixType const &b,MatrixType const& c, std::vectorconst & a_idx, std::vectorconst & b_idx){ +for(int i=0;i +void load_problem(SpMat &A, VT &b, VT &lb, VT &ub, int &dimension, + std::string problem_name) { + { + std::string fileName(problem_name); + fileName.append(".mm"); + SpMat X; + loadMarket(X, fileName); + int m = X.rows(); + dimension = X.cols() - 1; + A = X.leftCols(dimension); + b = VT(X.col(dimension)); + } + { + std::string fileName(problem_name); + fileName.append("_bounds.mm"); + SpMat bounds; + loadMarket(bounds, fileName); + lb = VT(bounds.col(0)); + ub = VT(bounds.col(1)); + } +} +#endif diff --git a/src/volesti/include/preprocess/crhmc/lewis_center.h b/src/volesti/include/preprocess/crhmc/lewis_center.h new file mode 100644 index 00000000..5bf89d15 --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/lewis_center.h @@ -0,0 +1,188 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef LEWIS_CENTER_H +#define LEWIS_CENTER_H +#include "Eigen/Eigen" +#include "PackedCSparse/PackedChol.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include "preprocess/crhmc/opts.h" +#include "preprocess/crhmc/two_sided_barrier.h" +#include +#include +#include +#ifndef SIMD_LEN +#define SIMD_LEN 0 +#endif +const size_t chol_k3 = (SIMD_LEN == 0) ? 1 : SIMD_LEN; + +/*This function computes the Lewis center of the polytope*/ +//And detects additional constraint that need to be added +// x - It outputs the minimizer of min f(x) subjects to {Ax=b} +// w - Output weights that correspond to the Lewis center, they are gone be used in the sampler to reduce the conditon number +// C - detected constraint matrix +// If the domain ({Ax=b} intersect dom(f)) is not full dimensional in {Ax=b} +// because of the dom(f), the algorithm will detect the collapsed dimension +// and output the detected constraint C x = d +// d - detected constraint vector +template +std::tuple lewis_center(SpMat const &A, VT const &b, Polytope &f, Opts const &options, VT x = VT::Zero(0, 1)) +{ + using CholObj = typename Polytope::CholObj; + using Triple = typename Polytope::Triple; + using Tx = typename Polytope::Tx; + NT epsilon = 1e-8; + // initial conditions + int n = A.cols(); + int m = A.rows(); + //If it is given use starting point + if (x.rows() == 0 || !f.barrier.feasible(x)) + { + x = f.barrier.center; + } + VT lambda = VT::Zero(n, 1); + int fullStep = 0; + NT tConst = 0; + NT primalErr = std::numeric_limits::max(); + NT dualErr = std::numeric_limits::max(); + NT primalErrMin = std::numeric_limits::max(); + NT primalFactor = 1; + NT dualFactor = 1 + b.norm(); + std::vector idx; + + CholObj solver = CholObj(transform_format(A)); + VT w = VT::Ones(n, 1); + VT wp = w; + for (int iter = 0; iter < options.ipmMaxIter; iter++) + { + std::pair pair_analytic_oracle = f.lewis_center_oracle(x, wp); + VT grad = pair_analytic_oracle.first; + VT hess = pair_analytic_oracle.second; + + // compute the residual + VT rx = lambda - grad; + VT rs = b - A * x; + + // check stagnation + primalErrMin = std::min(primalErr, primalErrMin); + primalErr = rx.norm() / primalFactor; + NT dualErrLast = dualErr; + dualErr = rs.norm() / dualFactor; + bool feasible = f.barrier.feasible(x); + if ((dualErr > (1 - 0.9 * tConst) * dualErrLast) || + (primalErr > 10 * primalErrMin) || !feasible) + { + VT dist = f.barrier.boundary_distance(x); + NT th = options.ipmDistanceTol; + visit_lambda(dist, [&idx, th](double v, int i, int j) + { + if (v < th) + idx.push_back(i); }); + + if (idx.size() > 0) + { + break; + } + } + + // compute the step direction + VT Hinv = hess.cwiseInverse(); + solver.decompose((Tx *)Hinv.data()); + VT out(m, 1); + solver.solve((Tx *)rs.data(), (Tx *)out.data()); + VT dr1 = A.transpose() * out; + VT in = A * Hinv.cwiseProduct(rx); + solver.solve((Tx *)in.data(), (Tx *)out.data()); + + VT dr2 = A.transpose() * out; + VT dx1 = Hinv.cwiseProduct(dr1); + VT dx2 = Hinv.cwiseProduct(rx - dr2); + + // compute the step size + VT dx = dx1 + dx2; + NT tGrad = std::min(f.barrier.step_size(x, dx), 1.0); + dx = dx1 + tGrad * dx2; + NT tConst = std::min(0.99 * f.barrier.step_size(x, dx), 1.0); + tGrad = tGrad * tConst; + + // make the step + x = x + tConst * dx; + lambda = lambda - dr2; + + // update weight + VT w_vector(n, 1); + solver.leverageScoreComplement((Tx *)w_vector.data()); + + VT wNew = w_vector.cwiseMax(0) + VT::Ones(n, 1) * epsilon; + w = (w + wNew) / 2; + wp = Eigen::pow(w.array(), 0.875).matrix(); + + if (!f.barrier.feasible(x)) + { + break; + } + + // stop if converged + if (tGrad == 1) + { + fullStep = fullStep + 1; + if (fullStep > log(dualErr / options.ipmDualTol) && + fullStep > options.min_convergence_steps) + { + break; + } + } + else + { + fullStep = 0; + } + } + + SpMat C; + VT d; + if (idx.size() == 0) + { + VT dist = f.barrier.boundary_distance(x); + NT th = options.ipmDistanceTol; + visit_lambda(dist, [&idx, th](double v, int i, int j) + { + if (v < th) + idx.push_back(i); }); + } + + if (idx.size() > 0) + { + C.resize(idx.size(), n); + std::pair pboundary = f.barrier.boundary(x); + VT A_ = pboundary.first; + VT b_ = pboundary.second; + std::vector sparseIdx; + for (int i = 0; i < idx.size(); i++) + { + sparseIdx.push_back(Triple(i, idx[i], A_(idx[i]))); + } + C.setFromTriplets(sparseIdx.begin(), sparseIdx.end()); + d.resize(idx.size(), 1); + copy_indicies(d, b_, idx); + } + else + { + C = MT::Zero(0, n).sparseView(); + d = VT::Zero(0, 1); + } + + return std::make_tuple(x, C, d, wp); +} +#endif diff --git a/src/volesti/include/preprocess/crhmc/opts.h b/src/volesti/include/preprocess/crhmc/opts.h new file mode 100644 index 00000000..573a69db --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/opts.h @@ -0,0 +1,62 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" + +#ifndef OPTS_H +#define OPTS_H + +/// @brief Crhmc options +/// @tparam Type Numer type +template class opts { +public: + /*Preprocess options*/ + const int ipmMaxIter = 200; //Maximum number of iterations for finding the analytic and lewis center + const Type ipmDistanceTol = 1e-8; + const Type ipmDualTol = 1e-12; + int maxNZ = 30; + Type max_coord = 1e9; + bool EnableReordering = true; + const int min_convergence_steps=8; + + /*ODE options*/ + const Type implicitTol = 1e-5; + const int maxODEStep = 30; + Type initialStep = 0.2; + Type convergence_bound = 1e16; + + /*PackedCS Solver Options*/ + Type solver_accuracy_threshold=1e-2; + int simdLen=1; + + /*Sampler options*/ + bool DynamicWeight = true; //Enable the use of dynamic weights for each variable when sampling + bool DynamicStepSize = true; // Enable adaptive step size that avoids low acceptance probability + bool DynamicRegularizer = true; //Enable the addition of a regularization term + Type regularization_factor=1e-20; + /*Dynamic step choices*/ + Type warmUpStep = 10; + int maxConsecutiveBadStep = 10; + Type targetODEStep = 10; + Type shrinkFactor = 1.1; + Type minStepSize = 1e-5; + Type effectiveStepSize = 1; + + opts() {} + void operator=(const opts &rhs) { + EnableReordering = rhs.EnableReordering; + maxNZ = rhs.maxNZ; + } +}; +#endif diff --git a/src/volesti/include/preprocess/crhmc/two_sided_barrier.h b/src/volesti/include/preprocess/crhmc/two_sided_barrier.h new file mode 100644 index 00000000..7e279ab7 --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/two_sided_barrier.h @@ -0,0 +1,173 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" + +// The log barrier for the domain {lu <= x <= ub}: +// phi(x) = - sum log(x - lb) - sum log(ub - x). +#ifndef TWO_SIDED_BARIER_H +#define TWO_SIDED_BARIER_H + +#include "Eigen/Eigen" +#include "cartesian_geom/cartesian_kernel.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include + +/// @brief A two sided barrier used by crhmc sampler +/// @tparam Point Point Type +template class two_sided_barrier { + + using NT = typename Point::FT; + using MT = Eigen::Matrix; + using VT = Eigen::Matrix; + +public: + VT lb; + VT ub; + int vdim; + int n; + std::vector upperIdx; + std::vector lowerIdx; + std::vector freeIdx; + VT center; + const NT max_step = 1e16; // largest step size + const NT regularization_constant = 1e-20; // small regularization to not have a large inverse + const NT unbounded_center_coord = 1e6; + MT extraHessian; + + const NT inf = std::numeric_limits::infinity(); + + void set_bound(VT const &_lb, VT const &_ub) { + n = _lb.rows(); + extraHessian.resize(n, 1); + lb.resize(n); + ub.resize(n); + lb = _lb; + ub = _ub; + extraHessian = regularization_constant * MT::Ones(n, 1); + int x1 = 0, x2 = 0, x3 = 0; + for (int i = 0; i < n; i++) { + if (lb(i) == -inf) { + upperIdx.push_back(i); + x1++; + } + if (ub(i) == inf) { + lowerIdx.push_back(i); + x2++; + } + if (ub(i) == inf && lb(i) == -inf) { + freeIdx.push_back(i); + } + } + + VT c = (ub + lb) / 2; + VT bias1=VT::Ones(x2, 1) * unbounded_center_coord; + saxpy(c,lb,bias1,lowerIdx,lowerIdx); + VT bias2=-VT::Ones(x1, 1) * unbounded_center_coord; + saxpy(c,ub,bias2,upperIdx,upperIdx); + set(c, freeIdx, 0.0); + + center = c; + } + two_sided_barrier(VT const &_lb, VT const &_ub, int _vdim = 1) { + set_bound(_lb, _ub); + vdim = _vdim; + extraHessian = regularization_constant * MT::Ones(n,1); + } + two_sided_barrier() { vdim = 1; } + + VT gradient(VT const &x) { + return (ub - x).cwiseInverse() - (x - lb).cwiseInverse(); + } + + VT hessian(VT const &x) { + VT d = ((ub - x).cwiseProduct((ub - x))).cwiseInverse() + + ((x - lb).cwiseProduct((x - lb))).cwiseInverse(); + return d + extraHessian; + } + MT hessian(MT const &x){ + MT d = (((- x).colwise()+ub).cwiseProduct(((- x).colwise()+ub))).cwiseInverse() + + ((x.colwise() - lb).cwiseProduct((x.colwise() - lb))).cwiseInverse(); + return d + extraHessian; + } + MT tensor(MT const &x) { + MT d = 2 * ((((-x).colwise()+ub).cwiseProduct(((-x).colwise()+ub))).cwiseProduct(((-x).colwise()+ub))).cwiseInverse() - + 2 * (((x.colwise() - lb).cwiseProduct(( x.colwise() - lb))).cwiseProduct(( x.colwise() - lb))).cwiseInverse(); + return d; + } + MT quadratic_form_gradient(MT const &x, MT const &u) { + // Output the -grad of u' (hess phi(x)) u. + return (u.cwiseProduct(u)).cwiseProduct(tensor(x)); + } + VT quadratic_form_gradient(VT const &x, VT const &u) { + // Output the -grad of u' (hess phi(x)) u. + + return (u.cwiseProduct(u)).cwiseProduct(tensor(x)); + } + NT step_size(VT const &x, VT const &v) { + // Output the maximum step size from x with direction v. + + // check positive direction + VT temp = (v.array() > 0).select((ub - x).cwiseQuotient(v), max_step); + NT t1 = temp.minCoeff(); + + // check negative direction + temp = (v.array() < 0).select((lb - x).cwiseQuotient(v), max_step); + NT t2 = temp.minCoeff(); + + return std::min(t1, t2); + } + VT boundary_distance(VT const &x) { + // Output the distance of x with its closest boundary for each + // coordinate + return ((x - lb).cwiseMin(ub - x)).cwiseAbs(); + } + + bool feasible(VT const &x) { + return (x.array() > lb.array() && x.array() < ub.array()).all(); + } + VT feasible(MT const &x) { + VT result=VT::Ones(x.cols()); + for(int i=0;i lb.array() && x.col(i).array() < ub.array()).all(); + } + return result; + } + + std::pair analytic_center_oracle(VT const &x) { + VT g = VT::Zero(n, 1); + VT h = VT::Zero(n, 1); + return std::make_pair(g + gradient(x), h + hessian(x)); + } + + std::pair lewis_center_oracle(VT const &x, VT const &w) { + VT g = VT::Zero(n, 1); + VT h = VT::Zero(n, 1); + return std::make_pair(g + w.cwiseProduct(gradient(x)),h + w.cwiseProduct(hessian(x))); + } + + std::pair boundary(VT const &x) { + // Output the normal at the boundary around x for each barrier. + // Assume: only 1 vector is given + + VT A = VT::Ones(x.rows(), 1); + VT b = ub; + + b = (x.array() < center.array()).select(-lb, b); + A = (x.array() < center.array()).select(-A, A); + + return std::make_pair(A, b); + } +}; +#endif diff --git a/src/volesti/include/preprocess/crhmc/weighted_two_sided_barrier.h b/src/volesti/include/preprocess/crhmc/weighted_two_sided_barrier.h new file mode 100644 index 00000000..09f5eb79 --- /dev/null +++ b/src/volesti/include/preprocess/crhmc/weighted_two_sided_barrier.h @@ -0,0 +1,166 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" + +// The log barrier for the domain {lu <= x <= ub}: +// phi(x) = - sum log(x - lb) - sum log(ub - x). +#ifndef WEIGHTED_TWO_SIDED_BARIER_H +#define WEIGHTED_TWO_SIDED_BARIER_H + +#include "Eigen/Eigen" +#include "cartesian_geom/cartesian_kernel.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include + +/// @brief A weighted two sided barrier used by crhmc sampler +/// @tparam Point Point Type +template class weighted_two_sided_barrier { + + using NT = typename Point::FT; + using MT = Eigen::Matrix; + using VT = Eigen::Matrix; + +public: + VT lb; + VT ub; + int vdim; + int n; + std::vector upperIdx; + std::vector lowerIdx; + std::vector freeIdx; + VT center; + const NT max_step = 1e16; // largest step size + const NT regularization_constant = 1e-20; // small regularization to not have a large inverse + const NT unbounded_center_coord = 1e6; + MT extraHessian; + const NT inf = std::numeric_limits::infinity(); + + VT w; + + weighted_two_sided_barrier(VT const &_lb, VT const &_ub, VT const &_w, + int _vdim = 1) { + set_bound(_lb, _ub); + w = _w; + vdim = _vdim; + extraHessian = regularization_constant * VT::Ones(n,1); + } + weighted_two_sided_barrier() { vdim = 1; } + + VT gradient(VT const &x) { + return w.cwiseQuotient(ub - x) - w.cwiseQuotient(x - lb); + } + + VT hessian(VT const &x) { + VT d = w.cwiseQuotient((ub - x).cwiseProduct((ub - x))) + + w.cwiseQuotient((x - lb).cwiseProduct((x - lb))); + return d + extraHessian; + } + MT hessian(MT const &x){ + MT d = (((- x).colwise()+ub).cwiseProduct(((- x).colwise()+ub))).cwiseInverse() + + ((x.colwise() - lb).cwiseProduct((x.colwise() - lb))).cwiseInverse(); + return w.asDiagonal()*d + extraHessian; + } + VT tensor(VT const &x) { + VT d = 2 * w.cwiseQuotient(((ub - x).cwiseProduct((ub - x))).cwiseProduct((ub - x))) - + 2 * w.cwiseQuotient(((x - lb).cwiseProduct((x - lb))).cwiseProduct((x - lb))); + return d; + } + MT tensor(MT const &x) { + MT d = 2 * ((((-x).colwise()+ub).cwiseProduct(((-x).colwise()+ub))).cwiseProduct(((-x).colwise()+ub))).cwiseInverse() - + 2 * (((x.colwise() - lb).cwiseProduct(( x.colwise() - lb))).cwiseProduct(( x.colwise() - lb))).cwiseInverse(); + return w.asDiagonal()*d; + } + MT quadratic_form_gradient(MT const &x, MT const &u) { + // Output the -grad of u' (hess phi(x)) u. + return (u.cwiseProduct(u)).cwiseProduct(tensor(x)); + } + VT quadratic_form_gradient(VT const &x, VT const &u) { + // Output the -grad of u' (hess phi(x)) u. + + return (u.cwiseProduct(u)).cwiseProduct(tensor(x)); + } + NT step_size(VT const &x, VT const &v) { + // Output the maximum step size from x with direction v or -v. + + // check positive direction + VT temp = (v.array() > 0).select((ub - x).cwiseQuotient(v), max_step); + NT t1 = temp.minCoeff(); + + // check negative direction + temp = (v.array() < 0).select((lb - x).cwiseQuotient(v), max_step); + NT t2 = temp.minCoeff(); + + return std::min(t1, t2); + } + VT boundary_distance(VT const &x) { + // Output the distance of x with its closest boundary for each + // coordinate + return ((x - lb).cwiseMin(ub - x)).cwiseAbs(); + } + + bool feasible(VT const &x) { + return (x.array() > lb.array() && x.array() < ub.array()).all(); + } + VT feasible(MT const &x) { + VT result=VT::Ones(x.cols()); + for(int i=0;i lb.array() && x.col(i).array() < ub.array()).all(); + } + return result; + } + void set_bound(VT const &_lb, VT const &_ub) { + + lb = _lb; + ub = _ub; + n = lb.rows(); + extraHessian = regularization_constant * VT::Ones(n); + int x1 = 0, x2 = 0, x3 = 0; + for (int i = 0; i < n; i++) { + if (lb(i) == -inf) { + upperIdx.push_back(i); + x1++; + } + if (ub(i) == inf) { + lowerIdx.push_back(i); + x2++; + } + if (ub(i) == inf && lb(i) == -inf) { + freeIdx.push_back(i); + } + } + + VT c = (ub + lb) / 2; + VT bias1=VT::Ones(x2, 1) * unbounded_center_coord; + saxpy(c,lb,bias1,lowerIdx,lowerIdx); + VT bias2=-VT::Ones(x1, 1) * unbounded_center_coord; + saxpy(c,ub,bias2,upperIdx,upperIdx); + set(c, freeIdx, 0.0); + + center = c; + } + + std::pair boundary(VT const &x) { + // Output the normal at the boundary around x for each barrier. + // Assume: only 1 vector is given + VT A = VT::Ones(x.rows(), 1); + VT b = ub; + + b = (x.array() < center.array()).select(-lb, b); + A = (x.array() < center.array()).select(-A, A); + + return std::make_pair(A, b); + } +}; +#endif diff --git a/src/volesti/include/preprocess/estimate_L_smooth_parameter.hpp b/src/volesti/include/preprocess/estimate_L_smooth_parameter.hpp new file mode 100644 index 00000000..8f5a1e89 --- /dev/null +++ b/src/volesti/include/preprocess/estimate_L_smooth_parameter.hpp @@ -0,0 +1,73 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef ESTIMATE_L_SMOOTH_PARAMETER_HPP +#define ESTIMATE_L_SMOOTH_PARAMETER_HPP + +#include "random_walks/random_walks.hpp" + +template +< + typename WalkTypePolicy = AcceleratedBilliardWalk, + typename Polytope, + typename Point, + typename NegativeGradientFunctor, + typename RandomNumberGenerator +> +double estimate_L_smooth(Polytope &P, Point &p, unsigned int const& walk_length, + NegativeGradientFunctor F, RandomNumberGenerator &rng) +{ + typedef typename Point::FT NT; + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > RandomWalk; + + P.ComputeInnerBall(); + + unsigned int d = P.dimension(); + unsigned int rnum = 5 * d; + std::vector randPoints(1); + std::vector vecPoint1; + std::vector vecPoint2; + std::vector< std::vector > listOfPoints; + Point F1; + + RandomWalk walk(P, p, rng); + for (unsigned int i=0; i::lowest(), Ltemp; + + for (auto pit=listOfPoints.begin(); pit!=(listOfPoints.end()-1); ++pit) + { + F1 = F(1, *pit, 0); + + for (auto qit=(pit+1); qit!=listOfPoints.end(); ++qit) + { + vecPoint2 = *qit; + Ltemp = (F1 - F(1, *qit, 0)).length() / ((*pit)[0] - (*qit)[0]).length(); + + if (Ltemp > L) + { + L = Ltemp; + } + } + } + return L; +} + + +#endif diff --git a/src/volesti/include/preprocess/max_inscribed_ball.hpp b/src/volesti/include/preprocess/max_inscribed_ball.hpp new file mode 100644 index 00000000..6fe70e9a --- /dev/null +++ b/src/volesti/include/preprocess/max_inscribed_ball.hpp @@ -0,0 +1,216 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef MAX_INNER_BALL +#define MAX_INNER_BALL + +/* + This implmentation computes the largest inscribed ball in a given convex polytope P. + The polytope has to be given in H-representation P = {x | Ax <= b} and the rows of A + has to be normalized. It solves the Linear program: max t, s.t. Ax + t*e <= b, where + e is the vector of ones. + + The implementation is based on Yin Zhang's Matlab implementation in https://github.com/Bounciness/Volume-and-Sampling/blob/1c7adfb46c2c01037e625db76ff00e73616441d4/external/mve11/mve_cobra/mve_presolve_cobra.m + + Input: matrix A, vector b such that the polytope P = {x | Ax<=b} + tolerance parameter tol + + Output: center of the ball x + radius r +*/ + +template +void calcstep(MT const& A, MT const& A_trans, MT const& R, VT &s, + VT &y, VT &r1, VT const& r2, NT const& r3, VT &r4, + VT &dx, VT &ds, NT &dt, VT &dy, VT &tmp, VT &rhs) +{ + int m = A.rows(), n = A.cols(); + NT *vec_iter1 = tmp.data(), *vec_iter2 = y.data(), *vec_iter3 = s.data(), + *vec_iter4 = r1.data(), *vec_iter5 = r4.data(); + for (int i = 0; i < m; ++i) { + *vec_iter1 = ((*vec_iter4) / (*vec_iter2) - (*vec_iter5)) / (*vec_iter3); + vec_iter1++; vec_iter2++; vec_iter3++; vec_iter4++; vec_iter5++; + } + + rhs.block(0,0,n,1).noalias() = r2 + A_trans * tmp; + rhs(n) = r3 + tmp.sum(); + VT dxdt = R.colPivHouseholderQr().solve(R.transpose().colPivHouseholderQr().solve(rhs)); + + dx = dxdt.block(0,0,n,1); + dt = dxdt(n); + ds.noalias() = r1 - A*dx - VT::Ones(m) * dt; + vec_iter1 = dy.data(); vec_iter2 = r4.data(); vec_iter3 = y.data(); + vec_iter4 = ds.data(); vec_iter5 = s.data(); + + for (int i = 0; i < m; ++i) { + *vec_iter1 = ((*vec_iter2) - (*vec_iter3) * (*vec_iter4)) / (*vec_iter5); + vec_iter1++; vec_iter2++; vec_iter3++; vec_iter4++; vec_iter5++; + } +} + + +template +std::tuple max_inscribed_ball(MT const& A, VT const& b, unsigned int maxiter, NT tol) +{ + int m = A.rows(), n = A.cols(); + bool converge; + + NT bnrm = b.norm(); + VT o_m = VT::Zero(m), o_n = VT::Zero(n), e_m = VT::Ones(m); + + VT x = o_n, y = e_m / m; + NT t = b.minCoeff() - 1.0; + VT s = b - e_m * t; + + VT dx = o_n; + VT dxc = dx, ds = o_m; + VT dsc = ds, dy = o_m, mu_ds_dy(m), tmp(m), rhs(n + 1); + VT dyc = dy, r1(m), r2(n), r4(m), r23(n + 1), AtDe(n), d(m); + + NT dt = NT(0), dtc = NT(0), tau, sigma0 = 0.2, r3, gap, + prif, drif, rgap, total_err, alphap, alphad, + ratio, sigma, mu, t_prev = 1000.0 * t + 100.0; + + NT const tau0 = 0.995, power_num = 5.0 * std::pow(10.0, 15.0); + NT *vec_iter1, *vec_iter2, *vec_iter3, *vec_iter4; + + MT B(n + 1, n + 1), AtD(n, m), R(n + 1, n + 1), + eEye = std::pow(10.0, -14.0) * MT::Identity(n + 1, n + 1), + A_trans = A.transpose(); + + for (unsigned int i = 0; i < maxiter; ++i) { + + // KKT residuals + r1.noalias() = b - (A * x + s + t * e_m); + r2.noalias() = -A_trans * y; + r3 = 1.0 - y.sum(); + r4 = -s.cwiseProduct(y); + + r23.block(0, 0, n, 1) = r2; + r23(n) = r3; + gap = -r4.sum(); + + // relative residual norms and gap + prif = r1.norm() / (1.0 + bnrm); + drif = r23.norm() / 10.0; + rgap = std::abs(b.dot(y) - t) / (1.0 + std::abs(t)); + total_err = std::max(prif, drif); + total_err = std::max(total_err, rgap); + + // progress output & check stopping + if (total_err < tol || ( t > 0 && ( (std::abs(t - t_prev) <= tol * t && std::abs(t - t_prev) <= tol * t_prev) + || (t_prev >= (1.0 - tol) * t && i > 0) + || (t <= (1.0 - tol) * t_prev && i > 0) ) ) ) + { + //converged + converge = true; + break; + } + + if (dt > 1000.0 * bnrm || t > 1000000.0 * bnrm) + { + //unbounded + converge = false; + break; + } + + // Shur complement matrix + vec_iter1 = d.data(); + vec_iter3 = s.data(); + vec_iter2 = y.data(); + for (int j = 0; j < m; ++j) { + *vec_iter1 = std::min(power_num, (*vec_iter2) / (*vec_iter3)); + AtD.col(j).noalias() = A_trans.col(j) * (*vec_iter1); + vec_iter1++; + vec_iter3++; + vec_iter2++; + } + + AtDe.noalias() = AtD * e_m; + B.block(0, 0, n, n).noalias() = AtD * A; + B.block(0, n, n, 1).noalias() = AtDe; + B.block(n, 0, 1, n).noalias() = AtDe.transpose(); + B(n, n) = d.sum(); + B.noalias() += eEye; + + // Cholesky decomposition + Eigen::LLT lltOfB(B); + R = lltOfB.matrixL().transpose(); + + // predictor step & length + calcstep(A, A_trans, R, s, y, r1, r2, r3, r4, dx, ds, dt, dy, tmp, rhs); + + alphap = -1.0; + alphad = -1.0; + vec_iter1 = ds.data(); + vec_iter2 = s.data(); + vec_iter3 = dy.data(); + vec_iter4 = y.data(); + + for (int j = 0; j < m; ++j) { + alphap = std::min(alphap, (*vec_iter1) / (*vec_iter2)); + alphad = std::min(alphad, (*vec_iter3) / (*vec_iter4)); + vec_iter1++; + vec_iter2++; + vec_iter3++; + vec_iter4++; + } + alphap = -1.0 / alphap; + alphad = -1.0 / alphad; + + // determine mu + ratio = (s + alphap * ds).dot((y + alphad * dy)) / gap; + sigma = std::min(sigma0, ratio * ratio); + mu = (sigma * gap) / NT(m); + + // corrector and combined step & length + mu_ds_dy.noalias() = e_m * mu - ds.cwiseProduct(dy); + calcstep(A, A_trans, R, s, y, o_m, o_n, 0.0, mu_ds_dy, dxc, dsc, dtc, dyc, tmp, rhs); + + dx += dxc; + ds += dsc; + dt += dtc; + dy += dyc; + + alphap = -0.5; + alphad = -0.5; + vec_iter1 = ds.data(); + vec_iter2 = s.data(); + vec_iter3 = dy.data(); + vec_iter4 = y.data(); + + for (int j = 0; j < m; ++j) { + alphap = std::min(alphap, (*vec_iter1) / (*vec_iter2)); + alphad = std::min(alphad, (*vec_iter3) / (*vec_iter4)); + vec_iter1++; + vec_iter2++; + vec_iter3++; + vec_iter4++; + } + alphap = -1.0 / alphap; + alphad = -1.0 / alphad; + + // update iterates + tau = std::max(tau0, 1.0 - gap / NT(m)); + alphap = std::min(1.0, tau * alphap); + alphad = std::min(1.0, tau * alphad); + + x += alphap * dx; + s += alphap * ds; + t_prev = t; + t += alphap * dt; + y += alphad * dy; + } + + std::tuple result = std::make_tuple(x, t, converge); + return result; +} + +#endif diff --git a/src/volesti/include/preprocess/max_inscribed_ellipsoid.hpp b/src/volesti/include/preprocess/max_inscribed_ellipsoid.hpp new file mode 100644 index 00000000..ebdb261e --- /dev/null +++ b/src/volesti/include/preprocess/max_inscribed_ellipsoid.hpp @@ -0,0 +1,236 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. +//Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef MVE_COMPUTATION_HPP +#define MVE_COMPUTATION_HPP + +#include +#include + + +/* + Implementation of the interior point method to compute the largest inscribed ellipsoid in a + given convex polytope by "Yin Zhang, An Interior-Point Algorithm for the Maximum-Volume Ellipsoid + Problem (1999)". + + This C++ implementation is based on the Matlab implementation in https://github.com/Bounciness/Volume-and-Sampling/blob/1c7adfb46c2c01037e625db76ff00e73616441d4/external/mve11/mve_cobra/mve_solver_cobra.m + + The implmentation computes the largest inscribed ellipsoid {x | x = y + Es, ||s|| = 1} + + Input: matrix A, vector b such that the polytope P = {x | Ax<=b} + interior point x0 + tolerance parameters tol, reg + + Output: center of the ellipsoid y + matrix V = E_transpose * E +*/ + +// using Custom_MT as to deal with both dense and sparse matrices, MT will be the type of result matrix +template +std::pair, bool> max_inscribed_ellipsoid(Custom_MT A, VT b, VT const& x0, + unsigned int const& maxiter, + NT const& tol, NT const& reg) +{ + typedef Eigen::DiagonalMatrix Diagonal_MT; + + int m = A.rows(), n = A.cols(); + bool converged = false; + + NT bnrm = b.norm(), + last_r1 = std::numeric_limits::lowest(), + last_r2 = std::numeric_limits::lowest(), + prev_obj = std::numeric_limits::lowest(), + gap, rmu, res, objval, r1, r2 ,r3, rel, Rel, + astep, ax, ay, az, tau; + + NT const reg_lim = std::pow(10.0, -10.0), tau0 = 0.75, minmu = std::pow(10.0, -8.0); + + NT *vec_iter1, *vec_iter2, *vec_iter3; + + VT x = VT::Zero(n), y = VT::Ones(m), bmAx = VT::Ones(m), + h(m), z(m), yz(m), yh(m), R1(n), R2(m), R3(m), y2h(m), y2h_z(m), h_z(m), + R3Dy(m), R23(m), dx(n), Adx(m), dyDy(m), dy(m), dz(m); + + VT const bmAx0 = b - A * x0, ones_m = VT::Ones(m); + + MT Q(m, m), E2(n, n), YQ(m,m), G(m,m), T(m,n), ATP(n,m), ATP_A(n,n); + Diagonal_MT Y(m); + Custom_MT YA(m, n); + + A = (ones_m.cwiseProduct(bmAx0.cwiseInverse())).asDiagonal() * A, b = ones_m; + Custom_MT A_trans = A.transpose(); + + int i = 1; + while (i <= maxiter) { + + Y = y.asDiagonal(); + + E2.noalias() = MT(A_trans * Y * A).inverse(); + + Q.noalias() = A * E2 * A_trans; + h = Q.diagonal(); + h = h.cwiseSqrt(); + + if (i == 1) { + // perform those computations only during the first iteration + NT t = bmAx.cwiseProduct(h.cwiseInverse()).minCoeff(); + y *= (1.0 / (t * t)); + h *= t; + vec_iter1 = bmAx.data(); + vec_iter2 = h.data(); + vec_iter3 = z.data(); + for (int j = 0; j < m; ++j) { + *vec_iter3 = std::max(0.1, (*vec_iter1 - (*vec_iter2))); + vec_iter1++; + vec_iter2++; + vec_iter3++; + } + Q *= (t * t); + Y = Y * (1.0 / (t * t)); + } + + yz = y.cwiseProduct(z); + yh = y.cwiseProduct(h); + + gap = yz.sum() / NT(m); // compute the gap between primal and dual solution + rmu = std::min(0.5, gap) * gap; + rmu = std::max(rmu, minmu); + + R1.noalias() = - A_trans * yh; + R2 = bmAx - h - z; + R3.noalias() = rmu * ones_m - yz; + + r1 = R1.template lpNorm(); + r2 = R2.template lpNorm(); + r3 = R3.template lpNorm(); + + res = std::max(r1, r2); + res = std::max(res, r3); + objval = std::log(E2.determinant()) / 2.0; + + Eigen::SelfAdjointEigenSolver eigensolver(E2); // E2 is positive definite matrix + // computing eigenvalues of E2 + rel = eigensolver.eigenvalues().minCoeff(); + Rel = eigensolver.eigenvalues().maxCoeff(); + + if (i % 10 == 0) { + + if (std::abs((last_r1 - r1) / std::min(NT(std::abs(last_r1)), NT(std::abs(r1)))) < 0.01 && + std::abs((last_r2 - r2) / std::min(NT(abs(last_r2)), NT(std::abs(r2)))) < 0.01 && + Rel / rel > 100.0 && + reg > reg_lim) { + converged = false; + //Stopped making progress + break; + } + last_r2 = r2; + last_r1 = r1; + } + + // stopping criterion + if ((res < tol * (1.0 + bnrm) && rmu <= minmu) || + (i > 4 && prev_obj != std::numeric_limits::lowest() && + ((std::abs(objval - prev_obj) <= tol * objval && std::abs(objval - prev_obj) <= tol * prev_obj) || + (prev_obj >= (1.0 - tol) * objval || objval <= (1.0 - tol) * prev_obj) ) ) ) { + //converged + x += x0; + converged = true; + break; + } + + prev_obj = objval; // storing the objective value of the previous iteration + YQ.noalias() = Y * Q; + G = YQ.cwiseProduct(YQ.transpose()); + y2h = 2.0 * yh; + YA = Y * A; + + vec_iter1 = y2h.data(); + vec_iter2 = z.data(); + vec_iter3 = y2h_z.data(); + for (int j = 0; j < m; ++j) { + *vec_iter3 = std::max(reg, (*vec_iter1) * (*vec_iter2)); + vec_iter1++; + vec_iter2++; + vec_iter3++; + } + + G.diagonal() += y2h_z; + h_z = h + z; + + for (int j = 0; j < n; ++j) { + T.col(j) = G.colPivHouseholderQr().solve( VT(YA.col(j).cwiseProduct(h_z)) ); + } + ATP.noalias() = MT(y2h.asDiagonal()*T - YA).transpose(); + + vec_iter1 = R3.data(); + vec_iter2 = y.data(); + vec_iter3 = R3Dy.data(); + for (int j = 0; j < m; ++j) { + *vec_iter3 = (*vec_iter1) / (*vec_iter2); + vec_iter1++; + vec_iter2++; + vec_iter3++; + } + + R23 = R2 - R3Dy; + ATP_A.noalias() = ATP * A; + ATP_A.diagonal() += ones_m * reg; + dx = ATP_A.colPivHouseholderQr().solve(R1 + ATP * R23); // predictor step + + // corrector and combined step & length + Adx.noalias() = A * dx; + dyDy = G.colPivHouseholderQr().solve(y2h.cwiseProduct(Adx-R23)); + + dy = y.cwiseProduct(dyDy); + dz = R3Dy - z.cwiseProduct(dyDy); + + vec_iter1 = Adx.data(); + vec_iter2 = bmAx.data(); + ax = -0.5; + for (int j = 0; j < m; ++j) { + ax = std::min(ax, - (*vec_iter1) / (*vec_iter2)); + vec_iter1++; + vec_iter2++; + } + + ax = -1.0 / ax; + ay = -1.0 / std::min(dyDy.minCoeff(), -0.5); + + vec_iter1 = dz.data(); + vec_iter2 = z.data(); + az = -0.5; + for (int j = 0; j < m; ++j) { + az = std::min(az, (*vec_iter1) / (*vec_iter2)); + vec_iter1++; + vec_iter2++; + } + + az = -1.0 / az; + tau = std::max(tau0, 1.0 - res); + astep = tau * std::min(std::min(1.0, ax), std::min(ay, az)); // compute the step + + // update iterates + x += astep * dx; + y += astep * dy; + z += astep * dz; + + bmAx -= astep * Adx; + + i++; + } + + return std::pair, bool>(std::pair(E2, x), converged); + +} + + +#endif diff --git a/src/volesti/include/preprocess/max_inscribed_ellipsoid_rounding.hpp b/src/volesti/include/preprocess/max_inscribed_ellipsoid_rounding.hpp new file mode 100644 index 00000000..c13d53f7 --- /dev/null +++ b/src/volesti/include/preprocess/max_inscribed_ellipsoid_rounding.hpp @@ -0,0 +1,77 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef MAX_ELLIPSOID_ROUNDING_HPP +#define MAX_ELLIPSOID_ROUNDING_HPP + +#include "max_inscribed_ellipsoid.hpp" + +template +< + typename MT, + typename VT, + typename NT, + typename Polytope, + typename Point +> +std::tuple max_inscribed_ellipsoid_rounding(Polytope &P, + Point const& InnerPoint) +{ + std::pair, bool> iter_res; + iter_res.second = false; + + VT x0 = InnerPoint.getCoefficients(); + MT E, L; + unsigned int maxiter = 150, iter = 1, d = P.dimension(); + + NT R = 100.0, r = 1.0, tol = std::pow(10, -6.0), reg = std::pow(10, -4.0), round_val = 1.0; + + MT T = MT::Identity(d, d); + VT shift = VT::Zero(d); + + while (true) + { + // compute the largest inscribed ellipsoid in P centered at x0 + iter_res = max_inscribed_ellipsoid(P.get_mat(), P.get_vec(), x0, maxiter, tol, reg); + E = iter_res.first.first; + E = (E + E.transpose()) / 2.0; + E = E + MT::Identity(d, d)*std::pow(10, -8.0); //normalize E + + Eigen::LLT lltOfA(E); // compute the Cholesky decomposition of E + L = lltOfA.matrixL(); + + Eigen::SelfAdjointEigenSolver eigensolver(L); + r = eigensolver.eigenvalues().minCoeff(); + R = eigensolver.eigenvalues().maxCoeff(); + + // check the roundness of the polytope + if(((std::abs(R / r) <= 2.3 && iter_res.second) || iter >= 20) && iter>2){ + break; + } + + // shift polytope and apply the linear transformation on P + P.shift(iter_res.first.second); + shift += T * iter_res.first.second; + T = T * L; + round_val *= L.transpose().determinant(); + P.linear_transformIt(L); + + reg = std::max(reg / 10.0, std::pow(10, -10.0)); + P.normalize(); + x0 = VT::Zero(d); + + iter++; + } + + std::tuple result = std::make_tuple(T, shift, std::abs(round_val)); + return result; +} + +#endif diff --git a/src/volesti/include/preprocess/min_sampling_covering_ellipsoid_rounding.hpp b/src/volesti/include/preprocess/min_sampling_covering_ellipsoid_rounding.hpp new file mode 100644 index 00000000..1e97835d --- /dev/null +++ b/src/volesti/include/preprocess/min_sampling_covering_ellipsoid_rounding.hpp @@ -0,0 +1,123 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef MIN_ELLIPSOID_ROUNDING_HPP +#define MIN_ELLIPSOID_ROUNDING_HPP + +#include +#include "khach.h" +#include "sampling/random_point_generators.hpp" +#include "volume/sampling_policies.hpp" + +template +< + typename WalkTypePolicy, + typename MT, + typename VT, + typename Polytope, + typename Point, + typename NT, + typename RandomNumberGenerator +> +std::tuple min_sampling_covering_ellipsoid_rounding(Polytope &P, + std::pair &InnerBall, + const unsigned int &walk_length, + RandomNumberGenerator &rng) +{ + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > WalkType; + typedef RandomPointGenerator RandomPointGenerator; + + unsigned int d = P.dimension(); + std::list randPoints; //ds for storing rand points + NT ratio = 10, round_val = 1.0; + unsigned int iter = 0, j, i; + const unsigned int num_of_samples = 10*d;//this is the number of sample points will used to compute min_ellipoid + PushBackWalkPolicy push_back_policy; + + MT T = MT::Identity(d,d); + VT shift = VT::Zero(d); + + while (ratio > 6.0 && iter < 3) + { + randPoints.clear(); + if (!P.get_points_for_rounding(randPoints)) + { // If P is a V-polytope then it will store its vertices in randPoints + // If P is not a V-Polytope or number_of_vertices>20*domension + // 2. Generate the first random point in P + // Perform random walk on random point in the Chebychev ball + Point c = InnerBall.first; + NT radius = InnerBall.second; + Point p = GetPointInDsphere::apply(d, radius, rng); + p += c; + RandomPointGenerator::apply(P, p, num_of_samples, walk_length, + randPoints, push_back_policy, rng); + } + + // Store points in a matrix to call Khachiyan algorithm for the minimum volume enclosing ellipsoid + MT Ap(d, randPoints.size()); + typename std::list::iterator rpit=randPoints.begin(); + + j = 0; + for ( ; rpit!=randPoints.end(); rpit++, j++) { + for (i=0 ; idimension(); i++){ + Ap(i,j)=double((*rpit)[i]); + } + } + MT Q(d,d); //TODO: remove dependence on ublas and copy to eigen + VT c2(d); + size_t w=1000; + KhachiyanAlgo(Ap,0.01,w,Q,c2); // call Khachiyan algorithm + + MT E(d, d); + VT e(d); + + //Get ellipsoid matrix and center as Eigen objects + for(i=0; i eigensolver(E); + NT rel = std::real(eigensolver.eigenvalues()[0]); + NT Rel = std::real(eigensolver.eigenvalues()[0]); + for(i=1; i Rel) Rel = std::real(eigensolver.eigenvalues()[i]); + } + + Eigen::LLT lltOfA(E); // compute the Cholesky decomposition of E + MT L = lltOfA.matrixL(); // retrieve factor L in the decomposition + + //Shift polytope in order to contain the origin (center of the ellipsoid) + P.shift(e); + + MT L_1 = L.inverse(); + shift = shift + T * e; + T = T * L_1.transpose(); + + P.linear_transformIt(L_1.transpose()); + InnerBall = P.ComputeInnerBall(); + round_val *= L_1.determinant(); + ratio = Rel / rel; + iter++; + } + + std::tuple result = std::make_tuple(T, shift, std::abs(round_val)); + return result; +} + + +#endif // MIN_ELLIPSOID_ROUNDING_HPP diff --git a/src/volesti/include/preprocess/svd_rounding.hpp b/src/volesti/include/preprocess/svd_rounding.hpp new file mode 100644 index 00000000..e62c03d2 --- /dev/null +++ b/src/volesti/include/preprocess/svd_rounding.hpp @@ -0,0 +1,176 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef SVD_ROUNDING_HPP +#define SVD_ROUNDING_HPP + + +template +< + typename WalkTypePolicy, + typename Polytope, + typename Point, + typename MT, + typename VT, + typename RandomNumberGenerator +> +void svd_on_sample(Polytope &P, Point &p, unsigned int const& num_rounding_steps, MT &V, VT &s, VT &Means, + unsigned int const& walk_length, RandomNumberGenerator &rng) +{ + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + typedef RandomPointGenerator RandomPointGenerator; + PushBackWalkPolicy push_back_policy; + + unsigned int N = num_rounding_steps; + + std::list randPoints; + MT RetMat(N, P.dimension()); + RandomPointGenerator::apply(P, p, N, walk_length, randPoints, + push_back_policy, rng); + + int jj = 0; + for (typename std::list::iterator rpit = randPoints.begin(); rpit!=randPoints.end(); rpit++, jj++) + { + RetMat.row(jj) = (*rpit).getCoefficients().transpose(); + } + + for (int i = 0; i < P.dimension(); ++i) { + Means(i) = RetMat.col(i).mean(); + } + + for (int i = 0; i < N; ++i) { + RetMat.row(i) = RetMat.row(i) - Means.transpose(); + } + + Eigen::BDCSVD svd(RetMat, Eigen::ComputeFullV); + s = svd.singularValues() / svd.singularValues().minCoeff(); + + if (s.maxCoeff() >= 2.0) { + for (int i = 0; i < s.size(); ++i) { + if (s(i) < 2.0) { + s(i) = 1.0; + } + } + V = svd.matrixV(); + } else { + s = VT::Ones(P.dimension()); + V = MT::Identity(P.dimension(), P.dimension()); + } +} + + +template +< + typename WalkTypePolicy, + typename MT, + typename VT, + typename Polytope, + typename Point, + typename NT, + typename RandomNumberGenerator +> + +std::tuple svd_rounding(Polytope &P, + std::pair &InnerBall, + const unsigned int &walk_length, + RandomNumberGenerator &rng) +{ + NT tol = 0.00000001; + NT R = std::pow(10,10), r = InnerBall.second; + + int n = P.dimension(), m = P.num_of_hyperplanes(); + + Polytope P_old(P); + + MT old_A(m,n), A = P.get_mat(), T = MT::Identity(n,n), round_mat, r_inv; + VT T_shift = VT::Zero(n), shift(n), s(n); + + Point p(n); + + bool done = false, last_round_under_p, fail; + + unsigned int tries=0, num_rounding_steps = 10 * n, rounding_samples = 0, round_it; + NT max_s, s_cutof, p_cutof, num_its, prev_max_s = std::numeric_limits::max(), + s_cutoff, p_cutoff; + MT V(n,n), S(n,n); + + while (!done) { + + T = MT::Identity(n, n); + T_shift = VT::Zero(n); + + round_it = 1; + max_s = std::numeric_limits::max(); + s_cutoff = 2.3; + p_cutoff = 10.0; + last_round_under_p = false; + fail = false; + num_its = 20; + + while (max_s > s_cutoff && round_it <= num_its) { + + p = InnerBall.first; + svd_on_sample(P, p, num_rounding_steps, V, s, + shift, walk_length, rng); + + rounding_samples = rounding_samples + num_rounding_steps; + max_s = s.maxCoeff(); + + if (max_s <= p_cutoff && max_s > s_cutoff) { + if (last_round_under_p) { + num_rounding_steps = num_rounding_steps * 2; + p = InnerBall.first; + svd_on_sample(P, p, num_rounding_steps, V, s, + shift, walk_length, rng); + max_s = s.maxCoeff(); + } else { + last_round_under_p = true; + } + } else { + last_round_under_p = false; + } + S = s.asDiagonal(); + round_mat = V * S; + r_inv = VT::Ones(n).cwiseProduct(s.cwiseInverse()).asDiagonal() * V.transpose(); + + if (round_it != 1 && max_s >= NT(4) * prev_max_s) { + fail = true; + break; + } + + round_it++; + prev_max_s = max_s; + + P.shift(shift); + P.linear_transformIt(round_mat); + InnerBall = P.ComputeInnerBall(); + T_shift += T * shift; + T = T * round_mat; + } + if (round_it <= num_its && !fail) { + done = true; + } else { + tries = tries + 1; + num_rounding_steps = num_rounding_steps * 2.0; + P = P_old; + } + } + + std::tuple result = std::make_tuple(T, shift, std::abs(T.determinant())); + return result; +} + + +#endif diff --git a/src/volesti/include/random_walks/boltzmann_hmc_walk.hpp b/src/volesti/include/random_walks/boltzmann_hmc_walk.hpp new file mode 100644 index 00000000..e69816e9 --- /dev/null +++ b/src/volesti/include/random_walks/boltzmann_hmc_walk.hpp @@ -0,0 +1,216 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLESTI_BOLTZMANN_HMC_WALK_HPP +#define VOLESTI_BOLTZMANN_HMC_WALK_HPP + +#include "generators/boost_random_number_generator.hpp" +#include "../sampling/sphere.hpp" + +/// The Hamiltonian Monte Carlo random walk, to sample from the Boltzmann distribution, i.e. e^(-c*x/T). +struct BoltzmannHMCWalk { +public: + + struct parameters {}; + parameters param; + + + /// The implementation of the walk + /// Currently implemented only for spectrahedra + /// with template specialization + ///@tparam ConvexBody a convex body + ///@tparam RandomNumberGenerator + template + struct Walk { + + /// The matrix/vector types we use + typedef typename ConvexBody::PointType Point; + typedef typename ConvexBody::MT MT; + typedef typename ConvexBody::VT VT; + typedef typename Point::FT NT; + + /// A struct containing the parameters for the random walk + struct Settings { + /// The number of points to "burn", before keeping the following as a sample + int walk_length; + /// For generating random numbers + RandomNumberGenerator randomNumberGenerator; + /// The c in the distribution + VT c; + /// The T in the distribution + NT temperature; + /// The diameter of the body + NT diameter; + /// Set the number of allowed reflections at each step: #reflections < reflectionsBound * dimension + unsigned int reflectionsBound; + /// When determining we can move d long till we reach the boundary, we walk d*dl, for numerical stability + NT dl; + + /// Constructs an object of Settings + /// \param[in] walkLength The number of points to "burn", before keeping the following as a sample + /// \param[in] rng For generating random numbers + /// \param[in] c The c in the distribution + /// \param[in] temperature The T in the distribution + /// \param[in] diameter The diameter of the convexbody + /// \param[in] reflectionsBound at each iteration allow reflectionsBound*dimension reflections at most + /// \param[in] dl approach the boundary with a factor of dl, for numerical stability + /// \return An instance of this struct + template + Settings(const int walkLength, const RandomNumberGenerator &randomNumberGenerator, const Point &c, const NT temperature, const NT diameter, + unsigned int reflectionsBound = 100, NT dl = 0.995) : walk_length(walkLength), randomNumberGenerator(randomNumberGenerator), + c(c.getCoefficients()), + temperature(temperature), + diameter(diameter), + reflectionsBound(reflectionsBound), dl(dl) {} + + Settings() {} + }; + + /// The parameters of the random walk + Settings settings; + + Walk() {} + + /// Constructor + /// \param[in] settings The settings of the random walk + Walk(Settings &settings) : settings(settings) {} + + /// Change the settings + /// \param[in] settings The settings of the random walk + void setSettings(Settings &settings) { + this->settings = settings; + } + + /// Samples random points from the convexbody from the Boltzmann distribution + /// \param[in] convexbody A convexbody + /// \param[in] interiorPoint A point in the interior of the convexbody + /// \param[in] pointsNum The number of points to sample + /// \param[out] points The list of the sampled points + /// \tparam Point class Point with NT and VT as declared above in this class + template + void apply(ConvexBody &convexbody, Point const & interiorPoint, const unsigned int pointsNum, + std::list &points) { + // store intermediate results between successive calls of methods + // of the class convexbody, to avoid repeating computations + + VT p = interiorPoint.getCoefficients(); + + // sample #pointsNum points + for (unsigned int i = 1; i <= pointsNum; ++i) { + // burn #walk_length points to get one sample + for (unsigned int j = 0; j < settings.walk_length; ++j) { + getNextPoint(convexbody, p); + } + + // add the sample in the return list + points.push_back(Point(p)); + } + } + + + /// A single step of the HMC random walk: choose a direction and walk on the trajectory for a random distance. + /// If it hits the boundary, the trajectory is reflected. If #reflections < reflectionsBound * dimension, it returns the same point + /// \param[in] convexbody A convexbody + /// \param[in, out] p An interior point, and the next point in the random walk + /// \tparam Point + template + void getNextPoint(ConvexBody &convexbody, VT &p) { + + // initialize + RandomNumberGenerator &rng = settings.randomNumberGenerator; + boost::random::uniform_real_distribution<> urdist(0, 1); + const NT dl = settings.dl; + unsigned int n = convexbody.dimension(); + int reflectionsNum = 0; + int reflectionsNumBound = settings.reflectionsBound * n; + VT previousPoint; + VT p0 = p; + + // choose a distance to walk + NT T = rng.sample_urdist() * settings.diameter; + + // The trajectory will be of the form a*t^2 + vt + p + // where a = -c / 2*temperature + // and at each reflection, v and p will change + + // crate vector a + VT a = -settings.c / (2 * settings.temperature); + + // The vector v will be a random a direction + VT v = GetDirection::apply(n, rng).getCoefficients(); + + // Begin a step of the random walk + // Also, count the reflections and respect the bound + while (reflectionsNum++ < reflectionsNumBound) { + + // we are at point p and the trajectory a*t^2 + vt + p + // find how long we can walk till we hit the boundary + NT lambda = convexbody.positiveQuadIntersection(a, v, p); + + // We just solved a quadratic polynomial eigenvalue system At^2 + Bt + C, + // where A = lmi(a) - A0, B = lmi(v) - A0 and C = lmi(p) + // and also did a linearization creating two matrices X, Y. + // For the subsequent calls, we don't have to compute these matrices from scratch, + // but can efficiently update them. + // A remains the same + // C := A*lambda^2 + B*lambda + C + // X, Y will be updated in class convexbody + // Set the flags + convexbody.set_flags(true); + + // if we can walk the remaining distance without reaching he boundary + if (T <= lambda) { + // set the new point p:= (T^2)*a + T*V + p + p += (T * T) * a + T * v; + + // update matrix C + convexbody.update_C(T); + return; + } + + // we hit the boundary and still have to walk + // don't go all the way to the boundary, for numerical stability + lambda *= dl; + + // save current and set new point + previousPoint = p; + p += (lambda * lambda) * a + lambda * v; + + // update remaining distance we must walk + T -= lambda; + + // update matrix C + convexbody.update_C(lambda); + //precomputedValues.C += (lambda * lambda) * precomputedValues.A + lambda * precomputedValues.B; + + // Set v to have the direction of the trajectory at t = lambda + // i.e. the gradient of at^2 + vt + p, for t = lambda + v += (lambda * 2) * a; + + // compute reflected direction + convexbody.compute_reflection(v, p); + } + + // if the #reflections exceeded the limit, don't move + if (reflectionsNum == reflectionsNumBound) { + p = p0; + convexbody.set_flags(false); + } + } + + /// Sets the temperature in the distribution + /// \param[in] temperature New value of temperature + void setTemperature(NT temperature) { + settings.temperature = temperature; + } + }; + +}; + +#endif //VOLESTI_BOLTZMANN_HMC_WALK_HPP diff --git a/src/volesti/include/random_walks/boundary_cdhr_walk.hpp b/src/volesti/include/random_walks/boundary_cdhr_walk.hpp new file mode 100644 index 00000000..5e6b5224 --- /dev/null +++ b/src/volesti/include/random_walks/boundary_cdhr_walk.hpp @@ -0,0 +1,91 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_BOUNDARY_CDHR_WALK_HPP +#define RANDOM_WALKS_BOUNDARY_CDHR_WALK_HPP + +#include "sampling/sphere.hpp" + +// random directions hit-and-run walk with uniform target distribution +// from boundary + +struct BCDHRWalk +{ + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, Point const& p, RandomNumberGenerator& rng) + { + initialize(P, p, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + Point& p1, // a point to start + Point& p2, + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + std::pair bpair; + for (auto j = 0u; j < walk_length; ++j) + { + auto rand_coord_prev = _rand_coord; + _rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + bpair = P.line_intersect_coord(_p, + _p_prev, + _rand_coord, + rand_coord_prev, + _lamdas); + _p_prev = _p; + _p.set_coord(_rand_coord, _p[_rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + p1 = _p_prev; + p2 = _p_prev; + p1.set_coord(_rand_coord, _p_prev[_rand_coord] + bpair.first); + p2.set_coord(_rand_coord, _p_prev[_rand_coord] + bpair.second); + } + + private : + + template + inline void initialize(GenericBody const& P, + Point const& p, + RandomNumberGenerator& rng) + { + _lamdas.setZero(P.num_of_hyperplanes()); + _rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + _p = p; + std::pair bpair = P.line_intersect_coord(_p, _rand_coord, + _lamdas); + _p_prev = _p; + _p.set_coord(_rand_coord, _p[_rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + + unsigned int _rand_coord; + Point _p; + Point _p_prev; + typename Point::Coeff _lamdas; + }; + +}; + +#endif // RANDOM_WALKS_BOUNDARY_CDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/boundary_rdhr_walk.hpp b/src/volesti/include/random_walks/boundary_rdhr_walk.hpp new file mode 100644 index 00000000..29d500b6 --- /dev/null +++ b/src/volesti/include/random_walks/boundary_rdhr_walk.hpp @@ -0,0 +1,85 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_BOUNDARY_RDHR_WALK_HPP +#define RANDOM_WALKS_BOUNDARY_RDHR_WALK_HPP + +#include "sampling/sphere.hpp" + +// Random directions hit-and-run walk with uniform target distribution +// from the boundary + +struct BRDHRWalk +{ + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, Point const& p, RandomNumberGenerator& rng) + { + initialize(P, p, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + Point& p1, // a point to start + Point& p2, + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + for (auto j=0u; j::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(_p, v, _lamdas, _Av, + _lambda); + _lambda = rng.sample_urdist() * (bpair.first - bpair.second) + + bpair.second; + p1 = (bpair.first * v); + p1 += _p; + p2 = (bpair.second * v); + p2 += _p; + _p += (_lambda * v); + } + } + + private : + + template + inline void initialize(GenericBody const& P, + Point const& p, + RandomNumberGenerator& rng) + { + _lamdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + + Point v = GetDirection::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(p, v, _lamdas, _Av); + _lambda = rng.sample_urdist() * (bpair.first - bpair.second) + bpair.second; + _p = (_lambda * v) + p; + } + + Point _p; + NT _lambda; + typename Point::Coeff _lamdas; + typename Point::Coeff _Av; + }; + +}; + + +#endif // RANDOM_WALKS_BOUNDARY_RDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/compute_diameter.hpp b/src/volesti/include/random_walks/compute_diameter.hpp new file mode 100644 index 00000000..b20aa839 --- /dev/null +++ b/src/volesti/include/random_walks/compute_diameter.hpp @@ -0,0 +1,225 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2022 Vissarion Fisikopoulos +// Copyright (c) 2018-2022 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_COMPUTE_DIAMETER_HPP +#define RANDOM_WALKS_COMPUTE_DIAMETER_HPP + +#include "convex_bodies/ball.h" +#include "convex_bodies/ballintersectconvex.h" +#include "convex_bodies/hpolytope.h" +#include "convex_bodies/spectrahedra/spectrahedron.h" +#ifndef DISABLE_LPSOLVE + #include "convex_bodies/vpolytope.h" + #include "convex_bodies/vpolyintersectvpoly.h" + #include "convex_bodies/zpolytope.h" + #include "convex_bodies/zonoIntersecthpoly.h" +#endif +#include "convex_bodies/orderpolytope.h" +#include "convex_bodies/ellipsoid.h" + + +template +struct compute_diameter +{ + template + static NT compute(GenericPolytope) {return NT(0);} +}; + + +template +struct compute_diameter> +{ + template + static NT compute(HPolytope &P) + { + return NT(2) * std::sqrt(NT(P.dimension())) * P.InnerBall().second; + } +}; + +template +struct compute_diameter> +{ + template + static NT compute(Spectrahedron &P) + { + std::pair inner_ball = P.ComputeInnerBall(); + return NT(6) * NT(P.dimension()) * inner_ball.second; + } +}; + +template +struct compute_diameter> +{ + template + static NT compute(CorrelationSpectrahedron &P) + { + std::pair inner_ball = P.getInnerBall(); + return NT(P.dimension()) * inner_ball.second; + } +}; + +template +struct compute_diameter> +{ + template + static NT compute(CorrelationSpectrahedron_MT &P) + { + std::pair inner_ball = P.getInnerBall(); + return NT(P.dimension()) * inner_ball.second; + } +}; + +#ifndef DISABLE_LPSOLVE +template +struct compute_diameter> +{ + template + static NT compute(VPolytope &P) + { + typedef typename VPolytope::MT MT; + NT diameter = NT(0), diam_iter; + MT V = P.get_mat(); + for (int i = 0; i < V.rows(); ++i) { + for (int j = 0; j < V.rows(); ++j) { + if (i != j) { + diam_iter = (V.row(i) - V.row(j)).norm(); + if (diam_iter > diameter) diameter = diam_iter; + } + } + } + return diameter; + } +}; + +template +struct compute_diameter> +{ + template + static NT compute(Zonotope &P) + { + typedef typename Zonotope::MT MT; + typedef typename Zonotope::VT VT; + + MT V = P.get_mat(); + int k = V.rows(), max_index = -1; + MT D = V.transpose() * V; + D = (D + D.transpose()) / 2.0; + Eigen::SelfAdjointEigenSolver es(D); + MT D2 = es.eigenvalues().asDiagonal(), Q = es.eigenvectors(); + + NT max_eig = NT(0); + for (int i = 0; i < P.dimension(); ++i) { + if (es.eigenvalues()[i] > max_eig) { + max_eig = es.eigenvalues()[i]; + max_index = i; + } + } + + VT max_eigvec = -1.0 * Q.col(max_index); + VT obj_fun = max_eigvec.transpose() * V.transpose(), x0(k); + + for (int j = 0; j < k; ++j) x0(j) = (obj_fun(j) < 0.0) ? -1.0 : 1.0; + + return NT(2) * (V.transpose() * x0).norm(); + } +}; + +template +struct compute_diameter, RandomNumberGenerator>> +{ + template + static NT compute(IntersectionOfVpoly, RandomNumberGenerator> &P) + { + return NT(2) * NT(P.dimension()) * P.InnerBall().second; + } +}; + +template +struct compute_diameter, HPolytope>> +{ + template + static NT compute(ZonoIntersectHPoly, HPolytope> &P) + { + typedef typename ZonoIntersectHPoly, HPolytope>::VT VT; + typedef typename ZonoIntersectHPoly, HPolytope>::MT MT; + typedef HPolytope Hpolytope; + + typedef BoostRandomNumberGenerator RandomNumberGenerator; + PushBackWalkPolicy push_back_policy; + typedef typename BCDHRWalk::template Walk + < + Hpolytope, + RandomNumberGenerator + > BCdhrWalk; + typedef BoundaryRandomPointGenerator BCdhrRandomPointGenerator; + + MT G = P.get_T().transpose(); + MT AG = P.get_mat()*G; + int k = G.cols(), d = P.dimension(); + MT eyes1(k, 2*k); + eyes1 << MT::Identity(k,k), NT(-1) * MT::Identity(k,k); + MT M1(k, 4*k); + M1 << AG.transpose(), eyes1; + MT M = M1.transpose(); + VT b = P.get_vec(); + + VT bb(4*k); + for (int i = 0; i < 4*k; ++i) bb(i) = (i < 2*k) ? b(i) : 1.0; + + Hpolytope HP(d, M, bb); + + RandomNumberGenerator rng(HP.dimension()); + + std::list randPoints; + std::pair InnerBall = HP.ComputeInnerBall(); + Point q = InnerBall.first; + BCdhrRandomPointGenerator::apply(HP, q, 4*d*d, 1, + randPoints, push_back_policy, rng); + typename std::list::iterator rpit=randPoints.begin(); + NT max_norm = NT(0), iter_norm; + for ( ; rpit!=randPoints.end(); rpit++) { + iter_norm = (G*(*rpit).getCoefficients()).norm(); + if (iter_norm > max_norm) max_norm = iter_norm; + } + return NT(2) * max_norm; + } +}; + +#endif + +template +struct compute_diameter>> +{ + template + static NT compute(BallIntersectPolytope> &P) + { + return NT(2) * P.radius(); + } +}; + +template +struct compute_diameter> +{ + template + static NT compute(OrderPolytope& P) + { + return std::sqrt(NT(P.dimension())); + } +}; + +template +struct compute_diameter, Ellipsoid > > +{ + template + static NT compute(BallIntersectPolytope, Ellipsoid>& P) + { + NT polytope_diameter = std::sqrt(NT(P.dimension())); + return std::min(polytope_diameter, (NT(2) * P.radius())); + } +}; + +#endif // RANDOM_WALKS_COMPUTE_DIAMETER_HPP diff --git a/src/volesti/include/random_walks/crhmc/additional_units/auto_tuner.hpp b/src/volesti/include/random_walks/crhmc/additional_units/auto_tuner.hpp new file mode 100644 index 00000000..66a8d51a --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/additional_units/auto_tuner.hpp @@ -0,0 +1,64 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef AUTO_TUNER_HPP +#define AUTO_TUNER_HPP +#include "random_walks/crhmc/additional_units/dynamic_regularizer.hpp" +#include "random_walks/crhmc/additional_units/dynamic_step_size.hpp" +#include "random_walks/crhmc/additional_units/dynamic_weight.hpp" + +// This class is responsible for calling the additional crhmc modules for: +// modifying the weights, ode step size and regularizer factor addaptively. + +template +class auto_tuner { + using weight_tuner = dynamic_weight; + using regularizion_tuner = + dynamic_regularizer; + using step_size_tuner = dynamic_step_size; + + using Opts = typename Sampler::Opts; + +public: + Opts options; + std::unique_ptr tune_weights; + std::unique_ptr tune_regularization; + std::unique_ptr tune_step_size; + auto_tuner(Sampler &s) : + options(s.params.options) + { + if (options.DynamicWeight) { + tune_weights = std::unique_ptr(new weight_tuner(s)); + } + if (options.DynamicRegularizer) { + tune_regularization = std::unique_ptr(new regularizion_tuner(s)); + } + if (options.DynamicStepSize) { + tune_step_size = std::unique_ptr(new step_size_tuner(s)); + } + } + void updateModules(Sampler &s, RandomNumberGenerator &rng) { + if (options.DynamicWeight) { + tune_weights->update_weights(s, rng); + } + if (options.DynamicRegularizer) { + tune_regularization->update_regularization_factor(s, rng); + } + if (options.DynamicStepSize) { + tune_step_size->update_step_size(s); + } + } +}; +#endif diff --git a/src/volesti/include/random_walks/crhmc/additional_units/dynamic_regularizer.hpp b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_regularizer.hpp new file mode 100644 index 00000000..81449836 --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_regularizer.hpp @@ -0,0 +1,59 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef DYNAMIC_REGULARIZER_HPP +#define DYNAMIC_REGULARIZER_HPP +#include "Eigen/Eigen" + +/// Module for updating the extra term we add to the barrier +/// This is nessecary for any polytope with free variables +/// Part of crhmc sampler +template +class dynamic_regularizer { +public: + using NT = typename Sampler::NT; + using Point = typename Sampler::point; + using MT = Eigen::Matrix; + using Opts = typename Sampler::Opts; + int n; + int simdLen; + MT bound; + Opts &options; + MT &extraHessian; + dynamic_regularizer(Sampler &s) : + simdLen(s.simdLen), + options(s.params.options), + extraHessian(options.DynamicWeight + ? s.solver->ham.weighted_barrier->extraHessian + : s.solver->ham.barrier->extraHessian) + { + n = s.dim; + bound = MT::Ones(n, simdLen); + extraHessian = MT::Ones(n, simdLen); + } + + void update_regularization_factor(Sampler &s, RandomNumberGenerator &rng) { + MT x = s.x; + x = (x.cwiseAbs()).cwiseMax(1); + bound = bound.cwiseMax(x); + if ((2 / (bound.array() * bound.array()) < n * extraHessian.array()).any()) { + extraHessian = (0.5 / n) * (bound.cwiseProduct(bound)).cwiseInverse(); + s.solver->ham.forceUpdate = true; + s.solver->ham.move({s.x, s.v}); + s.v = s.get_direction_with_momentum(n, rng, s.x, MT::Zero(n, simdLen), 0, false); + } + } +}; +#endif diff --git a/src/volesti/include/random_walks/crhmc/additional_units/dynamic_step_size.hpp b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_step_size.hpp new file mode 100644 index 00000000..36721196 --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_step_size.hpp @@ -0,0 +1,118 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef DYNAMIC_STEP_SIZE_HPP +#define DYNAMIC_STEP_SIZE_HPP + +/// Module for dynamically choosing the ODE step size and the velocity momentum +/// Part of crhmc sampler +template +class dynamic_step_size { + using NT = typename Sampler::NT; + using Opts = typename Sampler::Opts; + using IVT = Eigen::Array; + using VT = Eigen::Array; + +public: + int simdLen; + IVT consecutiveBadStep; + int iterSinceShrink = 0; + VT rejectSinceShrink; + int ODEStepSinceShrink = 0; + int effectiveStep = 0; + bool warmupFinished = false; + Opts &options; + NT η + NT &momentum; + VT acceptedStep; + VT nEffectiveStep; // number of effective steps + NT accumulatedMomentum = 0; + dynamic_step_size(Sampler &s) : + simdLen(s.simdLen), + options(s.params.options), + eta(s.solver->eta), + momentum(s.params.momentum) + { + nEffectiveStep = VT::Zero(simdLen); + acceptedStep = VT::Zero(simdLen); + consecutiveBadStep = IVT::Zero(simdLen); + rejectSinceShrink = VT::Zero(simdLen); + + if (options.warmUpStep > 0) { + eta = 1e-3; + } else { + warmupFinished = true; + } + } + void update_step_size(Sampler &s) { + acceptedStep = acceptedStep + s.prob.array(); + accumulatedMomentum = s.prob.mean() * momentum * accumulatedMomentum + eta; + Eigen::Matrix accept = (s.accept.template cast()); + nEffectiveStep = nEffectiveStep + eta * accumulatedMomentum * accept.array(); + IVT bad_step = IVT::Zero(simdLen); + if (s.solver->num_steps == options.maxODEStep) { + bad_step += 1; + } else { + bad_step = (s.prob.array() < 0.5).select(1, IVT::Zero(simdLen)); + } + consecutiveBadStep = bad_step * consecutiveBadStep + bad_step; + + NT warmupRatio = nEffectiveStep.mean() / options.warmUpStep; + if (warmupRatio < 1 && !warmupFinished && + consecutiveBadStep.maxCoeff() < options.maxConsecutiveBadStep) { + eta = options.initialStep * std::min(warmupRatio + 1e-2, 1.0); + momentum = 1 - std::min(1.0, eta / options.effectiveStepSize); + return; + } + + if (!warmupFinished) { + acceptedStep = VT::Zero(simdLen); + nEffectiveStep = VT::Zero(simdLen); + warmupFinished = true; + } + + iterSinceShrink++; + rejectSinceShrink += 1 - s.prob.array(); + ODEStepSinceShrink += s.solver->num_steps; + + int shrink = 0; + NT shiftedIter = iterSinceShrink + 20 / (1 - momentum); + + NT targetProbability = std::pow((1.0 - momentum), (2 / 3)) / 4; + if (rejectSinceShrink.maxCoeff() > targetProbability * shiftedIter|| + consecutiveBadStep.maxCoeff() > options.maxConsecutiveBadStep || + ODEStepSinceShrink > options.targetODEStep * shiftedIter) { + shrink = 1; + } + + if (shrink == 1) { + iterSinceShrink = 0; + ODEStepSinceShrink = 0; + rejectSinceShrink = VT::Zero(simdLen); + consecutiveBadStep = IVT::Zero(simdLen); + + eta /= options.shrinkFactor; + momentum = 1 - std::min(0.999, eta / options.effectiveStepSize); + + if (eta < options.minStepSize) { + s.P.terminate=true; + s.P.terminate_message="Algorithm fails to converge even with step size h = "+std::to_string(eta)+"\n"; + } + } + + iterSinceShrink = iterSinceShrink + 1; + } +}; +#endif diff --git a/src/volesti/include/random_walks/crhmc/additional_units/dynamic_weight.hpp b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_weight.hpp new file mode 100644 index 00000000..6b943377 --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/additional_units/dynamic_weight.hpp @@ -0,0 +1,76 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef DYNAMIC_WEIGHT_HPP +#define DYNAMIC_WEIGHT_HPP + +/// Class responsible for updating the weights of the barrier +/// Part of crhmc sampler +template +class dynamic_weight { + using NT = typename Sampler::NT; + using Point = typename Sampler::point; + using VT = Eigen::Matrix; + using MT = Eigen::Matrix; + using IVT = Eigen::Array; + using Opts = typename Sampler::Opts; + +public: + int simdLen; + IVT consecutiveBadStep; + int n; + VT &w; + Opts options; + dynamic_weight(Sampler &s) + : simdLen(s.simdLen), w(s.solver->ham.weighted_barrier->w), options(s.params.options) + { + n = s.dim; + consecutiveBadStep = IVT::Zero(simdLen); + } + // If we have consecutive bad steps update the weights with + // the help of the leverage scores. + void update_weights(Sampler &s, RandomNumberGenerator &rng) + { + IVT bad_step = IVT::Zero(simdLen); + if (s.solver->num_steps == options.maxODEStep) { + bad_step += 1; + } else { + bad_step = (s.prob.array() < 0.5).select(1, IVT::Zero(simdLen)); + } + NT threshold; + consecutiveBadStep = bad_step * consecutiveBadStep + bad_step; + + if (s.accept.sum() < simdLen) { + VT lsc = s.solver->ham.lsc.colwise().maxCoeff().transpose(); + /*The more bad steps in a row we have the higher we want the threshold to be + In order to change w more drasticaly according to the leverage scores. + So if we have more than 2 bad steps in a row we elect to set the threshold to 4 + else to 16. Not many changes will be possible as the w should be upperbounded by 1*/ + if (consecutiveBadStep.maxCoeff() > 2) { + threshold = 4; + } else { + threshold = 16; + } + bool changed = (lsc.array() > threshold * w.array()).any(); + if (changed) { + w = (lsc.array() > threshold * w.array()).select((w * threshold).cwiseMin(1), w); + s.solver->ham.forceUpdate = true; + s.solver->ham.move({s.x, s.v}); + s.v = s.get_direction_with_momentum(n, rng, s.x, MT::Zero(n, simdLen), false); + } + } + } +}; +#endif diff --git a/src/volesti/include/random_walks/crhmc/crhmc_walk.hpp b/src/volesti/include/random_walks/crhmc/crhmc_walk.hpp new file mode 100644 index 00000000..21ed8d6b --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/crhmc_walk.hpp @@ -0,0 +1,245 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef CRHMC_WALK_HPP +#define CRHMC_WALK_HPP +#include "generators/boost_random_number_generator.hpp" +#include "ode_solvers/ode_solvers.hpp" +#include "random_walks/crhmc/additional_units/auto_tuner.hpp" +#include "random_walks/gaussian_helpers.hpp" +#include +struct CRHMCWalk { + template + < + typename NT, + typename OracleFunctor + > + struct parameters { + using Opts = opts; + NT epsilon; // tolerance in mixing + NT eta = 0.2; // step size + NT momentum; + NT effectiveStepSize = 1; + Opts &options; + parameters(OracleFunctor const &F, + unsigned int dim, + Opts &user_options, + NT epsilon_ = 2) : + options(user_options) + { + epsilon = epsilon_; + eta = 1.0 / (dim * sqrt(F.params.L)); + momentum = 1 - std::min(1.0, eta / effectiveStepSize); + } + }; + + template + < + typename Point, + typename Polytope, + typename RandomNumberGenerator, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Solver + > + struct Walk { + using point = Point; + using pts = std::vector; + using NT = typename Point::FT; + using VT = Eigen::Matrix; + using MT = Eigen::Matrix; + using Sampler = CRHMCWalk::Walk; + + using Opts = typename Polytope::Opts; + using IVT = Eigen::Matrix; + + // Hyperparameters of the sampler + parameters ¶ms; + + // Numerical ODE solver + std::unique_ptr solver; + + // Dimension + unsigned int dim; + + // Polytope + Polytope &P; + + // Discarded Samples + long total_discarded_samples = 0; + long num_runs = 0; + float discard_ratio = 0; + + // Average acceptance probability + float total_acceptance_prob = 0; + float average_acceptance_prob = 0; + + // Acceptance probability + VT prob; + bool accepted; + IVT accept; + bool update_modules; + int simdLen; + // References to xs + MT x, v; + + // Proposal points + MT x_tilde, v_tilde; + + // Gradient function + NegativeGradientFunctor &F; + + // Auto tuner + std::unique_ptr>module_update; + + // Helper variables + VT H, H_tilde; + // Density exponent + NegativeLogprobFunctor &f; +#ifdef TIME_KEEPING + std::chrono::time_point start, end; + std::chrono::duration H_duration = std::chrono::duration::zero(); +#endif + Walk(Polytope &Problem, + Point &p, + NegativeGradientFunctor &neg_grad_f, + NegativeLogprobFunctor &neg_logprob_f, + parameters ¶m) : + params(param), + P(Problem), + F(neg_grad_f), + f(neg_logprob_f) + { + + dim = p.dimension(); + simdLen = params.options.simdLen; + // Starting point is provided from outside + x = p.getCoefficients() * MT::Ones(1, simdLen); + accepted = false; + // Initialize solver + solver = std::unique_ptr(new Solver(0.0, params.eta, {x, x}, F, Problem, params.options)); + v = MT::Zero(dim, simdLen); + module_update = std::unique_ptr>(new auto_tuner(*this)); + update_modules = params.options.DynamicWeight || + params.options.DynamicRegularizer || + params.options.DynamicStepSize; + }; + // Sample a new velocity with momentum + MT get_direction_with_momentum(unsigned int const &dim, + RandomNumberGenerator &rng, MT const &x, MT v, + NT momentum = 0, bool normalize = true) + { + MT z = MT(dim, simdLen); + for (int i = 0; i < simdLen; i++) + { + z.col(i) = GetDirection::apply(dim, rng, normalize).getCoefficients(); + } + solver->ham.move({x, v}); + MT sqrthess = (solver->ham.hess).cwiseSqrt(); + z = sqrthess.cwiseProduct(z); + return v * std::sqrt(momentum) + z * std::sqrt(1 - momentum); + } + // Returns the current point in the tranformed in the original space + inline MT getPoints() { return (P.T * x).colwise() + P.y; } + // Returns the current point in the tranformed in the original space + inline Point getPoint() { return Point(P.T * x.col(0) + P.y); } + + inline MT masked_choose(MT &x, MT &x_tilde, IVT &accept) { + return accept.transpose().replicate(x.rows(), 1).select(x_tilde, x); + } + inline void disable_adaptive(){ + update_modules=false; + } + inline void apply(RandomNumberGenerator &rng, + int walk_length = 1, + bool metropolis_filter = true) + { + num_runs++; + // Pick a random velocity with momentum + v = get_direction_with_momentum(dim, rng, x, v, params.momentum, false); + solver->set_state(0, x); + solver->set_state(1, v); + // Get proposals + solver->steps(walk_length, accepted); + x_tilde = solver->get_state(0); + v_tilde = solver->get_state(1); + if (metropolis_filter) { +#ifdef TIME_KEEPING + start = std::chrono::system_clock::now(); +#endif + // Calculate initial Hamiltonian + H = solver->ham.hamiltonian(x, v); + + // Calculate new Hamiltonian + H_tilde = solver->ham.hamiltonian(x_tilde, -v_tilde); + +#ifdef TIME_KEEPING + end = std::chrono::system_clock::now(); + H_duration += end - start; +#endif + VT feasible = solver->ham.feasible(x_tilde, + v_tilde); + prob = (1.0 < exp((H - H_tilde).array())).select(1.0, exp((H - H_tilde).array())); + prob = (feasible.array() > 0.5).select(prob, 0); + + total_acceptance_prob += prob.sum(); + VT rng_vector = VT(simdLen); + for (int i = 0; i < simdLen; i++) + { + rng_vector(i) = rng.sample_urdist(); + } + accept = (rng_vector.array() < prob.array()).select(1 * IVT::Ones(simdLen), 0 * IVT::Ones(simdLen)); + + x = masked_choose(x, x_tilde, accept); + v = -v; + v = masked_choose(v, v_tilde, accept); + total_discarded_samples += simdLen - accept.sum(); + discard_ratio = (1.0 * total_discarded_samples) / (num_runs * simdLen); + average_acceptance_prob = total_acceptance_prob / (num_runs * simdLen); + } else { + x = x_tilde; + v = v_tilde; + } + if (update_modules) { + module_update->updateModules(*this, rng); + } + } +#ifdef TIME_KEEPING + void initialize_timers() { + H_duration = std::chrono::duration::zero(); + solver->DU_duration = std::chrono::duration::zero(); + solver->approxDK_duration = std::chrono::duration::zero(); + } + template + void print_timing_information(StreamType &stream) { + stream << "---Sampling Timing Information" << std::endl; + double DU_time = solver->DU_duration.count(); + double DK_time = solver->approxDK_duration.count(); + double H_time = H_duration.count(); + double total_time = H_time + DK_time + DU_time; + stream << "Computing the Hamiltonian in time, " << H_time << " secs\n"; + stream << "Computing DU partial derivatives in time, " << DU_time + << " secs\n"; + stream << "Computing DK partial derivatives in time, " << DK_time + << " secs\n"; + stream << "H_time + DK_time + DU_time: " << total_time << "\n"; + } +#endif + }; +}; + +#endif diff --git a/src/volesti/include/random_walks/crhmc/hamiltonian.hpp b/src/volesti/include/random_walks/crhmc/hamiltonian.hpp new file mode 100644 index 00000000..0d9ef546 --- /dev/null +++ b/src/volesti/include/random_walks/crhmc/hamiltonian.hpp @@ -0,0 +1,249 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2022-2022 Ioannis Iakovidis + +// Contributed and/or modified by Ioannis Iakovidis, as part of Google Summer of +// Code 2022 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Yunbum Kook, Yin Tat Lee, Ruoqi Shen, Santosh S. Vempala. "Sampling with +// Riemannian Hamiltonian +// Monte Carlo in a Constrained Space" +#ifndef HAMILTONIAN_HPP +#define HAMILTONIAN_HPP +#include "preprocess/crhmc/two_sided_barrier.h" +#include "preprocess/crhmc/weighted_two_sided_barrier.h" +#include "PackedCSparse/PackedChol.h" +#include "preprocess/crhmc/crhmc_utils.h" +#include + +template +class Hamiltonian { + using VT = typename Polytope::VT; + using IVT = Eigen::Array; + using BVT = Eigen::Matrix; + using NT = typename Polytope::NT; + using MT = typename Polytope::MT; + using SpMat = typename Polytope::SpMat; + using Tx = FloatArray; + using CholObj = PackedChol; + using Opts = typename Polytope::Opts; + using pts = std::vector; + using Barrier = two_sided_barrier; + using WeightedBarrier = weighted_two_sided_barrier; + +public: + bool prepared = false; + bool forceUpdate = true; // Update function oracle temporary varibles + Polytope &P; + MT hess; + bool dUDx_empty = true; + MT last_dUdx; + CholObj solver; + pts xs; + MT x; + MT dfx; + MT lsc; + VT fx; + int n; + int m; + int num_runs = 0; + Barrier *barrier; + std::unique_ptr weighted_barrier; + Opts &options; + Hamiltonian(Polytope &boundaries) : + P(boundaries), + solver(CholObj(transform_format(boundaries.Asp))), + options(boundaries.options) + { + n = P.dimension(); + m = P.equations(); + x = MT::Zero(n, simdLen); + xs = {x, x}; + lsc = MT::Zero(simdLen, n); + solver.accuracyThreshold = options.solver_accuracy_threshold; + if (options.DynamicWeight) + { + weighted_barrier = + std::unique_ptr(new WeightedBarrier(P.barrier.lb, P.barrier.ub, P.w_center)); + weighted_barrier->extraHessian.resize(n, simdLen); + weighted_barrier->extraHessian = MT::Ones(n, simdLen) * options.regularization_factor; + } + barrier = &P.barrier; + barrier->extraHessian.resize(n, simdLen); + barrier->extraHessian = MT::Ones(n, simdLen) * options.regularization_factor; + } + + // Compute H(x,v) + VT hamiltonian(MT x, MT v) + { + prepare({x, v}); + pts pd = DK({x, v}); + VT K = 0.5 * (v.cwiseProduct(pd[0])).colwise().sum(); + Tx out=solver.logdet(); + VT logdet=VT(simdLen); + for (int i = 0; i < simdLen; i++) + logdet(i) = get(out, i); + VT U = ((hess.array()).log()).colwise().sum(); + U = (U + logdet) * 0.5 + fx; + VT E = U + K; + return E; + } + // Helper is nan function for vectors + template + IVT is_not_nan(MatrixType x) + { + IVT result = IVT::Ones(x.cols()); + for (int i = 0; i < x.rows(); i++) { + for (int j = 0; j < x.cols(); j++) { + if (std::isnan(x(i, j))) { + result(j) = 0; + } + } + } + return result; + } + // Test if the values of x and v are valid and if x is feasible + VT feasible(MT x, MT v) + { + VT feasible_coordinate = VT::Ones(x.cols()); + if (options.DynamicWeight) { + feasible_coordinate = weighted_barrier->feasible(x); + } else { + feasible_coordinate = barrier->feasible(x); + } + VT r = feasible_coordinate.cwiseProduct((is_not_nan(x) * is_not_nan(v)).matrix()); + return r; + } + // prepare the solver weighted by the hessian + void prepare(pts const &xs) + { + move(xs); + if (!prepared) { + MT Hinv = (hess.cwiseInverse()).transpose(); + solver.decompose((Tx *)Hinv.data()); + dUDx_empty = true; + } + prepared = true; + } + // Computation of the partial derivatives of the K term + pts DK(pts const &x_bar) + { + MT x = x_bar[0]; + MT v = x_bar[1]; + move(x_bar); + MT invHessV = v.cwiseQuotient(hess); + MT input_vector = P.Asp * invHessV; + input_vector.transposeInPlace(); + MT out_vector = MT::Zero(simdLen, m); + solver.solve((Tx *)input_vector.data(), (Tx *)out_vector.data()); + out_vector.transposeInPlace(); + MT dKdv = + invHessV - (P.Asp.transpose() * out_vector).cwiseQuotient(hess); + + MT dKdx = MT::Zero(n, simdLen); + if (options.DynamicWeight) { + dKdx = + weighted_barrier->quadratic_form_gradient(x, dKdv) / + 2; + } else { + dKdx = barrier->quadratic_form_gradient(x, dKdv) / + 2; + } + + return {dKdv, dKdx}; + } + // Approximate computation of the partial derivatives of the K term + pts approxDK(pts const &x_bar, MT &nu) + { + MT x = x_bar[0]; + MT v = x_bar[1]; + move(x_bar); + MT dUdv_b = P.Asp * (v - P.Asp.transpose() * nu).cwiseQuotient(hess); + dUdv_b.transposeInPlace(); + MT out_solver = MT(nu.cols(), nu.rows()); + solver.solve((Tx *)dUdv_b.data(), (Tx *)out_solver.data()); + nu = nu + out_solver.transpose(); + + MT dKdv = (v - P.Asp.transpose() * nu).cwiseQuotient(hess); + MT dKdx = MT::Zero(n, simdLen); + if (options.DynamicWeight) { + dKdx = + weighted_barrier->quadratic_form_gradient(x, dKdv) / + 2; + } else { + dKdx = barrier->quadratic_form_gradient(x, dKdv) / + 2; + } + return {dKdv, dKdx}; + } + // Compute the partial derivatives of one term + // This is only dependent on x and so DU/Dv=0 + pts DU(pts const &x_bar) + { + MT x = x_bar[0]; + move(x_bar); + if (!prepared || dUDx_empty) { + prepare(x_bar); + solver.leverageScoreComplement((Tx *)lsc.data()); + + if (options.DynamicWeight) { + last_dUdx = (weighted_barrier->tensor(x).cwiseProduct(lsc.transpose())) + .cwiseQuotient(2 * hess) + + dfx; + } else { + last_dUdx = + (barrier->tensor(x).cwiseProduct(lsc.transpose())).cwiseQuotient(2 * hess) + + dfx; + } + dUDx_empty = false; + } + + return {MT::Zero(n, simdLen), -last_dUdx}; + } + // Compute the computations involving only x iff x has been changed + // Else they are stored + void move(pts const &y) + { + if (y[0] == xs[0] && !forceUpdate) { + return; + } + xs = y; + x = xs[0]; + MT h; + std::tie(fx, dfx, h) = P.f_oracle(x); + if (options.DynamicWeight) { + hess = weighted_barrier->hessian(x) + h; + } else { + hess = barrier->hessian(x) + h; + } + forceUpdate = false; + prepared = false; + } + // Project x to the polytope + void project(pts &xs) { + move(xs); + MT x = xs[0]; + int m = P.Asp.rows(); + MT out_vector = MT(simdLen, m); + MT in_vector = (-P.Asp * x).colwise() + P.b; + in_vector.transposeInPlace(); + solver.solve((Tx *)in_vector.data(), (Tx *)out_vector.data()); + out_vector.transposeInPlace(); + out_vector = P.Asp.transpose() * out_vector; + xs[0] = xs[0] + (out_vector).cwiseQuotient(hess); + } + // Get the inner product of x and ds weighted by the hessian + VT x_norm(pts const &xs, pts const &dx) + { + move(xs); + MT dx_x = dx[0]; + MT r = (dx_x.cwiseProduct(dx_x)).cwiseProduct(hess); + return r.colwise().sum(); + } +}; +#endif diff --git a/src/volesti/include/random_walks/ellipsoid_walks/README.md b/src/volesti/include/random_walks/ellipsoid_walks/README.md new file mode 100644 index 00000000..ec1e5785 --- /dev/null +++ b/src/volesti/include/random_walks/ellipsoid_walks/README.md @@ -0,0 +1,7 @@ +1) This folder contains the C++ implementations -with several modifications- of the Dikin walk, + the Vaidya walk and the John walk from https://github.com/rzrsk/vaidya-walk by Raaz Dwivedi. + +2) The implemented random walks are presented in the paper of + Y. Chen, R. Dwivedi, M. J. Wainwright and B. Yu, + "Fast MCMC Sampling Algorithms on Polytopes", + Journal of Machine Learning Research, 2018. diff --git a/src/volesti/include/random_walks/ellipsoid_walks/dikin_walker.h b/src/volesti/include/random_walks/ellipsoid_walks/dikin_walker.h new file mode 100644 index 00000000..b16d997b --- /dev/null +++ b/src/volesti/include/random_walks/ellipsoid_walks/dikin_walker.h @@ -0,0 +1,157 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Original C++ code from https://github.com/rzrsk/vaidya-walk by Raaz Dwivedi. + +// Modified by Alexandros Manochis to be integrated in volesti, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// The implemented random walk is presented in the paper of +// Y. Chen, R. Dwivedi, M. J. Wainwright and B. Yu, +// "Fast MCMC Sampling Algorithms on Polytopes", +// Journal of Machine Learning Research, 2018. + +#ifndef PWALK_DIKIN_WALKER_HPP_ +#define PWALK_DIKIN_WALKER_HPP_ + +#include +#include "math_functions.h" + +/// @brief Class that defines the Dikin walk sampler +/// @tparam Dtype Number Type +template +class DikinWalker { +public: + + DikinWalker() {} + + DikinWalker(const Eigen::Matrix &initialization, const Eigen::Matrix &cons_A, const Eigen::Matrix &cons_b, const Dtype r){ + nb_dim_ = cons_A.cols(); + nb_cons_ = cons_A.rows(); + nb_curr_samples_ = 1; + initialization_ = initialization; + cons_A_ = cons_A; + cons_b_ = cons_b; + curr_sample_ = initialization; + r_ = r; + } + + // getter for radius + Dtype getRadius() { + return r_; + } + + // check whether a given point is in th polytope + bool checkInPolytope(const Eigen::Matrix &new_sample) { + return (cons_A_ * new_sample - cons_b_).maxCoeff() < 0; + } + + // getter for dimension + int getNbDim() { + return nb_dim_; + } + + // getter for nb current samples + int getNbCurrSamples() { + return nb_curr_samples_; + } + + // getter for current sample + Eigen::Matrix &getCurrSample() { + return curr_sample_; + } + + void proposal(Eigen::Matrix& new_sample){ + Eigen::Matrix gaussian_step = + Eigen::Matrix::Zero(this->nb_dim_); + sample_gaussian(this->nb_dim_, 0., 1., gaussian_step); + + // get hessian + Eigen::Matrix new_sqrt_inv_hess = + Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess); + + new_sample = this->curr_sample_ + r_ / std::sqrt(Dtype(this->nb_dim_)) * (new_sqrt_inv_hess * gaussian_step); + } + + bool acceptRejectReverse(const Eigen::Matrix& new_sample){ + // get hessian on x + Eigen::Matrix new_sqrt_inv_hess_x = + Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess_x); + // get hessian on y + Eigen::Matrix new_sqrt_inv_hess_y = + Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(new_sample, new_sqrt_inv_hess_y); + + Dtype scale = r_/std::sqrt(Dtype(this->nb_dim_)); + Dtype p_y_to_x = gaussian_density(this->curr_sample_, new_sample, new_sqrt_inv_hess_y.inverse()/scale); + Dtype p_x_to_y = gaussian_density(new_sample, this->curr_sample_, new_sqrt_inv_hess_x.inverse()/scale); + + Dtype ar_ratio = std::min(1., p_y_to_x/p_x_to_y); + + Dtype random_num = rng_uniform(0., 1.); + // lazy version of the walk + if (random_num > ar_ratio) { + return false; + } + + return true; + } + + bool doSample(Eigen::Matrix& new_sample, const Dtype lazy = Dtype(0.5)){ + proposal(new_sample); + this->nb_curr_samples_ += 1; + // for lazy markov chain + Dtype random_num = rng_uniform(0., 1.); + // check balance and check in polytope + if (random_num < lazy && this->checkInPolytope(new_sample) && acceptRejectReverse(new_sample)){ + this->curr_sample_ = new_sample; + return true; + } else { + new_sample = this->curr_sample_; + return false; + } + } + + void sqrtInvHessBarrier(const Eigen::Matrix& new_sample, Eigen::Matrix& new_sqrt_inv_hess) + { + Eigen::Matrix inv_slack = (this->cons_b_ - this->cons_A_ * new_sample).cwiseInverse(); + + Eigen::Matrix half_hess = inv_slack.asDiagonal()* this->cons_A_; + Eigen::Matrix new_hess = half_hess.transpose() * half_hess; + + // compute eigenvectors and eigenvalues + Eigen::SelfAdjointEigenSolver > es(new_hess); + + Eigen::Matrix V = es.eigenvectors(); + Eigen::Matrix Dv = es.eigenvalues(); + new_sqrt_inv_hess = V * Dv.cwiseInverse().cwiseSqrt().asDiagonal() * V.transpose(); + } + +private: + Dtype r_; + + // Dimension + int nb_dim_; + // number of constraints + int nb_cons_; + // current sample size + int nb_curr_samples_; + // Initial vector + Eigen::Matrix initialization_; + // constraints matrix A + Eigen::Matrix cons_A_; + // constraint vector b + Eigen::Matrix cons_b_; + // Current vector + Eigen::Matrix curr_sample_; +}; + + +#endif // PWALK_DIKIN_WALKER_HPP_ diff --git a/src/volesti/include/random_walks/ellipsoid_walks/john_walker.h b/src/volesti/include/random_walks/ellipsoid_walks/john_walker.h new file mode 100644 index 00000000..fd1d25f9 --- /dev/null +++ b/src/volesti/include/random_walks/ellipsoid_walks/john_walker.h @@ -0,0 +1,204 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Original C++ code from https://github.com/rzrsk/vaidya-walk by Raaz Dwivedi. + +// Modified by Alexandros Manochis to be integrated in volesti, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// The implemented random walk is presented in the paper of +// Y. Chen, R. Dwivedi, M. J. Wainwright and B. Yu, +// "Fast MCMC Sampling Algorithms on Polytopes", +// Journal of Machine Learning Research, 2018. + +#ifndef PWALK_JOHN_WALKER_HPP_ +#define PWALK_JOHN_WALKER_HPP_ + +#include +#include +#include "math_functions.h" + + +/// @brief Class that defines the John walk sampler +/// @tparam Dtype Number Type +template +class JohnWalker { +public: + + JohnWalker() {} + + JohnWalker(const Eigen::Matrix &initialization, const Eigen::Matrix &cons_A, const Eigen::Matrix &cons_b, const Dtype r) + { + nb_dim_ = cons_A.cols(); + nb_cons_ = cons_A.rows(); + nb_curr_samples_ = 1; + initialization_ = initialization; + cons_A_ = cons_A; + cons_b_ = cons_b; + curr_sample_ = initialization; + r_ = r; + alpha_ = 1. - 1. / std::log2(2. * Dtype(cons_A.rows()) / Dtype(cons_A.cols())); + beta_ = Dtype(cons_A.cols()) / 2. / Dtype(cons_A.rows()); + curr_weight_ = Eigen::Matrix::Ones(cons_A.rows()); + } + + // getter for radius + Dtype getRadius() { + return r_; + } + + // check whether a given point is in th polytope + bool checkInPolytope(const Eigen::Matrix &new_sample) { + return (cons_A_ * new_sample - cons_b_).maxCoeff() < 0; + } + + // getter for dimension + int getNbDim() { + return nb_dim_; + } + + // getter for nb current samples + int getNbCurrSamples() { + return nb_curr_samples_; + } + + // getter for current sample + Eigen::Matrix &getCurrSample() { + return curr_sample_; + } + + void proposal(Eigen::Matrix &new_sample) { + Eigen::Matrix gaussian_step = Eigen::Matrix::Zero( + this->nb_dim_); + sample_gaussian(this->nb_dim_, 0., 1., gaussian_step); + + // get hessian + Eigen::Matrix new_sqrt_inv_hess = + Eigen::Matrix::Zero( + this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess); + + new_sample = this->curr_sample_ + r_ / std::sqrt(Dtype(this->nb_dim_)) * (new_sqrt_inv_hess * gaussian_step); + } + + bool acceptRejectReverse(const Eigen::Matrix &new_sample) { + // get hessian on y + Eigen::Matrix new_sqrt_inv_hess_y = + Eigen::Matrix::Zero( + this->nb_dim_, this->nb_dim_); + + sqrtInvHessBarrier(new_sample, new_sqrt_inv_hess_y); + // get hessian on x + Eigen::Matrix new_sqrt_inv_hess_x = + Eigen::Matrix::Zero( + this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess_x); + + Dtype scale = r_ / std::sqrt(Dtype(this->nb_dim_)); + Dtype p_y_to_x = gaussian_density(this->curr_sample_, new_sample, + new_sqrt_inv_hess_y.inverse() / scale); + Dtype p_x_to_y = gaussian_density(new_sample, this->curr_sample_, + new_sqrt_inv_hess_x.inverse() / scale); + + Dtype ar_ratio = std::min(1., p_y_to_x / p_x_to_y); + + Dtype random_num = rng_uniform(0., 1.); + // lazy version of the walk + if (random_num > ar_ratio) { + return false; + } + + return true; + } + + bool doSample(Eigen::Matrix &new_sample, const Dtype lazy = Dtype(0.5)) { + proposal(new_sample); + this->nb_curr_samples_ += 1; + // for lazy markov chain + Dtype random_num = rng_uniform(0., 1.); + // check balance and check in polytope + if (random_num < lazy && this->checkInPolytope(new_sample) && acceptRejectReverse(new_sample)) { + this->curr_sample_ = new_sample; + return true; + } else { + new_sample = this->curr_sample_; + return false; + } + } + + void sqrtInvHessBarrier(const Eigen::Matrix &new_sample, + Eigen::Matrix &new_sqrt_inv_hess) { + Eigen::Matrix inv_slack = (this->cons_b_ - this->cons_A_ * new_sample).cwiseInverse(); + + Eigen::Matrix half_hess = inv_slack.asDiagonal() * this->cons_A_; + + Eigen::Matrix gradient; + Eigen::Matrix score; + Eigen::Matrix weight_half_alpha; + Eigen::Matrix weight_half_hess; + Eigen::Matrix new_hess; + Eigen::Matrix new_hess_inv; + Eigen::Matrix beta_ones = + beta_ * Eigen::Matrix::Ones(this->nb_cons_); + + Eigen::Matrix next_weight = curr_weight_; + // compute scores using gradient descent + do { + curr_weight_ = next_weight; + weight_half_alpha = curr_weight_.array().pow(alpha_ / 2.); + + weight_half_hess = (weight_half_alpha.cwiseProduct(inv_slack)).asDiagonal() * this->cons_A_; + + new_hess = weight_half_hess.transpose() * weight_half_hess; + + new_hess_inv = new_hess.inverse(); + + score = ((weight_half_hess * new_hess_inv).cwiseProduct(weight_half_hess)).rowwise().sum(); + + next_weight = (0.5 * (curr_weight_ + score + beta_ones)).cwiseMax(beta_ones); + + } while ((next_weight - curr_weight_).template lpNorm() > Dtype(0.00001)); + + // compute john hessian + Eigen::Matrix john_new_hess = + half_hess.transpose() * curr_weight_.asDiagonal() * half_hess; + + // compute eigenvectors and eigenvalues + Eigen::SelfAdjointEigenSolver > es(john_new_hess); + + Eigen::Matrix V = es.eigenvectors(); + Eigen::Matrix Dv = es.eigenvalues(); + new_sqrt_inv_hess = V * Dv.cwiseInverse().cwiseSqrt().asDiagonal() * V.transpose(); + } + +private: + + Dtype alpha_; + Dtype beta_; + + Dtype r_; + + // Dimension + int nb_dim_; + // number of constraints + int nb_cons_; + // current sample size + int nb_curr_samples_; + // Initial vector + Eigen::Matrix initialization_; + // constraints matrix A + Eigen::Matrix cons_A_; + // constraint vector b + Eigen::Matrix cons_b_; + // Current vector + Eigen::Matrix curr_sample_; + + Eigen::Matrix curr_weight_; +}; + +#endif // PWALK_JOHN_WALKER_HPP_ + diff --git a/src/volesti/include/random_walks/ellipsoid_walks/math_functions.h b/src/volesti/include/random_walks/ellipsoid_walks/math_functions.h new file mode 100644 index 00000000..c1c9566f --- /dev/null +++ b/src/volesti/include/random_walks/ellipsoid_walks/math_functions.h @@ -0,0 +1,54 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Original C++ code from https://github.com/rzrsk/vaidya-walk by Raaz Dwivedi. + +// Modified by Alexandros Manochis to be integrated in volesti, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef PWALK_UTIL_MATH_FUNCTIONS_HPP_ +#define PWALK_UTIL_MATH_FUNCTIONS_HPP_ + +#include +#include +#include + +// Define random number generator type +typedef boost::mt19937 rng_t; + +template +void sample_gaussian(const int n, const Dtype a, + const Dtype sigma, Eigen::Matrix& r) { + static rng_t gen(1234567); + static boost::normal_distribution random_distribution(a, sigma); + static boost::variate_generator > + variate_generator(gen, random_distribution); + + for (int i = 0; i < n; ++i) { + r[i] = variate_generator(); + } +} + +template +Dtype rng_uniform(const Dtype a, const Dtype b) { + static rng_t gen(1234567); + static boost::uniform_real random_distribution(a, b); + static boost::variate_generator > + variate_generator(gen, random_distribution); + + return variate_generator(); +} + +template +Dtype gaussian_density(const Eigen::Matrix& x, + const Eigen::Matrix& mu, + const Eigen::Matrix& sqrt_cov) { + Eigen::Matrix c = sqrt_cov * (x - mu); + return std::exp(-0.5*c.dot(c)) * sqrt_cov.determinant(); +} + + +#endif // PWALK_UTIL_MATH_FUNTIONS_HPP_ diff --git a/src/volesti/include/random_walks/ellipsoid_walks/vaidya_walker.h b/src/volesti/include/random_walks/ellipsoid_walks/vaidya_walker.h new file mode 100644 index 00000000..6da8b5a7 --- /dev/null +++ b/src/volesti/include/random_walks/ellipsoid_walks/vaidya_walker.h @@ -0,0 +1,171 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Original C++ code from https://github.com/rzrsk/vaidya-walk by Raaz Dwivedi. + +// Modified by Alexandros Manochis to be integrated in volesti, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// The implemented random walk is presented in the paper of +// Y. Chen, R. Dwivedi, M. J. Wainwright and B. Yu, +// "Fast MCMC Sampling Algorithms on Polytopes", +// Journal of Machine Learning Research, 2018. + +#ifndef PWALK_VAIDYA_WALKER_HPP_ +#define PWALK_VAIDYA_WALKER_HPP_ + +#include +#include "math_functions.h" + + +/// @brief Class that defines the Vaidya walk sampler +/// @tparam Dtype Number Type +template +class VaidyaWalker { +public: + + VaidyaWalker() {} + + VaidyaWalker(const Eigen::Matrix &initialization, const Eigen::Matrix &cons_A, const Eigen::Matrix &cons_b, const Dtype r) + { + nb_dim_ = cons_A.cols(); + nb_cons_ = cons_A.rows(); + nb_curr_samples_ = 1; + initialization_ = initialization; + cons_A_ = cons_A; + cons_b_ = cons_b; + curr_sample_ = initialization; + r_ = r; + } + + // getter for radius + Dtype getRadius() { + return r_; + } + + // check whether a given point is in th polytope + bool checkInPolytope(const Eigen::Matrix &new_sample) { + return (cons_A_ * new_sample - cons_b_).maxCoeff() < 0; + } + + // getter for dimension + int getNbDim() { + return nb_dim_; + } + + // getter for nb current samples + int getNbCurrSamples() { + return nb_curr_samples_; + } + + // getter for current sample + Eigen::Matrix &getCurrSample() { + return curr_sample_; + } + + void proposal(Eigen::Matrix &new_sample) { + Eigen::Matrix gaussian_step = Eigen::Matrix::Zero( + this->nb_dim_); + sample_gaussian(this->nb_dim_, 0., 1., gaussian_step); + + // get hessian + Eigen::Matrix new_sqrt_inv_hess = Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess); + + new_sample = this->curr_sample_ + r_ / std::sqrt(std::sqrt(Dtype(this->nb_dim_) * Dtype(this->nb_cons_))) + * (new_sqrt_inv_hess * gaussian_step); + } + + bool acceptRejectReverse(const Eigen::Matrix &new_sample) { + // get hessian on x + Eigen::Matrix new_sqrt_inv_hess_x = Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(this->curr_sample_, new_sqrt_inv_hess_x); + // get hessian on y + Eigen::Matrix new_sqrt_inv_hess_y = Eigen::Matrix::Zero(this->nb_dim_, this->nb_dim_); + sqrtInvHessBarrier(new_sample, new_sqrt_inv_hess_y); + + Dtype scale = r_ / std::sqrt(std::sqrt(Dtype(this->nb_dim_) * Dtype(this->nb_cons_))); + Dtype p_y_to_x = gaussian_density(this->curr_sample_, new_sample, new_sqrt_inv_hess_y.inverse() / scale); + Dtype p_x_to_y = gaussian_density(new_sample, this->curr_sample_, new_sqrt_inv_hess_x.inverse() / scale); + + Dtype ar_ratio = std::min(1., p_y_to_x / p_x_to_y); + + Dtype random_num = rng_uniform(0., 1.); + // lazy version of the walk + if (random_num > ar_ratio) { + return false; + } + + return true; + } + + bool doSample(Eigen::Matrix &new_sample, const Dtype lazy = Dtype(0.5)) { + proposal(new_sample); + this->nb_curr_samples_ += 1; + // for lazy markov chain + Dtype random_num = rng_uniform(0., 1.); + // check balance and check in polytope + if (random_num < lazy && this->checkInPolytope(new_sample) && acceptRejectReverse(new_sample)) { + this->curr_sample_ = new_sample; + return true; + } else { + new_sample = this->curr_sample_; + return false; + } + } + + void sqrtInvHessBarrier(const Eigen::Matrix &new_sample, + Eigen::Matrix &new_sqrt_inv_hess) { + Eigen::Matrix inv_slack = (this->cons_b_ - this->cons_A_ * new_sample).cwiseInverse(); + + Eigen::Matrix half_hess = inv_slack.asDiagonal() * this->cons_A_; + Eigen::Matrix new_hess = half_hess.transpose() * half_hess; + + Eigen::Matrix new_hess_inv = new_hess.inverse(); + + // compute leverage scores + // Eigen::Matrix score = ((this->cons_A_ * new_hess_inv).cwiseProduct(this->cons_A_)).rowwise().sum().cwiseProduct(inv_slack).cwiseProduct(inv_slack); + Eigen::Matrix score = ((half_hess * new_hess_inv).cwiseProduct( + half_hess)).rowwise().sum(); + + // compute vaidya hessian + Eigen::Matrix vaidya_new_hess = + half_hess.transpose() * score.asDiagonal() * half_hess + + Dtype(this->nb_dim_) / Dtype(this->nb_cons_) * new_hess; + + // compute eigenvectors and eigenvalues + Eigen::SelfAdjointEigenSolver > es(vaidya_new_hess); + + Eigen::Matrix V = es.eigenvectors(); + Eigen::Matrix Dv = es.eigenvalues(); + new_sqrt_inv_hess = V * Dv.cwiseInverse().cwiseSqrt().asDiagonal() * V.transpose(); + } + +private: + Dtype r_; + + // Dimension + int nb_dim_; + // number of constraints + int nb_cons_; + // current sample size + int nb_curr_samples_; + // Initial vector + Eigen::Matrix initialization_; + // constraints matrix A + Eigen::Matrix cons_A_; + // constraint vector b + Eigen::Matrix cons_b_; + // Current vector + Eigen::Matrix curr_sample_; +}; + +#endif // PWALK_VAIDYA_WALKER_HPP_ + diff --git a/src/volesti/include/random_walks/exponential_hamiltonian_monte_carlo_exact_walk.hpp b/src/volesti/include/random_walks/exponential_hamiltonian_monte_carlo_exact_walk.hpp new file mode 100644 index 00000000..6e5d2e00 --- /dev/null +++ b/src/volesti/include/random_walks/exponential_hamiltonian_monte_carlo_exact_walk.hpp @@ -0,0 +1,300 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_EXPONENTIAL_EXACT_HMC_WALK_HPP +#define RANDOM_WALKS_EXPONENTIAL_EXACT_HMC_WALK_HPP + +#define INSIDE_BODY_TOLLERANCE 1e-10 + +#include "sampling/sphere.hpp" + + +// Exact HMC for sampling from the Exponential distribution restricted to a convex polytope +struct ExponentialHamiltonianMonteCarloExactWalk +{ + ExponentialHamiltonianMonteCarloExactWalk(double L) + : param(L, true, 0, false) + {} + + ExponentialHamiltonianMonteCarloExactWalk(double L, unsigned int rho) + : param(L, true, rho, true) + {} + + ExponentialHamiltonianMonteCarloExactWalk() + : param(0, false, 0, false) + {} + + struct parameters + { + parameters(double L, bool set, unsigned int _rho, bool _set_rho) + : m_L(L), set_L(set), rho(_rho), set_rho(_set_rho) + {} + double m_L; + bool set_L; + unsigned int rho; + bool set_rho; + }; + + parameters param; + + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + + template + Walk(GenericPolytope &P, Point const& p, Point const& c, NT const& T, RandomNumberGenerator &rng) + { + _Len = compute_diameter + ::template compute(P); + _Ac = P.get_mat() * c.getCoefficients(); + _Temp = T; + _c = c; + _rho = 100 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, rng); + } + + template + Walk(GenericPolytope &P, Point const& p, Point const& c, NT const& T, RandomNumberGenerator &rng, + parameters const& params) + { + _Len = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + _Ac = P.get_mat() * c.getCoefficients(); + _Temp = T; + _c = c; + _rho = 100 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, rng); + } + + template + < + typename GenericPolytope + > + inline bool apply(GenericPolytope& P, + Point& p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + int failures = 0, it; + Point p0 = _p; + + for (auto j=0u; j::apply(n, rng, false); + + it = 0; + while (it < _rho) + { + auto pbpair = P.quadratic_positive_intersect(_p, _v, _Ac, _Temp, _lambdas, + _Av, _lambda_prev, _facet_prev); + if (T <= pbpair.first) { + _p += ((T * T) / (-2.0*_Temp)) *_c + (T * _v); + _lambda_prev = T; + break; + } + _lambda_prev = pbpair.first; + _p += ((_lambda_prev * _lambda_prev) / (-2.0*_Temp)) *_c + (_lambda_prev * _v); + T -= _lambda_prev; + _v += (-_lambda_prev/_Temp) * _c; + P.compute_reflection(_v, _p, pbpair.second); + it++; + } + + } while (P.is_in(_p, INSIDE_BODY_TOLLERANCE) == 0); + if (it == _rho) { + _p = p0; + } + } + p = _p; + return true; + } + + + template + < + typename GenericPolytope + > + inline void get_starting_point(GenericPolytope& P, + Point const& center, + Point &q, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT radius = P.InnerBall().second; + + q = GetPointInDsphere::apply(n, radius, rng); + q += center; + initialize(P, q, rng); + + apply(P, q, walk_length, rng); + } + + + template + < + typename GenericPolytope + > + inline void parameters_burnin(GenericPolytope& P, + Point const& center, + unsigned int const& num_points, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + Point p(n); + std::vector pointset; + pointset.push_back(center); + pointset.push_back(_p); + NT rad = NT(0), max_dist, Lmax = NT(0), radius = P.InnerBall().second; + + for (int i = 0; i < num_points; i++) + { + p = GetPointInDsphere::apply(n, radius, rng); + p += center; + initialize(P, p, rng); + + apply(P, p, walk_length, rng); + max_dist = get_max_distance(pointset, p, rad); + if (max_dist > Lmax) + { + Lmax = max_dist; + } + if (2.0 * rad > Lmax) + { + Lmax = 2.0 * rad; + } + pointset.push_back(p); + } + + if (Lmax > _Len) + { + update_delta(Lmax); + } + pointset.clear(); + } + + + inline void update_delta(NT L) + { + _Len = L; + } + + +private : + + template + < + typename GenericPolytope + > + inline void initialize(GenericPolytope& P, + Point const& p, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + _lambdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + _v = GetDirection::apply(n, rng, false); + + do { + _p = p; + T = rng.sample_urdist() * _Len; + int it = 0; + + std::pair pbpair + = P.quadratic_positive_intersect(_p, _v, _Ac, _Temp, _lambdas, _Av, _facet_prev); + if (T <= pbpair.first || pbpair.second < 0) { + _p += ((T * T) / (-2.0*_Temp)) *_c + (T * _v); + _lambda_prev = T; + return; + } + _lambda_prev = pbpair.first; + _p += ((_lambda_prev * _lambda_prev) / (-2.0*_Temp)) *_c + (_lambda_prev * _v); + _v += (-_lambda_prev/_Temp) * _c; + T -= _lambda_prev; + P.compute_reflection(_v, _p, pbpair.second); + + while (it <= _rho) + { + std::pair pbpair + = P.quadratic_positive_intersect(_p, _v, _Ac, _Temp, _lambdas, _Av, _lambda_prev, _facet_prev); + if (T <= pbpair.first || pbpair.second < 0) { + _p += ((T * T) / (-2.0*_Temp)) *_c + (T * _v); + _lambda_prev = T; + break; + } else if (it == _rho) { + _lambda_prev = rng.sample_urdist() * pbpair.first; + _p += ((_lambda_prev * _lambda_prev) / (-2.0*_Temp)) *_c + (_lambda_prev * _v); + break; + } + _lambda_prev = pbpair.first; + _p += ((_lambda_prev * _lambda_prev) / (-2.0*_Temp)) *_c + (_lambda_prev * _v); + _v += (-_lambda_prev/_Temp) * _c; + T -= _lambda_prev; + P.compute_reflection(_v, _p, pbpair.second); + it++; + } + } while (P.is_in(_p, INSIDE_BODY_TOLLERANCE) == 0); + } + + + inline double get_max_distance(std::vector &pointset, Point const& q, double &rad) + { + double dis = -1.0, temp_dis; + int jj = 0; + for (auto vecit = pointset.begin(); vecit!=pointset.end(); vecit++, jj++) + { + temp_dis = (q.getCoefficients() - (*vecit).getCoefficients()).norm(); + if (temp_dis > dis) { + dis = temp_dis; + } + if (jj == 0 && temp_dis > rad) { + rad = temp_dis; + } + } + return dis; + } + + + NT _Len; + VT _Ac; + Point _p; + Point _v; + Point _c; + NT _Temp; + NT _lambda_prev; + int _facet_prev; + unsigned int _rho; + typename Point::Coeff _lambdas; + typename Point::Coeff _Av; +}; + +}; + + +#endif // RANDOM_WALKS_EXPONENTIAL_HMC_WALK_HPP + diff --git a/src/volesti/include/random_walks/gaussian_accelerated_billiard_walk.hpp b/src/volesti/include/random_walks/gaussian_accelerated_billiard_walk.hpp new file mode 100644 index 00000000..fbf43f44 --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_accelerated_billiard_walk.hpp @@ -0,0 +1,248 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +// Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_GAUSSIAN_ACCELERATED_BILLIARD_WALK_HPP +#define RANDOM_WALKS_GAUSSIAN_ACCELERATED_BILLIARD_WALK_HPP + +#include "convex_bodies/orderpolytope.h" +#include "convex_bodies/ellipsoid.h" +#include "convex_bodies/ballintersectconvex.h" +#include "generators/boost_random_number_generator.hpp" +#include "sampling/ellipsoid.hpp" +#include "random_walks/uniform_billiard_walk.hpp" + +#include "random_walks/compute_diameter.hpp" + +// Billiard walk which accelarates each step for uniform distribution and also takes into account +// the shape of the polytope for generating directions. + +struct GaussianAcceleratedBilliardWalk +{ + GaussianAcceleratedBilliardWalk(double L) + : param(L, true) + {} + + GaussianAcceleratedBilliardWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + struct update_parameters + { + update_parameters() + : facet_prev(0), hit_ball(false), inner_vi_ak(0.0), ball_inner_norm(0.0) + {} + int facet_prev; + bool hit_ball; + double inner_vi_ak; + double ball_inner_norm; + }; + + parameters param; + + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + // typedef typename Polytope::MT MT; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, + Point const& p, + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + RandomNumberGenerator &rng) + { + _update_parameters = update_parameters(); + _Len = compute_diameter + ::template compute(P); + + // Removed as will be used for sparse matrices only + // _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + initialize(P, p, E, rng); + } + + template + Walk(GenericPolytope& P, + Point const& p, + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + RandomNumberGenerator &rng, + parameters const& params) + { + _update_parameters = update_parameters(); + _Len = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + + // Removed as will be used for sparse matrices only + // _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + initialize(P, p, E, rng); + } + + template + < + typename GenericPolytope, + typename Ellipsoid + > + inline void apply(GenericPolytope& P, + Point &p, // a point to return the result + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + const NT dl = 0.995; + int it; + + for (auto j=0u; j::apply(n, E, rng); + NT norm_v = _v.length(); + _v /= norm_v; + + Point p0 = _p; + Point v0 = _v; + typename Point::Coeff lambdas0 = _lambdas; + typename Point::Coeff Av0 = _Av; + NT lambda_prev0 = _lambda_prev; + + it = 0; + while (it < 100*n) + { + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av, _lambda_prev, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + it++; + } + if (it == 100*n) + { + _p = p0; + _lambdas = lambdas0; + _Av = Av0; + _lambda_prev = lambda_prev0; + } + else + { + // metropolis filter + NT u_logprob = log(rng.sample_urdist()); + NT accept_logprob = 0.5 * (norm_v * norm_v) * (E.mat_mult(v0) - E.mat_mult(_v)); + // std::cout << "diff: " << (accept_logprob - u_logprob) << std::endl; + if(u_logprob > accept_logprob) { + // reject + _p = p0; + _lambdas = lambdas0; + _Av = Av0; + _lambda_prev = lambda_prev0; + } + } + } + p = _p; + } + + inline void update_delta(NT L) + { + _Len = L; + } + + private : + + template + < + typename GenericPolytope, + typename Ellipsoid + > + inline void initialize(GenericPolytope& P, + Point const& p, // a point to start + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + const NT dl = 0.995; + _lambdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + _p = p; + _v = GetGaussianDirection::apply(n, E, rng); + NT norm_v = _v.length(); + _v /= norm_v; + + NT T = -std::log(rng.sample_urdist()) * _Len; + Point p0 = _p; + int it = 0; + + std::pair pbpair + = P.line_first_positive_intersect(_p, _v, _lambdas, _Av, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + return; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + + while (it <= 100*n) + { + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av, _lambda_prev, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + } else if (it == 100*n) { + _lambda_prev = rng.sample_urdist() * pbpair.first; + _p += (_lambda_prev * _v); + break; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + it++; + } + } + + NT _Len; + Point _p; + Point _v; + NT _lambda_prev; + // MT _AA; // Removed as will be used for sparse matrices only + update_parameters _update_parameters; + typename Point::Coeff _lambdas; + typename Point::Coeff _Av; + }; + +}; + + +#endif diff --git a/src/volesti/include/random_walks/gaussian_ball_walk.hpp b/src/volesti/include/random_walks/gaussian_ball_walk.hpp new file mode 100644 index 00000000..8c267e0f --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_ball_walk.hpp @@ -0,0 +1,107 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_GAUSSIAN_BALL_WALK_HPP +#define RANDOM_WALKS_GAUSSIAN_BALL_WALK_HPP + +#include "sampling/sphere.hpp" +#include "random_walks/gaussian_helpers.hpp" + +// Ball walk with spherical Gaussian target distribution + +struct GaussianBallWalk +{ + + GaussianBallWalk(double L) + : param(L, true) + {} + + GaussianBallWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_delta(set) + {} + double m_L; + bool set_delta; + }; + + parameters param; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + static inline NT compute_delta(GenericPolytope& P, NT const& a) + { + //return ((P.InnerBall()).second * NT(4)) / NT(P.dimension()); + return (NT(4) * (P.InnerBall()).second) / std::sqrt(std::max(NT(1), a) * NT(P.dimension())); + } + + Walk (Polytope& P, Point const& p, NT const& a, + RandomNumberGenerator &rng) + { + _delta = compute_delta(P, a); + } + + Walk (Polytope& P, + Point const& p, + NT const& a, + RandomNumberGenerator &rng, + parameters const& params) + { + _delta = params.set_delta ? params.m_L + : compute_delta(P, a); + } + + template + inline void apply(BallPolytope const& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + Point y = GetPointInDsphere::apply(P.dimension(), + _delta, + rng); + y += p; + if (P.is_in(y) == -1) + { + NT f_x = eval_exp(p, a_i); + NT f_y = eval_exp(y, a_i); + NT rnd = rng.sample_urdist(); + if (rnd <= f_y / f_x) { + p = y; + } + } + } + } + + inline void update_delta(NT delta) + { + _delta = delta; + } + +private : + NT _delta; +}; + +}; + +#endif // RANDOM_WALKS_GAUSSIAN_BALL_WALK_HPP diff --git a/src/volesti/include/random_walks/gaussian_cdhr_walk.hpp b/src/volesti/include/random_walks/gaussian_cdhr_walk.hpp new file mode 100644 index 00000000..04ebedbc --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_cdhr_walk.hpp @@ -0,0 +1,151 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_GAUSSIAN_CDHR_WALK_HPP +#define RANDOM_WALKS_GAUSSIAN_CDHR_WALK_HPP + +#include "sampling/sphere.hpp" +#include "generators/boost_random_number_generator.hpp" +#include "random_walks/gaussian_helpers.hpp" + +// Pick a point from the distribution exp(-a_i||x||^2) on the coordinate chord +template +< + typename NT, + typename RandomNumberGenerator +> +NT chord_random_point_generator_exp_coord(const NT &l, + const NT &u, + const NT &a_i, + RandomNumberGenerator& rng) +{ + NT r, r_val, fn, dis; + // pick from 1-dimensional gaussian if enough weight is inside polytope P + if (a_i > EXP_CHORD_TOLERENCE && u - l >= 2.0 / std::sqrt(2.0 * a_i)) + { + while (true) { + r = rng.sample_ndist();//rdist(rng2); + r = r / std::sqrt(2.0 * a_i); + if (r >= l && r <= u) { + break; + } + } + dis = r; + + // select using rejection sampling from a bounding rectangle + } + else + { + NT M = get_max_coord(l, u, a_i); + while (true) + { + r = rng.sample_urdist(); + dis = (1.0 - r) * l + r * u; + r_val = M * rng.sample_urdist(); + fn = std::exp(-a_i * dis * dis); + if (r_val < fn) + { + break; + } + } + } + return dis; +} + + +// Coordinate directions hit-and-run walk with spherical Gaussian target distribution + +struct GaussianCDHRWalk +{ + + struct parameters {}; + parameters param; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + Walk(Polytope& P, + Point const& p, + NT const& a_i, + RandomNumberGenerator &rng) + { + initialize(P, p, a_i, rng); + } + + Walk(Polytope& P, + Point const& p, + NT const& a_i, + RandomNumberGenerator& rng, + parameters&) + { + initialize(P, p, a_i, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + auto rand_coord_prev = _rand_coord; + _rand_coord = rng.sample_uidist(); + std::pair bpair = + P.line_intersect_coord(_p, _p_prev, _rand_coord, + rand_coord_prev, _lamdas); + NT dis = chord_random_point_generator_exp_coord + (_p[_rand_coord] + bpair.second, + _p[_rand_coord] + bpair.first, + a_i, + rng); + _p_prev = _p; + _p.set_coord(_rand_coord, dis); + } + p = _p; + } + +private : + + template + inline void initialize(BallPolytope const& P, + Point const& p, + NT const& a_i, + RandomNumberGenerator &rng) + { + _lamdas.setZero(P.num_of_hyperplanes()); + _rand_coord = rng.sample_uidist(); + _p = p; + std::pair bpair = P.line_intersect_coord(_p, _rand_coord, _lamdas); + NT dis = chord_random_point_generator_exp_coord + (_p[_rand_coord] + bpair.second, + _p[_rand_coord] + bpair.first, + a_i, rng); + _p_prev = p; + _p.set_coord(_rand_coord, dis); + } + + unsigned int _rand_coord; + Point _p; + Point _p_prev; + typename Point::Coeff _lamdas; +}; + +}; + +#endif // RANDOM_WALKS_GAUSSIAN_CDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/gaussian_hamiltonian_monte_carlo_exact_walk.hpp b/src/volesti/include/random_walks/gaussian_hamiltonian_monte_carlo_exact_walk.hpp new file mode 100644 index 00000000..8b83b053 --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_hamiltonian_monte_carlo_exact_walk.hpp @@ -0,0 +1,275 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_GAUSSIAN_EXACT_HMC_WALK_HPP +#define RANDOM_WALKS_GAUSSIAN_EXACT_HMC_WALK_HPP + +#include "sampling/sphere.hpp" + + + +// Exact HMC for sampling from the spherical Gaussian distribution + +struct GaussianHamiltonianMonteCarloExactWalk +{ + GaussianHamiltonianMonteCarloExactWalk(double L, unsigned int _rho) + : param(L, true, _rho, true) + {} + + GaussianHamiltonianMonteCarloExactWalk(double L) + : param(L, true, 0, false) + {} + + GaussianHamiltonianMonteCarloExactWalk() + : param(0, false, 0, false) + {} + + + struct parameters + { + parameters(double L, bool set, unsigned int _rho, bool _set_rho) + : m_L(L), set_L(set), rho(_rho), set_rho(_set_rho) + {} + double m_L; + bool set_L; + unsigned int rho; + bool set_rho; + }; + + parameters param; + + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef typename Polytope::VT VT; + + template + Walk(GenericPolytope &P, Point const& p, NT const& a_i, RandomNumberGenerator &rng) + { + _Len = compute_diameter + ::template compute(P); + _omega = std::sqrt(NT(2) * a_i); + _rho = 100 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, a_i, rng); + } + + template + Walk(GenericPolytope &P, Point const& p, NT const& a_i, RandomNumberGenerator &rng, + parameters const& params) + { + _Len = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + _omega = std::sqrt(NT(2) * a_i); + _rho = 100 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, a_i, rng); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope& P, + Point& p, + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + + for (auto j=0u; j::apply(n, rng, false); + Point p0 = _p; + int it = 0; + while (it < _rho) + { + auto pbpair = P.trigonometric_positive_intersect(_p, _v, _omega, _facet_prev); + if (T <= pbpair.first) { + update_position(_p, _v, T, _omega); + break; + } + _lambda_prev = pbpair.first; + T -= _lambda_prev; + update_position(_p, _v, _lambda_prev, _omega); + P.compute_reflection(_v, _p, pbpair.second); + it++; + } + if (it == _rho){ + _p = p0; + } + } + p = _p; + } + + + template + < + typename GenericPolytope + > + inline void get_starting_point(GenericPolytope& P, + Point const& center, + Point &q, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT radius = P.InnerBall().second; + + q = GetPointInDsphere::apply(n, radius, rng); + q += center; + initialize(P, q, rng); + + apply(P, q, walk_length, rng); + } + + + template + < + typename GenericPolytope + > + inline void parameters_burnin(GenericPolytope& P, + Point const& center, + unsigned int const& num_points, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + Point p(n); + std::vector pointset; + pointset.push_back(center); + pointset.push_back(_p); + NT rad = NT(0), max_dist, Lmax = NT(0), radius = P.InnerBall().second; + + for (int i = 0; i < num_points; i++) + { + p = GetPointInDsphere::apply(n, radius, rng); + p += center; + initialize(P, p, rng); + + apply(P, p, walk_length, rng); + max_dist = get_max_distance(pointset, p, rad); + if (max_dist > Lmax) + { + Lmax = max_dist; + } + if (2.0*rad > Lmax) + { + Lmax = 2.0 * rad; + } + pointset.push_back(p); + } + + if (Lmax > _Len) + { + update_delta(Lmax); + } + pointset.clear(); + } + + inline void update_delta(NT L) + { + _Len = L; + } + +private : + + template + < + typename GenericPolytope + > + inline void initialize(GenericPolytope& P, + Point const& p, + NT const& a_i, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + _facet_prev = -1; + _p = p; + _v = GetDirection::apply(n, rng, false); + + NT T = rng.sample_urdist() * _Len; + int it = 0; + + while (it <= _rho) + { + auto pbpair + = P.trigonometric_positive_intersect(_p, _v, _omega, _facet_prev); + if (T <= pbpair.first) { + update_position(_p, _v, T, _omega); + break; + } else if (it == _rho) { + _lambda_prev = rng.sample_urdist() * pbpair.first; + update_position(_p, _v, _lambda_prev, _omega); + break; + } + _lambda_prev = pbpair.first; + update_position(_p, _v, _lambda_prev, _omega); + T -= _lambda_prev; + P.compute_reflection(_v, _p, pbpair.second); + it++; + } + } + + inline void update_position(Point &p, Point &v, NT const& T, NT const& omega) + { + NT C, Phi; + for (size_t i = 0; i < p.dimension(); i++) + { + C = std::sqrt(p[i] * p[i] + (v[i] * v[i]) / (omega * omega)); + Phi = std::atan((-v[i]) / (p[i] * omega)); + if (v[i] < 0.0 && Phi < 0.0) { + Phi += M_PI; + } else if (v[i] > 0.0 && Phi > 0.0) { + Phi -= M_PI; + } + p.set_coord(i, C * std::cos(omega * T + Phi)); + v.set_coord(i, -C * omega * std::sin(omega * T + Phi)); + } + + } + + inline double get_max_distance(std::vector &pointset, Point const& q, double &rad) + { + double dis = -1.0, temp_dis; + int jj = 0; + for (auto vecit = pointset.begin(); vecit!=pointset.end(); vecit++, jj++) + { + temp_dis = (q.getCoefficients() - (*vecit).getCoefficients()).norm(); + if (temp_dis > dis) { + dis = temp_dis; + } + if (jj == 0 && temp_dis > rad) { + rad = temp_dis; + } + } + return dis; + } + + int _facet_prev; + unsigned int _rho; + NT _Len; + Point _p; + Point _v; + NT _omega; + NT _lambda_prev; +}; + +}; + + +#endif // RANDOM_WALKS_GAUSSIAN_HMC_WALK_HPP + diff --git a/src/volesti/include/random_walks/gaussian_helpers.hpp b/src/volesti/include/random_walks/gaussian_helpers.hpp new file mode 100644 index 00000000..d1ab5429 --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_helpers.hpp @@ -0,0 +1,48 @@ +#ifndef GAUSSIAN_HELPERS_HPP +#define GAUSSIAN_HELPERS_HPP + +#define EXP_CHORD_TOLERENCE 0.00000001 + +// evaluate the pdf of point p +template +NT eval_exp(Point const& p, NT const& a) +{ + return std::exp(-a * p.squared_length()); +} + + +template +NT get_max(Point const& l, Point const& u, NT const& a_i) +{ + NT res; + Point a = -1.0 * l; + Point bef = u - l; + Point b = (1.0 / std::sqrt((bef).squared_length())) * bef; + Point z = (a.dot(b) * b) + l; + NT low_bd = (l[0] - z[0]) / b[0], up_bd = (u[0] - z[0]) / b[0]; + if (low_bd * up_bd > 0) + { + //if(std::signbit(low_bd)==std::signbit(up_bd)){ + res = std::max(eval_exp(u, a_i), eval_exp(l, a_i)); + } + else + { + res = eval_exp(z, a_i); + } + + return res; +} + + +template +NT get_max_coord(NT const& l, NT const& u, NT const& a_i) +{ + const NT zero = 0; + if (l < zero && u > zero) + { + return NT(1); + } + return std::max(std::exp(-a_i * l * l), std::exp(-a_i * u * u)); +} + +#endif // GAUSSIAN_HELPERS_HPP diff --git a/src/volesti/include/random_walks/gaussian_rdhr_walk.hpp b/src/volesti/include/random_walks/gaussian_rdhr_walk.hpp new file mode 100644 index 00000000..b7a53abe --- /dev/null +++ b/src/volesti/include/random_walks/gaussian_rdhr_walk.hpp @@ -0,0 +1,118 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_GAUSSIAN_RDHR_WALK_HPP +#define RANDOM_WALKS_GAUSSIAN_RDHR_WALK_HPP + +#include "random_walks/gaussian_helpers.hpp" +#include "generators/boost_random_number_generator.hpp" + + +// Pick a point from the distribution exp(-a_i||x||^2) on the chord +template +< + typename Point, + typename NT, + typename RandomNumberGenerator +> +void chord_random_point_generator_exp(Point &lower, + Point & upper, + const NT &a_i, + Point &p, + RandomNumberGenerator& rng) +{ + NT r, r_val, fn; + Point bef = upper - lower; + // pick from 1-dimensional gaussian if enough weight is inside polytope P + if (a_i > EXP_CHORD_TOLERENCE && std::sqrt(bef.squared_length()) >= (2.0 / std::sqrt(2.0 * a_i))) + { + Point a = -1.0 * lower; + Point b = (1.0 / std::sqrt(bef.squared_length())) * bef; + Point z = (a.dot(b) * b) + lower; + NT low_bd = (lower[0] - z[0]) / b[0]; + NT up_bd = (upper[0] - z[0]) / b[0]; + while (true) { + r = rng.sample_ndist();//rdist(rng2); + r = r / std::sqrt(2.0 * a_i); + if (r >= low_bd && r <= up_bd) { + break; + } + } + p = (r * b) + z; + + // select using rejection sampling from a bounding rectangle + } else { + NT M = get_max(lower, upper, a_i); + while (true) { + r = rng.sample_urdist();//urdist(rng2); + Point pef = r * upper; + p = ((1.0 - r) * lower) + pef; + r_val = M * rng.sample_urdist();//urdist(var.rng); + fn = eval_exp(p, a_i); + if (r_val < fn) { + break; + } + } + } +} + +// Random directions hit-and-run walk with spherical Gaussian target distribution + +struct GaussianRDHRWalk +{ + + struct parameters {}; + parameters param; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + Walk(Polytope&, Point const&, NT const&, RandomNumberGenerator&) + {} + + Walk(Polytope&, Point const&, NT const&, RandomNumberGenerator&, + parameters&) + {} + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + Point v = GetDirection::apply(p.dimension(), rng); + std::pair dbpair = P.line_intersect(p, v); + + NT min_plus = dbpair.first; + NT max_minus = dbpair.second; + Point upper = (min_plus * v) + p; + Point lower = (max_minus * v) + p; + + chord_random_point_generator_exp(lower, upper, a_i, p, rng); + } + } +}; + +}; + + +#endif // RANDOM_WALKS_GAUSSIAN_RDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/hamiltonian_monte_carlo_walk.hpp b/src/volesti/include/random_walks/hamiltonian_monte_carlo_walk.hpp new file mode 100644 index 00000000..23342ea5 --- /dev/null +++ b/src/volesti/include/random_walks/hamiltonian_monte_carlo_walk.hpp @@ -0,0 +1,179 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Lee, Yin Tat, Ruoqi Shen, and Kevin Tian. "Logsmooth Gradient Concentration +// and Tighter Runtimes for Metropolized Hamiltonian Monte Carlo." +#ifndef HAMILTONIAN_MONTE_CARLO_WALK_HPP +#define HAMILTONIAN_MONTE_CARLO_WALK_HPP + + +#include "generators/boost_random_number_generator.hpp" +#include "random_walks/gaussian_helpers.hpp" +#include "ode_solvers/ode_solvers.hpp" + +struct HamiltonianMonteCarloWalk { + + template + < + typename NT, + typename OracleFunctor + > + struct parameters { + NT epsilon; // tolerance in mixing + NT eta; // step size + + parameters( + OracleFunctor const& F, + unsigned int dim, + NT epsilon_=2) + { + epsilon = epsilon_; + eta = 1.0 / (dim * sqrt(F.params.L)); + // eta = 1.0 / + // (sqrt(20 * F.params.L * pow(dim, 3))); + } + }; + + template + < + typename Point, + typename Polytope, + typename RandomNumberGenerator, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Solver + > + struct Walk { + + typedef std::vector pts; + typedef typename Point::FT NT; + typedef std::vector bounds; + + // Hyperparameters of the sampler + parameters ¶ms; + + // Numerical ODE solver + Solver *solver; + + // Dimension + unsigned int dim; + + // Discarded Samples + long total_discarded_samples = 0; + long num_runs = 0; + float discard_ratio = 0; + + // Average acceptance probability + float total_acceptance_log_prob = 0; + float average_acceptance_log_prob = 0; + + // References to xs + Point x, v; + + // Proposal points + Point x_tilde, v_tilde; + + // Gradient function + NegativeGradientFunctor &F; + + bool accepted; + + // Helper variables + NT H, H_tilde, log_prob, u_logprob; + + // Density exponent + NegativeLogprobFunctor &f; + + Walk(Polytope *P, + Point &p, + NegativeGradientFunctor &neg_grad_f, + NegativeLogprobFunctor &neg_logprob_f, + parameters ¶m) : + params(param), + F(neg_grad_f), + f(neg_logprob_f) + { + + dim = p.dimension(); + + // Starting point is provided from outside + x = p; + + accepted = false; + + // Initialize solver + solver = new Solver(0, params.eta, pts{x, x}, F, bounds{P, NULL}); + + }; + + inline void apply(RandomNumberGenerator &rng, + int walk_length=1, + bool metropolis_filter=true) + { + num_runs++; + + // Pick a random velocity + v = GetDirection::apply(dim, rng, false); + + solver->set_state(0, x); + solver->set_state(1, v); + + // Get proposals + solver->steps(walk_length, accepted); + x_tilde = solver->get_state(0); + v_tilde = solver->get_state(1); + + if (metropolis_filter) { + // Calculate initial Hamiltonian + H = hamiltonian(x, v); + + // Calculate new Hamiltonian + H_tilde = hamiltonian(x_tilde, v_tilde); + + // Log-sum-exp trick + log_prob = H - H_tilde < 0 ? H - H_tilde : 0; + + // Decide to switch + u_logprob = log(rng.sample_urdist()); + total_acceptance_log_prob += log_prob; + if (u_logprob < log_prob) { + x = x_tilde; + accepted = true; + } + else { + total_discarded_samples++; + accepted = false; + } + } else { + x = x_tilde; + accepted = true; + } + + discard_ratio = (1.0 * total_discarded_samples) / num_runs; + average_acceptance_log_prob = total_acceptance_log_prob / num_runs; + + } + + inline NT hamiltonian(Point &pos, Point &vel) const { + return f(pos) + 0.5 * vel.dot(vel); + } + + void disable_adaptive() { + solver->disable_adaptive(); + } + + void enable_adaptive() { + solver->enable_adaptive(); + } + }; +}; + +#endif // HAMILTONIAN_MONTE_CARLO_WALK_HPP diff --git a/src/volesti/include/random_walks/langevin_walk.hpp b/src/volesti/include/random_walks/langevin_walk.hpp new file mode 100644 index 00000000..df23dbac --- /dev/null +++ b/src/volesti/include/random_walks/langevin_walk.hpp @@ -0,0 +1,153 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Shen, Ruoqi, and Yin Tat Lee. "The randomized midpoint method for +// log-concave sampling." Advances in Neural Information Processing Systems. 2019. + +#ifndef LANGEVIN_WALK_HPP +#define LANGEVIN_WALK_HPP + +#include "generators/boost_random_number_generator.hpp" +#include "random_walks/gaussian_helpers.hpp" + +struct UnderdampedLangevinWalk { + + template + < + typename NT, + typename OracleFunctor + > + struct parameters { + NT epsilon; // tolerance in mixing + NT eta; // step size + NT u; + parameters( + OracleFunctor const& F, + unsigned int dim, + NT epsilon_=1e-4) + { + epsilon = epsilon_; + u = 1.0 / F.params.L; + eta = 1.0 / (sqrt(20 * F.params.L)); + + // eta = std::min(pow(epsilon, 1.0 / 3) / + // pow(F.params.kappa, 1.0 / 6) * + // pow(log(1.0 / epsilon), - 1.0 / 6), + // pow(epsilon, 2.0 / 3) * + // pow(log(1.0 / epsilon), - 1.0 / 3)); + } + }; + + template + < + typename Point, + typename Polytope, + typename RandomNumberGenerator, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Solver_ // Dummy template argument + > + struct Walk { + + typedef std::vector pts; + typedef typename Point::FT NT; + typedef std::vector bounds; + typedef RandomizedMipointSDESolver Solver; + + parameters ¶ms; + + // Numerical ODE solver + Solver *solver; + + unsigned int dim; + + // References to xs + Point x, v; + // Proposal points + Point x_tilde, v_tilde; + + // Gradient Function + NegativeGradientFunctor &F; + + // Density exponent + NegativeLogprobFunctor &f; + + NT H_tilde, H, log_prob, u_logprob; + + Walk(Polytope *P, + Point &initial_x, + NegativeGradientFunctor &neg_grad_f, + NegativeLogprobFunctor &neg_logprob_f, + parameters ¶m) : + params(param), + F(neg_grad_f), + f(neg_logprob_f) + { + // Starting point is provided from outside + x = initial_x; + dim = initial_x.dimension(); + v = Point(dim); + + solver = new Solver(0, params.eta, pts{x, v}, F, bounds{P, NULL}, params.u); + + }; + + inline void apply( + RandomNumberGenerator &rng, + int walk_length=1, + bool metropolis_filter=false) + { + solver->set_state(0, x); + solver->set_state(1, v); + + // Get proposals + solver->steps(walk_length, rng); + x_tilde = solver->get_state(0); + v_tilde = solver->get_state(1); + + if (metropolis_filter) { + // Calculate initial Hamiltonian + H = hamiltonian(x, v); + + // Calculate new Hamiltonian + H_tilde = hamiltonian(x_tilde, v_tilde); + + // Log-sum-exp trick + log_prob = H - H_tilde < 0 ? H - H_tilde : 0; + + // Decide to switch + u_logprob = log(rng.sample_urdist()); + if (u_logprob < log_prob) { + x = x_tilde; + v = v_tilde; + } + } else { + x = x_tilde; + v = v_tilde; + } + + } + + inline NT hamiltonian(Point &pos, Point &vel) const { + return f(pos) + 1.0 / (2 * params.u) * vel.dot(vel); + } + + void disable_adaptive() { + // TODO Implement + } + + void enable_adaptive() { + // TODO Implement + } + }; +}; + +#endif // LANGEVIN_WALK_HPP diff --git a/src/volesti/include/random_walks/multithread_walks.hpp b/src/volesti/include/random_walks/multithread_walks.hpp new file mode 100644 index 00000000..c42fba1f --- /dev/null +++ b/src/volesti/include/random_walks/multithread_walks.hpp @@ -0,0 +1,739 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Konstantinos Pallikaris, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +// THIS IS A TEMPORAL FILE. IT CONTAINS A FEW PROTOTYPES + +#ifndef RANDOM_WALKS_MULTITHREAD_WALKS_HPP +#define RANDOM_WALKS_MULTITHREAD_WALKS_HPP + +#include "sampling/sphere.hpp" +#include "generators/boost_random_number_generator.hpp" +#include "random_walks/gaussian_helpers.hpp" +#include "random_walks/gaussian_cdhr_walk.hpp" +#include "random_walks/gaussian_rdhr_walk.hpp" + + +// random directions hit-and-run walk with uniform target distribution +// from boundary + +struct BCDHRWalk_multithread +{ + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + p1 = Point(d); + p2 = Point(d); + p_prev = Point(d); + lambdas.setZero(m); + } + + Point p; + Point p1; + Point p2; + Point p_prev; + unsigned int rand_coord_prev; + unsigned int rand_coord; + typename Point::Coeff lambdas; + }; + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator& rng) + { + initialize(P, parameters, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + std::pair bpair; + for (auto j = 0u; j < walk_length; ++j) + { + params.rand_coord_prev = params.rand_coord; + params.rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + bpair = P.line_intersect_coord(params.p, + params.p_prev, + params.rand_coord, + params.rand_coord_prev, + params.lambdas); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, params.p[params.rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + params.p1 = params.p_prev; + params.p2 = params.p_prev; + params.p1.set_coord(params.rand_coord, params.p_prev[params.rand_coord] + bpair.first); + params.p2.set_coord(params.rand_coord, params.p_prev[params.rand_coord] + bpair.second); + } + + private : + + template + inline void initialize(GenericBody const& P, + thread_parameters_ ¶ms, // parameters + RandomNumberGenerator& rng) + { + params.lambdas.setZero(P.num_of_hyperplanes()); + params.rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + + std::pair bpair = P.line_intersect_coord(params.p, params.rand_coord, + params.lambdas); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, params.p[params.rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + + }; + +}; + + +// Random directions hit-and-run walk with uniform target distribution +// from the boundary, multithread version + +struct BRDHRWalk_multithread +{ + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + p1 = Point(d); + p2 = Point(d); + v = Point(d); + lambdas.setZero(m); + Av.setZero(m); + lambda_prev = NT(0); + } + + Point p; + Point p1; + Point p2; + Point v; + NT lambda_prev; + typename Point::Coeff lambdas; + typename Point::Coeff Av; + }; + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator& rng) + { + initialize(P, parameters, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + for (auto j=0u; j::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(params.p, params.v, params.lambdas, params.Av, + params.lambda_prev); + params.lambda_prev = rng.sample_urdist() * (bpair.first - bpair.second) + + bpair.second; + params.p1 = (bpair.first * params.v); + params.p1 += params.p; + params.p2 = (bpair.second * params.v); + params.p2 += params.p; + params.p += (params.lambda_prev * params.v); + } + } + + private : + + template + inline void initialize(GenericBody const& P, + thread_parameters_ ¶ms, // parameters + RandomNumberGenerator& rng) + { + params.lambdas.setZero(P.num_of_hyperplanes()); + params.Av.setZero(P.num_of_hyperplanes()); + + params.v = GetDirection::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(params.p, params.v, params.lambdas, params.Av); + params.lambda_prev = rng.sample_urdist() * (bpair.first - bpair.second) + bpair.second; + params.p = (params.lambda_prev * params.v) + params.p; + } + + }; + +}; + + +// Coordinate directions hit-and-run walk with spherical Gaussian target distribution +// Multithred version + +struct GaussianCDHRWalk_multithread +{ + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + p_prev = Point(d); + lambdas.setZero(m); + } + + Point p; + Point p_prev; + unsigned int rand_coord_prev; + unsigned int rand_coord; + typename Point::Coeff lambdas; + }; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + Walk(Polytope& P, + thread_parameters_ ¶ms, + NT const& a_i, + RandomNumberGenerator &rng) + { + initialize(P, params, a_i, rng); + } + + template + Walk(Polytope& P, + thread_parameters_ ¶ms, + NT const& a_i, + RandomNumberGenerator &rng, + parameters&) + { + initialize(P, params, a_i, rng); + } + + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + params.rand_coord_prev = params.rand_coord; + params.rand_coord = rng.sample_uidist(); + std::pair bpair = + P.line_intersect_coord(params.p, params.p_prev, params.rand_coord, + params.rand_coord_prev, params.lambdas); + NT dis = chord_random_point_generator_exp_coord + (params.p[params.rand_coord] + bpair.second, + params.p[params.rand_coord] + bpair.first, + a_i, + rng); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, dis); + } + } + +private : + + template + inline void initialize(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + NT const& a_i, + RandomNumberGenerator &rng) + { + params.lambdas.setZero(P.num_of_hyperplanes()); + params.rand_coord = rng.sample_uidist(); + + std::pair bpair = P.line_intersect_coord(params.p, params.rand_coord, params.lambdas); + NT dis = chord_random_point_generator_exp_coord + (params.p[params.rand_coord] + bpair.second, + params.p[params.rand_coord] + bpair.first, + a_i, rng); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, dis); + } + +}; + +}; + + +// Random directions hit-and-run walk with spherical Gaussian target distribution +// multithread version + +struct GaussianRDHRWalk_multithread +{ + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + v = Point(d); + } + + Point p; + Point v; + }; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + Walk(Polytope&, thread_parameters_ &, NT const&, RandomNumberGenerator&) + {} + + template + Walk(Polytope&, thread_parameters_ &, NT const&, RandomNumberGenerator&, + parameters&) + {} + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + NT const& a_i, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + params.v = GetDirection::apply(P.dimension(), rng); + std::pair dbpair = P.line_intersect(params.p, params.v); + + NT min_plus = dbpair.first; + NT max_minus = dbpair.second; + Point upper = (min_plus * params.v) + params.p; + Point lower = (max_minus * params.v) + params.p; + + chord_random_point_generator_exp(lower, upper, a_i, params.p, rng); + } + } +}; + +}; + + +// Billiard walk for uniform distribution + +struct BilliardWalk_multithread +{ + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + p0 = Point(d); + v = Point(d); + lambdas.setZero(m); + Av.setZero(m); + lambda_prev = NT(0); + } + + Point p; + Point p0; + Point v; + NT lambda_prev; + typename Point::Coeff lambdas; + typename Point::Coeff Av; + }; + + BilliardWalk_multithread(double L) + : param(L, true) + {} + + BilliardWalk_multithread() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + parameters param; + + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator &rng) + { + _Len = compute_diameter + ::template compute(P); + initialize(P, parameters, rng); + } + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator &rng, + parameters_ const& params) + { + _Len = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + initialize(P, parameters, rng); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope& P, + thread_parameters_ ¶meters, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T = rng.sample_urdist() * _Len; + const NT dl = 0.995; + + for (auto j=0u; j::apply(n, rng); + parameters.p0 = parameters.p; + int it = 0; + while (it < 50*n) + { + auto pbpair = P.line_positive_intersect(parameters.p, parameters.v, parameters.lambdas, + parameters.Av, parameters.lambda_prev); + if (T <= pbpair.first) { + parameters.p += (T * parameters.v); + parameters.lambda_prev = T; + break; + } + parameters.lambda_prev = dl * pbpair.first; + parameters.p += (parameters.lambda_prev * parameters.v); + T -= parameters.lambda_prev; + P.compute_reflection(parameters.v, parameters.p, pbpair.second); + it++; + } + if (it == 50*n) + { + parameters.p = parameters.p0; + } + } + } + + inline void update_delta(NT L) + { + _Len = L; + } + +private : + + template + < + typename GenericPolytope + > + inline void initialize(GenericPolytope& P, + thread_parameters_ ¶meters, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + const NT dl = 0.995; + parameters.lambdas.setZero(P.num_of_hyperplanes()); + parameters.Av.setZero(P.num_of_hyperplanes()); + + parameters.v = GetDirection::apply(n, rng); + NT T = rng.sample_urdist() * _Len; + int it = 0; + + std::pair pbpair + = P.line_positive_intersect(parameters.p, parameters.v, parameters.lambdas, parameters.Av); + if (T <= pbpair.first) + { + parameters.p += (T * parameters.v); + parameters.lambda_prev = T; + return; + } + parameters.lambda_prev = dl * pbpair.first; + parameters.p += (parameters.lambda_prev * parameters.v); + T -= parameters.lambda_prev; + P.compute_reflection(parameters.v, parameters.p, pbpair.second); + + while (it <= 50*n) + { + std::pair pbpair + = P.line_positive_intersect(parameters.p, parameters.v, parameters.lambdas, + parameters.Av, parameters.lambda_prev); + if (T <= pbpair.first) + { + parameters.p += (T * parameters.v); + parameters.lambda_prev = T; + break; + }else if (it == 50*n) { + parameters.lambda_prev = rng.sample_urdist() * pbpair.first; + parameters.p += (parameters.lambda_prev * parameters.v); + break; + } + parameters.lambda_prev = dl * pbpair.first; + parameters.p += (parameters.lambda_prev * parameters.v); + T -= parameters.lambda_prev; + P.compute_reflection(parameters.v, parameters.p, pbpair.second); + it++; + } + } + + NT _Len; +}; + +}; + + +// coordinate directions hit-and-run walk with uniform target distribution +// Parallel version + +struct CDHRWalk_multithread +{ + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + p_prev = Point(d); + lambdas.setZero(m); + } + + Point p; + Point p_prev; + unsigned int rand_coord_prev; + unsigned int rand_coord; + typename Point::Coeff lambdas; + }; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator& rng) + { + initialize(P, parameters, rng); + } + + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j=0u; j bpair = P.line_intersect_coord(params.p, + params.p_prev, + params.rand_coord, + params.rand_coord_prev, + params.lambdas); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, params.p[params.rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + } + +private : + + template + inline void initialize(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + RandomNumberGenerator &rng) + { + params.lambdas.setZero(P.num_of_hyperplanes()); + params.rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + + std::pair bpair = P.line_intersect_coord(params.p, params.rand_coord, params.lambdas); + params.p_prev = params.p; + params.p.set_coord(params.rand_coord, params.p[params.rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + +}; + +}; + + +// Random directions hit-and-run walk with uniform target distribution +// Parallel version + +struct RDHRWalk_multithread +{ + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + p = Point(d); + v = Point(d); + lambdas.setZero(m); + Av.setZero(m); + lambda_prev = NT(0); + } + + Point p; + Point v; + NT lambda_prev; + typename Point::Coeff lambdas; + typename Point::Coeff Av; + }; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef thread_parameters thread_parameters_; + //typedef thread_params thread_parameters_; + + template + Walk(GenericPolytope& P, thread_parameters_ ¶meters, RandomNumberGenerator& rng) + { + initialize(P, parameters, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + for (auto j=0u; j::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(params.p, params.v, params.lambdas, params.Av, + params.lambda_prev); + params.lambda_prev = rng.sample_urdist() * (bpair.first - bpair.second) + + bpair.second; + params.p += (params.lambda_prev * params.v); + } + } + +private : + + template + inline void initialize(BallPolytope const& P, + thread_parameters_ ¶ms, // parameters + RandomNumberGenerator &rng) + { + params.lambdas.setZero(P.num_of_hyperplanes()); + params.Av.setZero(P.num_of_hyperplanes()); + + params.v = GetDirection::apply(P.dimension(), rng); + std::pair bpair = P.line_intersect(params.p, params.v, params.lambdas, params.Av); + params.lambda_prev = rng.sample_urdist() * (bpair.first - bpair.second) + bpair.second; + params.p += (params.lambda_prev * params.v); + } + +}; + +}; + + +#endif // RANDOM_WALKS_MULTITHREAD_WALKS_HPP diff --git a/src/volesti/include/random_walks/nuts_hmc_walk.hpp b/src/volesti/include/random_walks/nuts_hmc_walk.hpp new file mode 100644 index 00000000..857064c0 --- /dev/null +++ b/src/volesti/include/random_walks/nuts_hmc_walk.hpp @@ -0,0 +1,380 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2022 Vissarion Fisikopoulos +// Copyright (c) 2018-2022 Apostolos Chalkis +// Copyright (c) 2020-2022 Elias Tsigaridas +// Copyright (c) 2020-2022 Marios Papachristou + +// Licensed under GNU LGPL.3, see LICENCE file + +// References +// Matthew D. Hoffman, Andrew Gelman. "The No-U-Turn Sampler: +// Adaptively Setting Path Lengths in Hamiltonian Monte Carlo", 2011. + +// Comment: Compared to [Matthew D. Hoffman, Andrew Gelman, 2011] +// we modify the step of Nesterov's algorithm in the burn in phase. + +#ifndef NUTS_HAMILTONIAN_MONTE_CARLO_WALK_HPP +#define NUTS_HAMILTONIAN_MONTE_CARLO_WALK_HPP + + +#include "generators/boost_random_number_generator.hpp" +#include "random_walks/gaussian_helpers.hpp" +#include "ode_solvers/ode_solvers.hpp" +#include "preprocess/estimate_L_smooth_parameter.hpp" + +struct NutsHamiltonianMonteCarloWalk { + + template + < + typename NT, + typename OracleFunctor + > + struct parameters { + NT epsilon; // tolerance in mixing + NT eta; // step size + + parameters( + OracleFunctor const& F, + unsigned int dim, + NT epsilon_=2) + { + epsilon = epsilon_; + eta = F.params.L > 0 ? 10.0 / (dim * sqrt(F.params.L)) : 0.005; + } + }; + + template + < + typename Point, + typename Polytope, + typename RandomNumberGenerator, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Solver + > + struct Walk { + + typedef std::vector pts; + typedef typename Point::FT NT; + typedef std::vector bounds; + + // Hyperparameters of the sampler + parameters ¶ms; + + // Numerical ODE solver + Solver *solver; + + // Dimension + unsigned int dim; + + // Discarded Samples + long num_runs = 0; + long total_acceptance = 0; + + // Average acceptance probability + NT average_acceptance = 0; + + // References to xs + Point x, v; + + // Helper points + Point v_pl, v_min, v_min_j, v_pl_j, X_pl, X_pl_j, X_min, X, X_rnd_j, X_min_j, x_pl_min; + + // Gradient function + NegativeGradientFunctor &F; + + bool accepted; + + // Burnin parameters + NT eps_step, mu, log_tilde_eps, H_tilde, alpha, na; + const NT delta = NT(0.65), Delta_max = NT(1000), gamma = NT(0.05), t0 = NT(10), kk = NT(0.85); + + // Density exponent + NegativeLogprobFunctor &f; + + Walk(Polytope *P, + Point &p, + NegativeGradientFunctor &neg_grad_f, + NegativeLogprobFunctor &neg_logprob_f, + parameters ¶m, + bool burn_in_phase = true) : + params(param), + F(neg_grad_f), + f(neg_logprob_f) + { + dim = p.dimension(); + + v_pl.set_dimension(dim); + v_min.set_dimension(dim); + v_min_j.set_dimension(dim); + v_pl_j.set_dimension(dim); + X_pl.set_dimension(dim); + X_pl_j.set_dimension(dim); + X_min.set_dimension(dim); + X.set_dimension(dim); + X_rnd_j.set_dimension(dim); + X_min_j.set_dimension(dim); + x_pl_min.set_dimension(dim); + + eps_step = params.eta; + mu = std::log(10*eps_step); + log_tilde_eps = NT(0); + H_tilde = NT(0); + alpha = NT(0); + na = NT(0); + + // Starting point is provided from outside + x = p; + + accepted = false; + + // Initialize solver + solver = new Solver(0, params.eta, pts{x, x}, F, bounds{P, NULL}); + disable_adaptive(); + + if (burn_in_phase) + { + RandomNumberGenerator rng(dim); + burnin(rng); + } + }; + + + inline void burnin(RandomNumberGenerator &rng, + unsigned int N = 1000, + unsigned int walk_length=1) + { + reset_num_runs(); + Point p = x; + NT L; + + if ((solver->get_bounds())[0] == NULL) + { + L = (NT(100) / NT(dim)) * (NT(100) / NT(dim)); + } + else + { + Polytope K = *(solver->get_bounds())[0]; + L = estimate_L_smooth(K, p, walk_length, F, rng); + } + + eps_step = NT(5) / (NT(dim) * std::sqrt(L)); + solver->set_eta(eps_step); + + for (int i = 0; i < N; i++) + { + apply(rng, walk_length, true); + solver->set_eta(eps_step); + } + reset_num_runs(); + } + + + inline void apply(RandomNumberGenerator &rng, + unsigned int walk_length=1, + bool burnin = false) + { + num_runs++; + + int x_counting_total = 0; + + // Pick a random velocity + v = GetDirection::apply(dim, rng, false); + + v_pl = v; + v_min = NT(-1) * v; + X_pl = x; + X_min = x; + + NT h1 = hamiltonian(x, v); + + NT uu = std::log(rng.sample_urdist()) - h1; + int j = -1; + bool s = true; + bool updated = false; + bool pos_state_single = false; + + if (burnin) + { + alpha = NT(0); + } + + while (s) + { + j++; + + if (burnin) + { + na = std::pow(NT(2), NT(j)); + } + + NT dir = rng.sample_urdist(); + + if (dir > 0.5) + { + v = v_pl; + X = X_pl; + } + else + { + v = v_min; + X = X_min; + } + X_rnd_j = X; + + int x_counting = 0; + int num_samples = int(std::pow(NT(2), NT(j))); + accepted = false; + + for (int k = 1; k <= num_samples; k++) + { + if (!accepted) + { + solver->set_state(0, X); + solver->set_state(1, v); + } + + // Get proposals + solver->steps(walk_length, accepted); + accepted = true; + + X = solver->get_state(0); + v = solver->get_state(1); + + NT hj = hamiltonian(X, v); + + if (burnin) + { + alpha += std::min(NT(1), std::exp(-hj + h1)); + } + + if (uu > Delta_max - hj) + { + s = false; + break; + } + + bool pos_state = false; + if (uu < -hj) + { + pos_state = true; + pos_state_single = true; + x_counting = x_counting + 1; + x_counting_total = x_counting_total + 1; + } + + if (k == 1) + { + if (dir > 0.5) + { + X_min_j = X; + v_min_j = v; + } + else + { + X_pl_j = X; + v_pl_j = v; + } + } + if (k == num_samples) + { + if (dir > 0.5) + { + x_pl_min = X - X_min_j; + if ((x_pl_min.dot(v) < 0) || (x_pl_min.dot(v_min_j) < 0)) + { + s = false; + } + } + else + { + x_pl_min = X_pl_j - X; + if ((x_pl_min.dot(v) < 0) || (x_pl_min.dot(v_pl_j) < 0)) + { + s = false; + } + } + } + if ((rng.sample_urdist() < (1/NT(x_counting))) && pos_state) + { + X_rnd_j = X; + } + } + + if (dir > 0.5) + { + X_pl = X; + v_pl = v; + } + else + { + X_min = X; + v_min = v; + } + + if (s && (rng.sample_urdist() < (NT(x_counting) / NT(x_counting_total)))) + { + x = X_rnd_j; + if (pos_state_single) + { + updated = true; + } + } + + if (s) + { + x_pl_min = X_pl - X_min; + if ((x_pl_min.dot(v_min) < 0) || (x_pl_min.dot(v_pl) < 0)) + { + s = false; + } + } + } + + if (updated) + { + total_acceptance++; + } + average_acceptance = NT(total_acceptance) / NT(num_runs); + + if (burnin) + { + H_tilde = (NT(1) - NT(1) / (NT(num_runs) + t0)) * H_tilde + (NT(1) / (NT(num_runs) + t0)) * (delta - alpha / na); + NT log_eps = mu - (std::sqrt(NT(num_runs)) / gamma) * H_tilde; + + // TODO: use the following to generalize Nesterov's algorithm + //log_tilde_eps = std::pow(mu,-kk) * log_eps + (NT(1) - std::pow(mu,-kk))*log_tilde_eps; + + eps_step = std::exp(log_eps); + } + } + + inline NT hamiltonian(Point &pos, Point &vel) const { + return f(pos) + 0.5 * vel.dot(vel); + } + + inline NT get_eta_solver() { + return solver->get_eta(); + } + + void disable_adaptive() { + solver->disable_adaptive(); + } + + void enable_adaptive() { + solver->enable_adaptive(); + } + + void reset_num_runs() { + num_runs = 0; + total_acceptance = 0; + } + + NT get_ratio_acceptance() { + return average_acceptance; + } + }; +}; + +#endif // HAMILTONIAN_MONTE_CARLO_WALK_HPP diff --git a/src/volesti/include/random_walks/random_walks.hpp b/src/volesti/include/random_walks/random_walks.hpp new file mode 100644 index 00000000..6c36457c --- /dev/null +++ b/src/volesti/include/random_walks/random_walks.hpp @@ -0,0 +1,34 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2020-2021 Vissarion Fisikopoulos +// Copyright (c) 2020-2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_RANDOM_WALKS_HPP +#define RANDOM_WALKS_RANDOM_WALKS_HPP + +#include + +#include "random_walks/boundary_cdhr_walk.hpp" +#include "random_walks/boundary_rdhr_walk.hpp" +#include "random_walks/gaussian_ball_walk.hpp" +#include "random_walks/gaussian_cdhr_walk.hpp" +#include "random_walks/gaussian_rdhr_walk.hpp" +#include "random_walks/uniform_ball_walk.hpp" +#include "random_walks/uniform_billiard_walk.hpp" +#include "random_walks/uniform_cdhr_walk.hpp" +#include "random_walks/uniform_rdhr_walk.hpp" +#include "random_walks/uniform_dikin_walk.hpp" +#include "random_walks/uniform_john_walk.hpp" +#include "random_walks/uniform_vaidya_walk.hpp" +#include "random_walks/uniform_accelerated_billiard_walk.hpp" +#include "random_walks/gaussian_accelerated_billiard_walk.hpp" +#include "random_walks/gaussian_hamiltonian_monte_carlo_exact_walk.hpp" +#include "random_walks/exponential_hamiltonian_monte_carlo_exact_walk.hpp" +#include "random_walks/uniform_accelerated_billiard_walk_parallel.hpp" +#include "random_walks/hamiltonian_monte_carlo_walk.hpp" +#include "random_walks/nuts_hmc_walk.hpp" +#include "random_walks/langevin_walk.hpp" +#include "random_walks/crhmc/crhmc_walk.hpp" +#endif // RANDOM_WALKS_RANDOM_WALKS_HPP diff --git a/src/volesti/include/random_walks/uniform_accelerated_billiard_walk.hpp b/src/volesti/include/random_walks/uniform_accelerated_billiard_walk.hpp new file mode 100644 index 00000000..980c8e5d --- /dev/null +++ b/src/volesti/include/random_walks/uniform_accelerated_billiard_walk.hpp @@ -0,0 +1,309 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_ACCELERATED_IMPROVED_BILLIARD_WALK_HPP +#define RANDOM_WALKS_ACCELERATED_IMPROVED_BILLIARD_WALK_HPP + +#include "sampling/sphere.hpp" + + +// Billiard walk which accelarates each step for uniform distribution + +struct AcceleratedBilliardWalk +{ + AcceleratedBilliardWalk(double L) + : param(L, true) + {} + + AcceleratedBilliardWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + struct update_parameters + { + update_parameters() + : facet_prev(0), hit_ball(false), inner_vi_ak(0.0), ball_inner_norm(0.0) + {} + int facet_prev; + bool hit_ball; + double inner_vi_ak; + double ball_inner_norm; + }; + + parameters param; + + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Polytope::MT MT; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope &P, Point const& p, RandomNumberGenerator &rng) + { + _update_parameters = update_parameters(); + _L = compute_diameter + ::template compute(P); + _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + _rho = 1000 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, rng); + } + + template + Walk(GenericPolytope &P, Point const& p, RandomNumberGenerator &rng, + parameters const& params) + { + _update_parameters = update_parameters(); + _L = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + _rho = 1000 * P.dimension(); // upper bound for the number of reflections (experimental) + initialize(P, p, rng); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope &P, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + const NT dl = 0.995; + int it; + + for (auto j=0u; j::apply(n, rng); + Point p0 = _p; + + it = 0; + std::pair pbpair = P.line_positive_intersect(_p, _v, _lambdas, _Av, + _lambda_prev, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + continue; + } + + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + it++; + + while (it < _rho) + { + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av, _lambda_prev, + _AA, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + it++; + } + if (it == _rho) _p = p0; + } + p = _p; + } + + + template + < + typename GenericPolytope + > + inline void get_starting_point(GenericPolytope &P, + Point const& center, + Point &q, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT radius = P.InnerBall().second; + + q = GetPointInDsphere::apply(n, radius, rng); + q += center; + initialize(P, q, rng); + + apply(P, q, walk_length, rng); + } + + + template + < + typename GenericPolytope + > + inline void parameters_burnin(GenericPolytope &P, + Point const& center, + unsigned int const& num_points, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + Point p(P.dimension()); + std::vector pointset; + pointset.push_back(center); + pointset.push_back(_p); + NT rad = NT(0), max_dist, Lmax = get_delta(), radius = P.InnerBall().second; + + for (int i = 0; i < num_points; i++) + { + Point p = GetPointInDsphere::apply(P.dimension(), radius, rng); + p += center; + initialize(P, p, rng); + + apply(P, p, walk_length, rng); + max_dist = get_max_distance(pointset, p, rad); + if (max_dist > Lmax) + { + Lmax = max_dist; + } + if (2.0*rad > Lmax) { + Lmax = 2.0 * rad; + } + pointset.push_back(p); + } + + if (Lmax > _L) { + if (P.dimension() <= 2500) + { + update_delta(Lmax); + } + else{ + update_delta(2.0 * get_delta()); + } + } + pointset.clear(); + } + + + + inline void update_delta(NT L) + { + _L = L; + } + + NT get_delta() + { + return _L; + } + + private : + + template + < + typename GenericPolytope + > + inline void initialize(GenericPolytope &P, + Point const& p, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + const NT dl = 0.995; + _lambdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + _p = p; + _v = GetDirection::apply(n, rng); + + NT T = -std::log(rng.sample_urdist()) * _L; + Point p0 = _p; + int it = 0; + + std::pair pbpair + = P.line_first_positive_intersect(_p, _v, _lambdas, _Av, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + return; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + + while (it <= _rho) + { + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av, _lambda_prev, _AA, _update_parameters); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + } else if (it == _rho) { + _lambda_prev = rng.sample_urdist() * pbpair.first; + _p += (_lambda_prev * _v); + break; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, _update_parameters); + it++; + } + } + + inline double get_max_distance(std::vector &pointset, Point const& q, double &rad) + { + double dis = -1.0, temp_dis; + int jj = 0; + for (auto vecit = pointset.begin(); vecit!=pointset.end(); vecit++, jj++) + { + temp_dis = (q.getCoefficients() - (*vecit).getCoefficients()).norm(); + if (temp_dis > dis) { + dis = temp_dis; + } + if (jj == 0) { + if (temp_dis > rad) { + rad = temp_dis; + } + } + } + return dis; + } + + double _L; + Point _p; + Point _v; + NT _lambda_prev; + MT _AA; + unsigned int _rho; + update_parameters _update_parameters; + typename Point::Coeff _lambdas; + typename Point::Coeff _Av; + }; + +}; + + +#endif diff --git a/src/volesti/include/random_walks/uniform_accelerated_billiard_walk_parallel.hpp b/src/volesti/include/random_walks/uniform_accelerated_billiard_walk_parallel.hpp new file mode 100644 index 00000000..1c2630f9 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_accelerated_billiard_walk_parallel.hpp @@ -0,0 +1,311 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_ACCELERATED_IMPROVED_BILLIARD_WALK_PARALLEL_HPP +#define RANDOM_WALKS_ACCELERATED_IMPROVED_BILLIARD_WALK_PARALLEL_HPP + +#include "sampling/sphere.hpp" + + +// Billiard walk which accelarates each step for uniform distribution and can be used for a parallel use by threads + +struct AcceleratedBilliardWalkParallel +{ + AcceleratedBilliardWalkParallel() + {} + + + struct update_parameters + { + update_parameters() + : facet_prev(0), hit_ball(false), inner_vi_ak(0.0), ball_inner_norm(0.0) + {} + int facet_prev; + bool hit_ball; + double inner_vi_ak; + double ball_inner_norm; + }; + + template + struct thread_parameters + { + thread_parameters(unsigned int d, unsigned int m) + { + update_step_parameters = update_parameters(); + p = Point(d); + v = Point(d); + lambdas.setZero(m); + Av.setZero(m); + lambda_prev = NT(0); + } + + update_parameters update_step_parameters; + Point p; + Point v; + NT lambda_prev; + typename Point::Coeff lambdas; + typename Point::Coeff Av; + }; + + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Polytope::MT MT; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope &P) + { + _L = compute_diameter + ::template compute(P); + _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + _p0 = Point(P.dimension()); + _rho = 1000 * P.dimension(); + } + + template + Walk(GenericPolytope &P, NT const& L) + { + _L = L > NT(0) ? L + : compute_diameter + ::template compute(P); + _AA.noalias() = P.get_mat() * P.get_mat().transpose(); + _p0 = Point(P.dimension()); + _rho = 1000 * P.dimension(); + } + + template + < + typename GenericPolytope, + typename thread_params + > + inline void apply(GenericPolytope &P, + thread_params ¶ms, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T; + const NT dl = 0.995; + int it; + + for (auto j=0u; j::apply(n, rng); + _p0 = params.p; + + it = 0; + std::pair pbpair = P.line_positive_intersect(params.p, params.v, params.lambdas, params.Av, + params.lambda_prev, params.update_step_parameters); + if (T <= pbpair.first) + { + params.p += (T * params.v); + params.lambda_prev = T; + continue; + } + + params.lambda_prev = dl * pbpair.first; + params.p += (params.lambda_prev * params.v); + T -= params.lambda_prev; + P.compute_reflection(params.v, params.p, params.update_step_parameters); + it++; + + while (it < _rho) + { + std::pair pbpair + = P.line_positive_intersect(params.p, params.v, params.lambdas, params.Av, + params.lambda_prev, _AA, params.update_step_parameters); + if (T <= pbpair.first) { + params.p += (T * params.v); + params.lambda_prev = T; + break; + } + params.lambda_prev = dl * pbpair.first; + params.p += (params.lambda_prev * params.v); + T -= params.lambda_prev; + P.compute_reflection(params.v, params.p, params.update_step_parameters); + it++; + } + if (it == _rho) params.p = _p0; + } + } + + + template + < + typename GenericPolytope, + typename thread_params + > + inline void get_starting_point(GenericPolytope &P, + Point const& center, + thread_params ¶ms, + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT radius = P.InnerBall().second; + + params.p = GetPointInDsphere::apply(n, radius, rng); + params.p += center; + initialize(P, params, rng); + + apply(P, params, walk_length, rng); + } + + + template + < + typename GenericPolytope, + typename thread_params + > + inline void parameters_burnin(GenericPolytope &P, + Point const& center, + unsigned int const& num_points, + unsigned int const& walk_length, + RandomNumberGenerator &rng, + thread_params ¶ms) + { + std::vector pointset; + pointset.push_back(center); + + params.p = Point(P.dimension()); + NT rad = NT(0), max_dist, Lmax = get_delta(), radius = P.InnerBall().second; + + for (int i = 0; i < num_points; i++) + { + params.p = GetPointInDsphere::apply(P.dimension(), radius, rng); + params.p += center; + initialize(P, params, rng); + + apply(P, params, walk_length, rng); + max_dist = get_max_distance(pointset, params.p, rad); + if (max_dist > Lmax) + { + Lmax = max_dist; + } + if (2.0*rad > Lmax) { + Lmax = 2.0 * rad; + } + pointset.push_back(params.p); + } + + if (Lmax > _L) + { + if (P.dimension() <= 2500) + { + update_delta(Lmax); + } + else{ + update_delta(2.0 * get_delta()); + } + } + pointset.clear(); + } + + + + inline void update_delta(NT L) + { + _L = L; + } + + NT get_delta() + { + return _L; + } + + private : + + template + < + typename GenericPolytope, + typename thread_params + > + inline void initialize(GenericPolytope &P, + thread_params ¶ms, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + const NT dl = 0.995; + params.v = GetDirection::apply(n, rng); + + NT T = -std::log(rng.sample_urdist()) * _L; + int it = 0; + + std::pair pbpair + = P.line_first_positive_intersect(params.p, params.v, params.lambdas, + params.Av, params.update_step_parameters); + if (T <= pbpair.first) { + params.p += (T * params.v); + params.lambda_prev = T; + return; + } + params.lambda_prev = dl * pbpair.first; + params.p += (params.lambda_prev * params.v); + T -= params.lambda_prev; + P.compute_reflection(params.v, params.p, params.update_step_parameters); + + while (it <= _rho) + { + std::pair pbpair + = P.line_positive_intersect(params.p, params.v, params.lambdas, params.Av, + params.lambda_prev, _AA, params.update_step_parameters); + if (T <= pbpair.first) { + params.p += (T * params.v); + params.lambda_prev = T; + break; + } else if (it == _rho) { + params.lambda_prev = rng.sample_urdist() * pbpair.first; + params.p += (params.lambda_prev * params.v); + break; + } + params.lambda_prev = dl * pbpair.first; + params.p += (params.lambda_prev * params.v); + T -= params.lambda_prev; + P.compute_reflection(params.v, params.p, params.update_step_parameters); + it++; + } + } + + inline double get_max_distance(std::vector &pointset, Point const& q, double &rad) + { + double dis = -1.0, temp_dis; + int jj = 0; + for (auto vecit = pointset.begin(); vecit!=pointset.end(); vecit++, jj++) + { + temp_dis = (q.getCoefficients() - (*vecit).getCoefficients()).norm(); + if (temp_dis > dis) { + dis = temp_dis; + } + if (jj == 0) { + if (temp_dis > rad) { + rad = temp_dis; + } + } + } + return dis; + } + + NT _L; + MT _AA; + Point _p0; + unsigned int _rho; + }; + +}; + + +#endif + + diff --git a/src/volesti/include/random_walks/uniform_ball_walk.hpp b/src/volesti/include/random_walks/uniform_ball_walk.hpp new file mode 100644 index 00000000..77c608b7 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_ball_walk.hpp @@ -0,0 +1,96 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_UNIFORM_BALL_WALK_HPP +#define RANDOM_WALKS_UNIFORM_BALL_WALK_HPP + +#include "generators/boost_random_number_generator.hpp" + +// Ball walk with uniform target distribution + +struct BallWalk +{ + BallWalk(double L) + : param(L, true) + {} + + BallWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_delta(set) + {} + double m_L; + bool set_delta; + }; + + parameters param; + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, Point const& /*p*/, + RandomNumberGenerator& /*rng*/) + { + _delta = compute_delta(P); + } + + template + Walk(GenericPolytope& P, Point const& /*p*/, + RandomNumberGenerator& /*rng*/, parameters const& params) + { + _delta = params.set_delta ? params.m_L + : compute_delta(P); + } + + template + static inline NT compute_delta(GenericPolytope& P) + { + //return ((P.InnerBall()).second * NT(4)) / NT(P.dimension()); + return (NT(4) * (P.InnerBall()).second) / std::sqrt(NT(P.dimension())); + } + + template + inline void apply(BallPolytope const& P, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j = 0u; j < walk_length; ++j) + { + Point y = GetPointInDsphere::apply(P.dimension(), + _delta, + rng); + y += p; + if (P.is_in(y) == -1) p = y; + } + } + + inline void update_delta(NT delta) + { + _delta = delta; + } + + private: + double _delta; + }; +}; + +#endif // RANDOM_WALKS_UNIFORM_BALL_WALK_HPP diff --git a/src/volesti/include/random_walks/uniform_billiard_walk.hpp b/src/volesti/include/random_walks/uniform_billiard_walk.hpp new file mode 100644 index 00000000..dcbc96b1 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_billiard_walk.hpp @@ -0,0 +1,200 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2019 program. +// Contributed and modified by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_UNIFORM_BILLIARD_WALK_HPP +#define RANDOM_WALKS_UNIFORM_BILLIARD_WALK_HPP + +#include + +#include "convex_bodies/ball.h" +#include "convex_bodies/ballintersectconvex.h" +#include "convex_bodies/hpolytope.h" +#include "convex_bodies/spectrahedra/spectrahedron.h" +#include "convex_bodies/correlation_matrices/correlation_spectrahedron.hpp" +#include "convex_bodies/correlation_matrices/correlation_spectrahedron_MT.hpp" +#ifndef DISABLE_LPSOLVE + #include "convex_bodies/vpolytope.h" + #include "convex_bodies/vpolyintersectvpoly.h" + #include "convex_bodies/zpolytope.h" + #include "convex_bodies/zonoIntersecthpoly.h" +#endif +#include "sampling/sphere.hpp" +#include "random_walks/boundary_cdhr_walk.hpp" +#include "generators/boost_random_number_generator.hpp" +#include "sampling/random_point_generators.hpp" +#include "volume/sampling_policies.hpp" +#include "random_walks/compute_diameter.hpp" + +// Billiard walk for uniform distribution + +struct BilliardWalk +{ + BilliardWalk(double L) + : param(L, true) + {} + + BilliardWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + parameters param; + + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope &P, Point const& p, RandomNumberGenerator &rng) + { + _Len = compute_diameter + ::template compute(P); + initialize(P, p, rng); + } + + template + Walk(GenericPolytope &P, Point const& p, RandomNumberGenerator &rng, + parameters const& params) + { + _Len = params.set_L ? params.m_L + : compute_diameter + ::template compute(P); + initialize(P, p, rng); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope &P, + Point& p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + NT T = rng.sample_urdist() * _Len; + const NT dl = 0.995; + + for (auto j=0u; j::apply(n, rng); + + Point p0 = _p; + int it = 0; + while (it < 50*n) + { + auto pbpair = P.line_positive_intersect(_p, _v, _lambdas, + _Av, _lambda_prev); + + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + } + + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + + P.compute_reflection(_v, _p, pbpair.second); + + it++; + } + if (it == 50*n){ + _p = p0; + } + } + p = _p; + } + + inline void update_delta(NT L) + { + _Len = L; + } + +private : + + template + < + typename GenericPolytope + > + inline void initialize(GenericPolytope &P, + Point const& p, + RandomNumberGenerator &rng) + { + unsigned int n = P.dimension(); + const NT dl = 0.995; + _lambdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + _p = p; + _v = GetDirection::apply(n, rng); + + NT T = rng.sample_urdist() * _Len; + Point p0 = _p; + int it = 0; + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + return; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, pbpair.second); + while (it <= 50*n) + { + std::pair pbpair + = P.line_positive_intersect(_p, _v, _lambdas, _Av, _lambda_prev); + if (T <= pbpair.first) { + _p += (T * _v); + _lambda_prev = T; + break; + }else if (it == 50*n) { + _lambda_prev = rng.sample_urdist() * pbpair.first; + _p += (_lambda_prev * _v); + break; + } + _lambda_prev = dl * pbpair.first; + _p += (_lambda_prev * _v); + T -= _lambda_prev; + P.compute_reflection(_v, _p, pbpair.second); + it++; + } + } + + NT _Len; + Point _p; + Point _v; + NT _lambda_prev; + typename Point::Coeff _lambdas; + typename Point::Coeff _Av; +}; + +}; + +#endif // RANDOM_WALKS_UNIFORM_BILLIARD_WALK_HPP diff --git a/src/volesti/include/random_walks/uniform_cdhr_walk.hpp b/src/volesti/include/random_walks/uniform_cdhr_walk.hpp new file mode 100644 index 00000000..12d2ff78 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_cdhr_walk.hpp @@ -0,0 +1,97 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_UNIFORM_CDHR_WALK_HPP +#define RANDOM_WALKS_UNIFORM_CDHR_WALK_HPP + +#include "sampling/sphere.hpp" + +// coordinate directions hit-and-run walk with uniform target distribution + +struct CDHRWalk +{ + struct parameters {}; + parameters param; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, Point const& p, RandomNumberGenerator& rng) + { + initialize(P, p, rng); + } + + template + Walk(GenericPolytope& P, Point const& p, + RandomNumberGenerator& rng, parameters const& params) + { + initialize(P, p, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope const& P, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &rng) + { + for (auto j=0u; j bpair = P.line_intersect_coord(_p, + _p_prev, + _rand_coord, + rand_coord_prev, + _lamdas); + _p_prev = _p; + _p.set_coord(_rand_coord, _p[_rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + p = _p; + } + +private : + + template + inline void initialize(BallPolytope const& P, + Point const& p, + RandomNumberGenerator &rng) + { + _lamdas.setZero(P.num_of_hyperplanes()); + _rand_coord = rng.sample_uidist(); + NT kapa = rng.sample_urdist(); + _p = p; + std::pair bpair = P.line_intersect_coord(_p, _rand_coord, _lamdas); + _p_prev = _p; + _p.set_coord(_rand_coord, _p[_rand_coord] + bpair.first + kapa + * (bpair.second - bpair.first)); + } + + unsigned int _rand_coord; + Point _p; + Point _p_prev; + typename Point::Coeff _lamdas; +}; + +}; + + +#endif // RANDOM_WALKS_UNIFORM_CDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/uniform_dikin_walk.hpp b/src/volesti/include/random_walks/uniform_dikin_walk.hpp new file mode 100644 index 00000000..e791b254 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_dikin_walk.hpp @@ -0,0 +1,102 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_DIKIN_WALK_HPP +#define RANDOM_WALKS_DIKIN_WALK_HPP + +#include "convex_bodies/ball.h" +#include "convex_bodies/ballintersectconvex.h" +#include "convex_bodies/hpolytope.h" +#ifndef DISABLE_LPSOLVE + #include "convex_bodies/vpolytope.h" + #include "convex_bodies/vpolyintersectvpoly.h" + #include "convex_bodies/zpolytope.h" +#endif +#include "convex_bodies/zonoIntersecthpoly.h" +#include "ellipsoid_walks/dikin_walker.h" + + +// Dikin walk for uniform distribution + +struct DikinWalk +{ + DikinWalk(double L) + : param(L, true) + {} + + DikinWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + parameters param; + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Point::FT NT; + + Walk(Polytope &P, Point &p, RandomNumberGenerator &) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = P.ComputeInnerBall().second; + dikinw = DikinWalker(p0, A, b, r); + } + + Walk(Polytope &P, Point & p, RandomNumberGenerator &, parameters const& params) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = params.set_L ? params.m_L + : P.ComputeInnerBall().second; + dikinw = DikinWalker(p0, A, b, r); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope &, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &) + { + for (auto j=0u; j dikinw; + VT _vec_point; + }; + +}; + + +#endif diff --git a/src/volesti/include/random_walks/uniform_john_walk.hpp b/src/volesti/include/random_walks/uniform_john_walk.hpp new file mode 100644 index 00000000..2c6bc42e --- /dev/null +++ b/src/volesti/include/random_walks/uniform_john_walk.hpp @@ -0,0 +1,94 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_JOHN_WALK_HPP +#define RANDOM_WALKS_JOHN_WALK_HPP + + +#include "ellipsoid_walks/john_walker.h" + + +// John walk for uniform distribution + +struct JohnWalk +{ + JohnWalk(double L) + : param(L, true) + {} + + JohnWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + parameters param; + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Point::FT NT; + + Walk(Polytope &P, Point &p, RandomNumberGenerator &) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = P.ComputeInnerBall().second; + johnw = JohnWalker(p0, A, b, r); + } + + Walk(Polytope &P, Point & p, RandomNumberGenerator &, parameters const& params) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = params.set_L ? params.m_L + : P.ComputeInnerBall().second; + johnw = JohnWalker(p0, A, b, r); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope &, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &) + { + for (auto j=0u; j johnw; + VT _vec_point; + }; + +}; + + +#endif diff --git a/src/volesti/include/random_walks/uniform_rdhr_walk.hpp b/src/volesti/include/random_walks/uniform_rdhr_walk.hpp new file mode 100644 index 00000000..5e0eb475 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_rdhr_walk.hpp @@ -0,0 +1,92 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_UNIFORM_RDHR_WALK_HPP +#define RANDOM_WALKS_UNIFORM_RDHR_WALK_HPP + + +#include "sampling/sphere.hpp" + +// Random directions hit-and-run walk with uniform target distribution + +struct RDHRWalk +{ + struct parameters {}; + parameters param; + +template +< + typename Polytope, + typename RandomNumberGenerator +> +struct Walk +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + + template + Walk(GenericPolytope& P, Point const& p, RandomNumberGenerator& rng) + { + initialize(P, p, rng); + } + + template + Walk(GenericPolytope& P, Point const& p, + RandomNumberGenerator& rng, parameters const& params) + { + initialize(P, p, rng); + } + + template + < + typename BallPolytope + > + inline void apply(BallPolytope& P, + Point& p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator& rng) + { + for (auto j=0u; j::apply(p.dimension(), rng); + std::pair bpair = P.line_intersect(_p, v, _lamdas, _Av, + _lambda); + _lambda = rng.sample_urdist() * (bpair.first - bpair.second) + + bpair.second; + _p += (_lambda * v); + } + p = _p; + } + +private : + + template + inline void initialize(BallPolytope& P, + Point const& p, + RandomNumberGenerator &rng) + { + _lamdas.setZero(P.num_of_hyperplanes()); + _Av.setZero(P.num_of_hyperplanes()); + + Point v = GetDirection::apply(p.dimension(), rng); + std::pair bpair = P.line_intersect(p, v, _lamdas, _Av); + _lambda = rng.sample_urdist() * (bpair.first - bpair.second) + bpair.second; + _p = (_lambda * v) + p; + } + + Point _p; + NT _lambda; + typename Point::Coeff _lamdas; + typename Point::Coeff _Av; +}; + +}; + + +#endif // RANDOM_WALKS_UNIFORM_RDHR_WALK_HPP diff --git a/src/volesti/include/random_walks/uniform_vaidya_walk.hpp b/src/volesti/include/random_walks/uniform_vaidya_walk.hpp new file mode 100644 index 00000000..5bbbe991 --- /dev/null +++ b/src/volesti/include/random_walks/uniform_vaidya_walk.hpp @@ -0,0 +1,95 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Contributed and/or modified by Alexandros Manochis, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef RANDOM_WALKS_VAIDYA_WALK_HPP +#define RANDOM_WALKS_VAIDYA_WALK_HPP + + +#include "ellipsoid_walks/vaidya_walker.h" + + +// Vaidya walk for uniform distribution + +struct VaidyaWalk +{ + VaidyaWalk(double L) + : param(L, true) + {} + + VaidyaWalk() + : param(0, false) + {} + + struct parameters + { + parameters(double L, bool set) + : m_L(L), set_L(set) + {} + double m_L; + bool set_L; + }; + + parameters param; + + + template + < + typename Polytope, + typename RandomNumberGenerator + > + struct Walk + { + typedef typename Polytope::PointType Point; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + typedef typename Point::FT NT; + + Walk(Polytope &P, Point &p, RandomNumberGenerator &) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = P.ComputeInnerBall().second; + vaidyaw = VaidyaWalker(p0, A, b, r); + } + + Walk(Polytope &P, Point & p, RandomNumberGenerator &, parameters const& params) + { + MT A = P.get_mat(); + VT b = P.get_vec(), _vec_point = VT::Zero(P.dimension()), p0 = p.getCoefficients(); + NT r = params.set_L ? params.m_L + : P.ComputeInnerBall().second; + vaidyaw = VaidyaWalker(p0, A, b, r); + } + + template + < + typename GenericPolytope + > + inline void apply(GenericPolytope &, + Point &p, // a point to start + unsigned int const& walk_length, + RandomNumberGenerator &) + { + for (auto j=0u; j vaidyaw; + VT _vec_point; + }; + +}; + + +#endif diff --git a/src/volesti/include/root_finders/mp_solve_wrapper.hpp b/src/volesti/include/root_finders/mp_solve_wrapper.hpp new file mode 100644 index 00000000..1193f07c --- /dev/null +++ b/src/volesti/include/root_finders/mp_solve_wrapper.hpp @@ -0,0 +1,75 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef MP_SOLVE_WRAPPER_HPP +#define MP_SOLVE_WRAPPER_HPP + +template +std::vector> mpsolve(std::vector &coeffs, bool positive_real=false) { + + long n = (long) coeffs.size(); + + while (std::abs(coeffs[n-1]) < NT(1e-9)) { + n--; + + } + + + mps_monomial_poly *p; + mps_context *s; + + s = mps_context_new (); + p = mps_monomial_poly_new (s, n-1); + + mps_context_select_algorithm(s, MPS_ALGORITHM_SECULAR_GA); + + for (long i = 0; i < n; i++) { + mps_monomial_poly_set_coefficient_d (s, p, i, coeffs[i], 0); + } + + + /* Set the input polynomial */ + mps_context_set_input_poly (s, MPS_POLYNOMIAL (p)); + + /* Allocate space to hold the results. We check only floating point results + * in here */ + cplx_t *results = cplx_valloc (n-1); + + /* Actually solve the polynomial */ + mps_mpsolve (s); + + /* Save roots computed in the vector results */ + mps_context_get_roots_d (s, &results, NULL); + + std::vector> results_vector; + + NT real, im; + + for (long i = 0; i < n - 1; i++) { + real = (NT) cplx_Re(*results); + im = (NT) cplx_Im(*results); + + #ifdef VOLESTI_DEBUG + std::cout << real << " + " << im << "i" << std::endl; + #endif + results++; + if (positive_real) { + if (real > 0 && std::abs(im) < 1e-8) { + results_vector.push_back(std::make_pair(real, im)); + } + } else { + results_vector.push_back(std::make_pair(real, im)); + } + } + + return results_vector; +} + +#endif diff --git a/src/volesti/include/root_finders/newton_raphson.hpp b/src/volesti/include/root_finders/newton_raphson.hpp new file mode 100644 index 00000000..27411d76 --- /dev/null +++ b/src/volesti/include/root_finders/newton_raphson.hpp @@ -0,0 +1,49 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef NEWTON_RAPHSON_HPP +#define NEWTON_RAPHSON_HPP + +/// @brief Function implementing the Newton-Raphson numerical method +/// @tparam NT Number type +/// @tparam func Function type +template +std::pair newton_raphson(NT t0, func f, func grad_f, const NT rtol, + const NT reg=0, const unsigned int max_tries=1000000) { + NT t, t_prev, err; + NT y, y_prime; + t = t0; + unsigned int tries = 0; + + do { + tries++; + y = f(t_prev); + y_prime = grad_f(t_prev); + + if (std::abs(y_prime) < rtol) y_prime += reg; + + t = t_prev - y / y_prime; + if (t_prev != 0) { + err = std::abs(t - t_prev) / t_prev; + } else { + err = std::abs(t - t_prev); + } + + t_prev = t; + + if (tries > max_tries) break; + + } while (err > rtol); + + return std::make_pair(t, tries > max_tries); + +} + +#endif diff --git a/src/volesti/include/root_finders/quadratic_polynomial_solvers.hpp b/src/volesti/include/root_finders/quadratic_polynomial_solvers.hpp new file mode 100644 index 00000000..ad39c2ef --- /dev/null +++ b/src/volesti/include/root_finders/quadratic_polynomial_solvers.hpp @@ -0,0 +1,42 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + + +#ifndef QUADRATIC_POLYNOMIAL_SOLVERS_H +#define QUADRATIC_POLYNOMIAL_SOLVERS_H + +// The function compute the roots of a quadratic polynomial equation +template +bool solve_quadratic_polynomial(NT const& a, NT const& b, NT const& c, NT &x1, NT &x2) +{ + if (a == NT(0)) { + x1 = -c / b; + x2 = x1; + return true; + } + + NT Delta = b * b - 4.0 * a * c; + if (Delta < NT(0)) + { + return false; + } + + if (b >= NT(0)) + { + x1 = (- b - std::sqrt(Delta)) / (2.0 * a); + x2 = (2.0 * c) / (- b - std::sqrt(Delta)); + } + else + { + x1 = (2.0 * c) / (- b + std::sqrt(Delta)); + x2 = (- b + std::sqrt(Delta)) / (2.0 * a); + } + return true; +} + + +#endif + diff --git a/src/volesti/include/root_finders/root_finders.hpp b/src/volesti/include/root_finders/root_finders.hpp new file mode 100644 index 00000000..d9b36f17 --- /dev/null +++ b/src/volesti/include/root_finders/root_finders.hpp @@ -0,0 +1,28 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2020-2020 Marios Papachristou + +// Contributed and/or modified by Marios Papachristou, as part of Google Summer of Code 2020 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "newton_raphson.hpp" +#include "mp_solve_wrapper.hpp" + +#ifndef ROOT_FINDERS_HPP +#define ROOT_FINDERS_HPP + +#endif diff --git a/src/volesti/include/sampling/ellipsoid.hpp b/src/volesti/include/sampling/ellipsoid.hpp new file mode 100644 index 00000000..7507b989 --- /dev/null +++ b/src/volesti/include/sampling/ellipsoid.hpp @@ -0,0 +1,82 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2021 Vissarion Fisikopoulos +// Copyright (c) 2018-2021 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +//Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + + +#ifndef SAMPLERS_ELLIPSOID_HPP +#define SAMPLERS_ELLIPSOID_HPP + +#include "sphere.hpp" +#include "Eigen/Eigen" + + +template +struct GetPointInDellipsoid +{ + typedef typename Point::FT NT; + + template + inline static Point apply(unsigned int const& dim, + Ellipsoid const& E, + RandomNumberGenerator& rng) + { + // Generate a point inside a sphere of radius 1.0 + Point p = GetPointInDsphere::apply(dim, NT(1.0), rng); + + // transform it to a point inside an ellipsoid + return Point(E.mult_Lcov(p.getCoefficients())); + } +}; + + +template +struct GetGaussianDirection +{ + typedef typename Point::FT NT; + typedef typename Eigen::Matrix VT; + + template + inline static Point apply(unsigned int const& dim, + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + RandomNumberGenerator &rng) + { + // Generate a point inside a sphere of radius 1.0 + Point p = GetDirection::apply(dim, rng, false); + + // Multiply with cholesky matrix + VT gaussian_vec = E.mult_Lcov(p.getCoefficients()); + + // convert to point + return Point(gaussian_vec); + } +}; + + +template +struct GetPointOnDellipsoid +{ + typedef typename Point::FT NT; + typedef typename Eigen::Matrix VT; + + template + inline static Point apply(unsigned int const& dim, + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + RandomNumberGenerator &rng) + { + // Generate a point inside a sphere of radius 1.0 + Point p = GetDirection::apply(dim, rng, true); + + // Multiply with cholesky matrix + VT gaussian_vec = E.mult_Lcov(p.getCoefficients()); + + // convert to point + return Point(gaussian_vec); + } +}; + + +#endif // ELLIPSOID_HPP \ No newline at end of file diff --git a/src/volesti/include/sampling/mmcs.hpp b/src/volesti/include/sampling/mmcs.hpp new file mode 100644 index 00000000..07529ad6 --- /dev/null +++ b/src/volesti/include/sampling/mmcs.hpp @@ -0,0 +1,128 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef MMCS_HPP +#define MMCS_HPP + +#include "diagnostics/ess_window_updater.hpp" + +/** + * The class implements a single step of the Multiphase Monte Carlo Sampling algorithm + * given in, + * + * A. Chalkis, V. Fisikopoulos, E. Tsigaridas, H. Zafeiropoulos, Geometric algorithms for sampling the flux space of metabolic networks, SoCG 21. + * + * @tparam Polytope convex polytope type + * @tparam RandomNumberGenerator random number generator type + * @tparam MT matrix type + * @tparam Point cartensian point type + * @tparam WalkTypePolicy random walk type +*/ +template +< + typename Polytope, + typename RandomNumberGenerator, + typename MT, + typename Point, + typename WalkTypePolicy +> +bool perform_mmcs_step(Polytope &P, + RandomNumberGenerator &rng, + unsigned int const& walk_length, + unsigned int const& target_ess, + unsigned int const& max_num_samples, + unsigned int const& window, + unsigned int &Neff_sampled, + unsigned int &total_samples, + unsigned int const& num_rounding_steps, + MT &TotalRandPoints, + const Point &starting_point, + unsigned int const& nburns, + bool request_rounding, + WalkTypePolicy &WalkType) +{ + typedef typename Polytope::NT NT; + typedef typename Polytope::VT VT; + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > Walk; + + bool done = false; + unsigned int points_to_sample = target_ess; + int min_eff_samples; + total_samples = 0; + MT winPoints(P.dimension(), window); + Point q(P.dimension()); + + Point p = starting_point; + + if (request_rounding) + { + TotalRandPoints.setZero(num_rounding_steps, P.dimension()); + } + else + { + TotalRandPoints.setZero(max_num_samples, P.dimension()); + } + + Walk walk(P, p, rng, WalkType.param); + ESSestimator estimator(window, P.dimension()); + + walk.template parameters_burnin(P, p, 10 + int(std::log(NT(P.dimension()))), 10, rng); + + while (!done) + { + walk.template get_starting_point(P, p, q, 10, rng); + for (int i = 0; i < window; i++) + { + walk.template apply(P, q, walk_length, rng); + winPoints.col(i) = q.getCoefficients(); + } + estimator.update_estimator(winPoints); + total_samples += window; + if (total_samples >= TotalRandPoints.rows()) + { + if (total_samples > TotalRandPoints.rows()) + { + TotalRandPoints.conservativeResize(total_samples, P.dimension()); + } + if (request_rounding || total_samples >= max_num_samples) + { + done = true; + } + } + TotalRandPoints.block(total_samples - window, 0, window, P.dimension()) = winPoints.transpose(); + if (done || total_samples >= points_to_sample) + { + estimator.estimate_effective_sample_size(); + min_eff_samples = int(estimator.get_effective_sample_size().minCoeff()); + if (done && min_eff_samples < target_ess) + { + Neff_sampled = min_eff_samples; + return false; + } + if (min_eff_samples >= target_ess) + { + Neff_sampled = min_eff_samples; + return true; + } + if (min_eff_samples > 0) + { + points_to_sample += (total_samples / min_eff_samples) * (target_ess - min_eff_samples) + 100; + } + else + { + points_to_sample = total_samples + 100; + } + } + } + return false; +} + +#endif diff --git a/src/volesti/include/sampling/parallel_mmcs.hpp b/src/volesti/include/sampling/parallel_mmcs.hpp new file mode 100644 index 00000000..a8308f4c --- /dev/null +++ b/src/volesti/include/sampling/parallel_mmcs.hpp @@ -0,0 +1,203 @@ +// VolEsti (volume computation and sampling library) +// Copyright (c) 2021 Vissarion Fisikopoulos +// Copyright (c) 2021 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef PARALLEL_MMCS_HPP +#define PARALLEL_MMCS_HPP + + +#include +#include +#include +#include "diagnostics/ess_window_updater.hpp" + +/** + * The class implements a single step of the Parallel Multiphase Monte Carlo Sampling algorithm + * given in, + * + * A. Chalkis, V. Fisikopoulos, E. Tsigaridas, H. Zafeiropoulos, Geometric algorithms for sampling the flux space of metabolic networks, SoCG 21. + * + * @tparam WalkTypePolicy random walk type + * @tparam Polytope convex polytope type + * @tparam RandomNumberGenerator random number generator type + * @tparam MT matrix type + * @tparam Point cartensian point type + * @tparam NT number type +*/ +template +< + typename WalkTypePolicy, + typename Polytope, + typename RandomNumberGenerator, + typename MT, + typename Point, + typename NT +> +bool perform_parallel_mmcs_step(Polytope &P, + RandomNumberGenerator &rng, + unsigned int const& walk_length, + unsigned int const& target_ess, + unsigned int const& max_num_samples, + unsigned int const& window, + unsigned int &Neff_sampled, + unsigned int &total_samples, + unsigned int const& num_rounding_steps, + MT &TotalRandPoints, + const Point &starting_point, + unsigned int const& nburns, + unsigned int const& num_threads, + bool request_rounding, + NT L) +{ + typedef typename Polytope::VT VT; // vector type + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > Walk; + + typedef typename WalkTypePolicy::template thread_parameters + < + NT, + Point + > _thread_parameters; + + omp_set_num_threads(num_threads); + std::vector num_starting_points_per_thread(num_threads, 0); + std::vector bound_on_num_points_per_thread(num_threads, 0); + std::vector num_generated_points_per_thread(num_threads, 0); + unsigned int jj = 0, d = P.dimension(), m = P.num_of_hyperplanes(); + bool complete = false; + + while (jj < nburns) + { + for (unsigned int i = 0; i < num_threads; i++) + { + num_starting_points_per_thread[i]++; + bound_on_num_points_per_thread[i] += window; + jj++; + } + } + + std::vector winPoints_per_thread(num_threads, MT::Zero(d, window)); + std::vector TotalRandPoints_per_thread(num_threads); + + ESSestimator estimator(window, P.dimension()); + + bool done = false, done_all = false; + unsigned int points_to_sample = target_ess; + int min_eff_samples; + total_samples = 0; + + Point pp = starting_point; + for (unsigned int i = 0; i < num_threads; i++) + { + TotalRandPoints_per_thread[i].setZero(bound_on_num_points_per_thread[i], d); + } + unsigned int upper_bound_on_total_num_of_samples; + if (request_rounding) + { + upper_bound_on_total_num_of_samples = num_rounding_steps; + } + else + { + upper_bound_on_total_num_of_samples = max_num_samples; + } + TotalRandPoints.resize(0, 0); + Walk walk(P, L); + + _thread_parameters random_walk_parameters(d, m); + walk.template parameters_burnin(P, pp, 10 + int(std::log(NT(d))), 10, rng, random_walk_parameters); + Point const p = pp; + + #pragma omp parallel + { + int thread_index = omp_get_thread_num(); + _thread_parameters thread_random_walk_parameters(d, m); + + for (unsigned int it = 0; it < num_starting_points_per_thread[thread_index]; it++) + { + if (done_all) + { + break; + } + walk.template get_starting_point(P, p, thread_random_walk_parameters, 10, rng); + for (int i = 0; i < window; i++) + { + walk.template apply(P, thread_random_walk_parameters, walk_length, rng); + winPoints_per_thread[thread_index].col(i) = thread_random_walk_parameters.p.getCoefficients(); + } + + #pragma omp critical + { + estimator.update_estimator(winPoints_per_thread[thread_index]); + } + + num_generated_points_per_thread[thread_index] += window; + + #pragma omp critical + { + total_samples += window; + } + #pragma omp single + { + if (total_samples >= upper_bound_on_total_num_of_samples) + { + done = true; + } + } + TotalRandPoints_per_thread[thread_index].block(num_generated_points_per_thread[thread_index] - window, 0, window, d) = winPoints_per_thread[thread_index].transpose(); + #pragma omp single + { + if (done || (total_samples >= points_to_sample)) + { + estimator.estimate_effective_sample_size(); + + min_eff_samples = int(estimator.get_effective_sample_size().minCoeff()); + if (done && min_eff_samples < target_ess) + { + Neff_sampled = min_eff_samples; + done_all = true; + } + if (min_eff_samples >= target_ess) + { + complete = true; + Neff_sampled = min_eff_samples; + done_all = true; + } + if (min_eff_samples > 0 && !done_all) + { + points_to_sample += (total_samples / min_eff_samples) * (target_ess - min_eff_samples) + 100; + } + else if (!done_all) + { + points_to_sample = total_samples + 100; + } + } + } + } + } + + estimator.estimate_effective_sample_size(); + min_eff_samples = int(estimator.get_effective_sample_size().minCoeff()); + Neff_sampled = min_eff_samples; + if (min_eff_samples >= target_ess) + { + complete = true; + } + + unsigned int current_num_samples = 0; + for (unsigned int i = 0; i < num_threads; i++) + { + TotalRandPoints.conservativeResize(TotalRandPoints.rows() + num_generated_points_per_thread[i], d); + TotalRandPoints.block(current_num_samples, 0, num_generated_points_per_thread[i], d) = TotalRandPoints_per_thread[i].block(0, 0, num_generated_points_per_thread[i], d); + current_num_samples += num_generated_points_per_thread[i]; + TotalRandPoints_per_thread[i].resize(0, 0); + } + + return complete; +} + +#endif diff --git a/src/volesti/include/sampling/random_point_generators.hpp b/src/volesti/include/sampling/random_point_generators.hpp new file mode 100644 index 00000000..e70e1461 --- /dev/null +++ b/src/volesti/include/sampling/random_point_generators.hpp @@ -0,0 +1,395 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SAMPLERS_RANDOM_POINT_GENERATORS_HPP +#define SAMPLERS_RANDOM_POINT_GENERATORS_HPP + +template +< + typename Walk +> +struct RandomPointGenerator +{ + template + < + typename Polytope, + typename Point, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator, + typename Parameters + > + static void apply(Polytope& P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + Walk walk(P, p, rng, parameters); + for (unsigned int i=0; i + static void apply(Polytope& P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + Walk walk(P, p, rng); + for (unsigned int i=0; i +struct MultivariateGaussianRandomPointGenerator +{ + template + < + typename Polytope, + typename Point, + typename Ellipsoid, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator, + typename Parameters + > + static void apply(Polytope& P, + Point &p, // a point to start + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + Walk walk(P, p, E, rng, parameters); + for (unsigned int i=0; i + static void apply(Polytope& P, + Point &p, // a point to start + Ellipsoid const& E, // ellipsoid representing the Gaussian distribution + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + Walk walk(P, p, E, rng); + for (unsigned int i=0; i +struct GaussianRandomPointGenerator +{ + template + < + typename Polytope, + typename Point, + typename NT, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(Polytope& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + Walk walk(P, p, a_i, rng); + for (unsigned int i=0; i + static void apply(Polytope& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + Walk walk(P, p, a_i, rng, parameters); + + for (unsigned int i=0; i +struct BoundaryRandomPointGenerator +{ + template + < + typename Polytope, + typename Point, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(Polytope& P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + Walk walk(P, p, rng); + Point p1(P.dimension()), p2(P.dimension()); + for (unsigned int i=0; i +struct LogconcaveRandomPointGenerator +{ + + template + < typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Walk &walk) + { + typedef double NT; + + for (unsigned int i = 0; i < rnum; ++i) + { + // Gather one sample + walk.apply(rng, walk_length); + + // Use PushBackWalkPolicy + policy.apply(randPoints, walk.x); + } + } +}; + +template +< + typename Walk +> +struct CrhmcRandomPointGenerator +{ + + template + < + typename Polytope, + typename Point, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Parameters + > + static void apply(Polytope &P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + NegativeGradientFunctor &F, + NegativeLogprobFunctor &f, + Parameters ¶meters, + Walk &walk, + int simdLen=1, + bool raw_output= false) + { + typedef typename Walk::MT MT; + for (unsigned int i = 0; i < std::ceil((float)rnum/simdLen); ++i) + { + // Gather one sample + walk.apply(rng, walk_length); + if(walk.P.terminate){return;} + MT x; + if(raw_output){ + x=walk.x; + }else{ + x=walk.getPoints(); + } + if((i + 1) * simdLen > rnum){ + for(int j = 0; j < rnum-simdLen*i; j++){ + Point p = Point(x.col(j)); + policy.apply(randPoints, p); + } + break; + } + // Use PushBackWalkPolicy + for(int j=0; j +struct ExponentialRandomPointGenerator +{ + template + < + typename Polytope, + typename Point, + typename NT, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(Polytope& P, + Point &p, // a point to start + Point const& c, // bias function + NT const& T, // temperature/variance + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + Walk walk(P, p, c, T, rng); + bool success; + for (unsigned int i=0; i + static void apply(Polytope& P, + Point &p, // a point to start + Point const& c, // bias function + NT const& T, // temperature/variance + unsigned int const& rnum, + unsigned int const& walk_length, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + Walk walk(P, p, c, T, rng, parameters); + bool success; + + for (unsigned int i=0; i +#include +#include + + +template +struct policy_storing +{ + template + static void store(WalkPolicy &policy, PointList &randPoints, ThreadParameters &thread_random_walk_parameters) + { + policy.apply(randPoints, thread_random_walk_parameters.p); + } +}; + + +template <> +struct policy_storing +{ + template + static void store(WalkPolicy &policy, PointList &randPoints, ThreadParameters &thread_random_walk_parameters) + { + policy.apply(randPoints, thread_random_walk_parameters.p1); + policy.apply(randPoints, thread_random_walk_parameters.p2); + } +}; + +template <> +struct policy_storing : policy_storing +{}; + + +template +< + typename Walk +> +struct RandomPointGeneratorMultiThread +{ + template + < + typename Polytope, + typename Point, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator, + typename Parameters + > + static void apply(Polytope& P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + unsigned int const& num_threads, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + typedef typename Point::FT NT; + typedef typename Walk::thread_parameters_ _thread_parameters; + + omp_set_num_threads(num_threads); + unsigned int d = P.dimension(), m = P.num_of_hyperplanes(); + + std::vector num_points_per_thread(rnum%num_threads, rnum/num_threads+1); + std::vector a(num_threads - rnum%num_threads, rnum/num_threads); + num_points_per_thread.insert(num_points_per_thread.end(), a.begin(), a.end()); + + _thread_parameters thread_random_walk_parameters_temp(d, m); + Walk walk(P, thread_random_walk_parameters_temp, rng, parameters); + + #pragma omp parallel + { + int thread_index = omp_get_thread_num(); + _thread_parameters thread_random_walk_parameters(d, m); + thread_random_walk_parameters = thread_random_walk_parameters_temp; + + for (unsigned int it = 0; it < num_points_per_thread[thread_index]; it++) + { + walk.template apply(P, thread_random_walk_parameters, walk_length, rng); + #pragma omp critical + { + policy_storing::template store(policy, randPoints, thread_random_walk_parameters); + } + } + } + } + + template + < + typename Polytope, + typename Point, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(Polytope& P, + Point &p, // a point to start + unsigned int const& rnum, + unsigned int const& walk_length, + unsigned int const& num_threads, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + typedef typename Point::FT NT; + typedef typename Walk::thread_parameters_ _thread_parameters; + + omp_set_num_threads(num_threads); + unsigned int d = P.dimension(), m = P.num_of_hyperplanes(); + + std::vector num_points_per_thread(rnum%num_threads, rnum/num_threads+1); + std::vector a(num_threads - rnum%num_threads, rnum/num_threads); + num_points_per_thread.insert(num_points_per_thread.end(), a.begin(), a.end()); + + _thread_parameters thread_random_walk_parameters_temp(d, m); + Walk walk(P, thread_random_walk_parameters_temp, rng); + + #pragma omp parallel + { + int thread_index = omp_get_thread_num(); + _thread_parameters thread_random_walk_parameters(d, m); + thread_random_walk_parameters = thread_random_walk_parameters_temp; + + for (unsigned int it = 0; it < num_points_per_thread[thread_index]; it++) + { + walk.template apply(P, thread_random_walk_parameters, walk_length, rng); + #pragma omp critical + { + policy_storing::template store(policy, randPoints, thread_random_walk_parameters); + } + } + } + } +}; + + + +template +< + typename Walk +> +struct GaussianPointGeneratorMultiThread +{ + template + < + typename Polytope, + typename Point, + typename NT, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator, + typename Parameters + > + static void apply(Polytope& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& rnum, + unsigned int const& walk_length, + unsigned int const& num_threads, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng, + Parameters const& parameters) + { + typedef typename Walk::thread_parameters_ _thread_parameters; + + omp_set_num_threads(num_threads); + unsigned int d = P.dimension(), m = P.num_of_hyperplanes(); + + std::vector num_points_per_thread(rnum%num_threads, rnum/num_threads+1); + std::vector a(num_threads - rnum%num_threads, rnum/num_threads); + num_points_per_thread.insert(num_points_per_thread.end(), a.begin(), a.end()); + + _thread_parameters thread_random_walk_parameters_temp(d, m); + Walk walk(P, thread_random_walk_parameters_temp, a_i, rng, parameters); + + #pragma omp parallel + { + int thread_index = omp_get_thread_num(); + _thread_parameters thread_random_walk_parameters(d, m); + thread_random_walk_parameters = thread_random_walk_parameters_temp; + + for (unsigned int it = 0; it < num_points_per_thread[thread_index]; it++) + { + walk.template apply(P, thread_random_walk_parameters, a_i, walk_length, rng); + #pragma omp critical + { + policy_storing::template store(policy, randPoints, thread_random_walk_parameters); + } + } + } + } + + template + < + typename Polytope, + typename Point, + typename NT, + typename PointList, + typename WalkPolicy, + typename RandomNumberGenerator + > + static void apply(Polytope& P, + Point &p, // a point to start + NT const& a_i, + unsigned int const& rnum, + unsigned int const& walk_length, + unsigned int const& num_threads, + PointList &randPoints, + WalkPolicy &policy, + RandomNumberGenerator &rng) + { + typedef typename Walk::thread_parameters_ _thread_parameters; + + omp_set_num_threads(num_threads); + unsigned int d = P.dimension(), m = P.num_of_hyperplanes(); + + std::vector num_points_per_thread(rnum%num_threads, rnum/num_threads+1); + std::vector a(num_threads - rnum%num_threads, rnum/num_threads); + num_points_per_thread.insert(num_points_per_thread.end(), a.begin(), a.end()); + + _thread_parameters thread_random_walk_parameters_temp(d, m); + Walk walk(P, thread_random_walk_parameters_temp, a_i, rng); + + #pragma omp parallel + { + int thread_index = omp_get_thread_num(); + _thread_parameters thread_random_walk_parameters(d, m); + thread_random_walk_parameters = thread_random_walk_parameters_temp; + + for (unsigned int it = 0; it < num_points_per_thread[thread_index]; it++) + { + walk.template apply(P, thread_random_walk_parameters, a_i, walk_length, rng); + #pragma omp critical + { + policy_storing::template store(policy, randPoints, thread_random_walk_parameters); + } + } + } + } +}; + + +#endif // SAMPLERS_RANDOM_POINT_GENERATORS_MULTITHREAD_HPP diff --git a/src/volesti/include/sampling/sample_correlation_matrices.hpp b/src/volesti/include/sampling/sample_correlation_matrices.hpp new file mode 100644 index 00000000..9a1de231 --- /dev/null +++ b/src/volesti/include/sampling/sample_correlation_matrices.hpp @@ -0,0 +1,153 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2020 Apostolos Chalkis + +// Contributed by Huu Phuoc Le as part of Google Summer of Code 2022 program + +// Licensed under GNU LGPL.3, see LICENCE file + +/// Functions to sample correlation matrices w.r.t. a truncated density + +#ifndef VOLESTI_SAMPLING_SAMPLE_CORRELATION_MATRICES_HPP +#define VOLESTI_SAMPLING_SAMPLE_CORRELATION_MATRICES_HPP + +#include + +// New implementations for sampling correlation matrices with CorrelationSpectrahedron and CorrelationSpectrahedron_MT + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList +> +void uniform_correlation_sampling(const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + unsigned int const& nburns){ + CorrelationSpectrahedron P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(d); + RNGType rng(d); + + uniform_sampling(randPoints, P, rng, walkL, num_points, startingPoint, nburns); +} + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList +> +void uniform_correlation_sampling_MT( const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + unsigned int const& nburns){ + CorrelationSpectrahedron_MT P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(n); + RNGType rng(d); + + uniform_sampling(randPoints, P, rng, walkL, num_points, startingPoint, nburns); +} + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList, + typename NT +> +void gaussian_correlation_sampling( const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + const NT &a, + unsigned int const& nburns = 0){ + CorrelationSpectrahedron P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(d); + RNGType rng(d); + + gaussian_sampling(randPoints, P, rng, walkL, num_points, a, startingPoint, nburns); +} + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList, + typename NT +> +void gaussian_correlation_sampling_MT( const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + const NT &a, + unsigned int const& nburns = 0){ + CorrelationSpectrahedron_MT P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(n); + RNGType rng(d); + + gaussian_sampling(randPoints, P, rng, walkL, num_points, a, startingPoint, nburns); +} + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList, + typename NT, + typename VT +> +void exponential_correlation_sampling( const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + const VT &c, + const NT &T, + unsigned int const& nburns = 0){ + CorrelationSpectrahedron P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(d); + RNGType rng(d); + PointType _c(c); + + exponential_sampling(randPoints, P, rng, walkL, num_points, _c, T, startingPoint, nburns); +} + +template +< + typename WalkTypePolicy, + typename PointType, + typename RNGType, + typename PointList, + typename NT, + typename VT +> +void exponential_correlation_sampling_MT( const unsigned int &n, + PointList &randPoints, + const unsigned int &walkL, + const unsigned int &num_points, + const VT &c, + const NT &T, + unsigned int const& nburns = 0){ + CorrelationSpectrahedron_MT P(n); + const unsigned int d = P.dimension(); + PointType startingPoint(n); + RNGType rng(d); + PointType _c(c); + + exponential_sampling(randPoints, P, rng, walkL, num_points, _c, T, startingPoint, nburns); +} + +#endif //VOLESTI_SAMPLING_SAMPLE_CORRELATION_MATRICES_HPP diff --git a/src/volesti/include/sampling/sampling.hpp b/src/volesti/include/sampling/sampling.hpp new file mode 100644 index 00000000..69618609 --- /dev/null +++ b/src/volesti/include/sampling/sampling.hpp @@ -0,0 +1,603 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// VolEsti is free software: you can redistribute it and/or modify it +// under the terms of the GNU Lesser General Public License as published by +// the Free Software Foundation, either version 3 of the License, or (at +// your option) any later version. +// +// VolEsti is distributed in the hope that it will be useful, but WITHOUT +// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +// for more details. +// +// See the file COPYING.LESSER for the text of the GNU Lesser General +// Public License. If you did not receive this file along with HeaDDaCHe, +// see . + + +#ifndef SAMPLE_ONLY_H +#define SAMPLE_ONLY_H + +template +void uniform_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + //RandomNumberGenerator rng(P.dimension()); + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef RandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, nburns, walk_len, randPoints, + push_back_policy, rng); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, rnum, walk_len, randPoints, + push_back_policy, rng); + + +} + +template < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename Point +> +void uniform_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + WalkTypePolicy &WalkType, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &starting_point, + unsigned int const& nburns) +{ + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + //RandomNumberGenerator rng(P.dimension()); + PushBackWalkPolicy push_back_policy; + typedef RandomPointGenerator RandomPointGenerator; + + Point p = starting_point; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, nburns, walk_len, randPoints, + push_back_policy, rng, WalkType.param); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, rnum, walk_len, randPoints, + push_back_policy, rng, WalkType.param); +} + + +template +< + typename WalkTypePolicy, + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename Point +> +void uniform_sampling_boundary(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &starting_point, + unsigned int const& nburns) +{ + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + //RandomNumberGenerator rng(P.dimension()); + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef BoundaryRandomPointGenerator BoundaryRandomPointGenerator; + if (nburns > 0) { + BoundaryRandomPointGenerator::apply(P, p, nburns, walk_len, + randPoints, push_back_policy, rng); + randPoints.clear(); + } + unsigned int n = rnum / 2; + BoundaryRandomPointGenerator::apply(P, p, rnum / 2, walk_len, + randPoints, push_back_policy, rng); + +} + + +template +< + typename WalkTypePolicy, + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename NT, + typename Point +> +void gaussian_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const NT &a, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + //RandomNumberGenerator rng(P.dimension()); + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef GaussianRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, a, nburns, walk_len, randPoints, + push_back_policy, rng); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, a, rnum, walk_len, randPoints, + push_back_policy, rng); + + +} + + +template < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename NT, + typename Point + > +void gaussian_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + WalkTypePolicy &WalkType, + const unsigned int &walk_len, + const unsigned int &rnum, + const NT &a, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + //RandomNumberGenerator rng(P.dimension()); + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef GaussianRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, a, nburns, walk_len, randPoints, + push_back_policy, rng, WalkType.param); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, a, rnum, walk_len, randPoints, + push_back_policy, rng, WalkType.param); +} + +template < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename NT, + typename Point, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename Solver + > +void logconcave_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &starting_point, + unsigned int const& nburns, + NegativeGradientFunctor &F, + NegativeLogprobFunctor &f) +{ + typedef typename WalkTypePolicy::template Walk + < + Point, + Polytope, + RandomNumberGenerator, + NegativeGradientFunctor, + NegativeLogprobFunctor, + Solver + > walk; + + typedef typename WalkTypePolicy::template parameters + < + NT, + NegativeGradientFunctor + > walk_params; + + // Initialize random walk parameters + unsigned int dim = starting_point.dimension(); + walk_params params(F, dim); + + if (F.params.eta > 0) { + params.eta = F.params.eta; + } + + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + walk logconcave_walk(&P, p, F, f, params); + + typedef LogconcaveRandomPointGenerator RandomPointGenerator; + + if (nburns > 0) { + RandomPointGenerator::apply(nburns, walk_len, randPoints, + push_back_policy, rng, logconcave_walk); + } + logconcave_walk.disable_adaptive(); + randPoints.clear(); + + RandomPointGenerator::apply(rnum, walk_len, randPoints, + push_back_policy, rng, logconcave_walk); +} +#include "preprocess/crhmc/crhmc_input.h" +#include "preprocess/crhmc/crhmc_problem.h" +template + < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename NT, + typename Point, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename HessianFunctor, + typename Solver + > +void crhmc_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const int walk_len, + const unsigned int rnum, + const unsigned int nburns, + NegativeGradientFunctor &F, + NegativeLogprobFunctor &f, + HessianFunctor &h, + int simdLen = 1, + bool raw_output=false) { + typedef typename Polytope::MT MatrixType; + typedef crhmc_input + < + MatrixType, + Point, + NegativeLogprobFunctor, + NegativeGradientFunctor, + HessianFunctor + > Input; + Input input = convert2crhmc_input(P, f, F, h); + typedef crhmc_problem CrhmcProblem; + CrhmcProblem problem = CrhmcProblem(input); + if(problem.terminate){return;} + typedef typename WalkTypePolicy::template Walk + < + Point, + CrhmcProblem, + RandomNumberGenerator, + NegativeGradientFunctor, + NegativeLogprobFunctor, + Solver + > walk; + typedef typename WalkTypePolicy::template parameters + < + NT, + NegativeGradientFunctor + > walk_params; + Point p = Point(problem.center); + problem.options.simdLen=simdLen; + walk_params params(input.df, p.dimension(), problem.options); + + if (input.df.params.eta > 0) { + params.eta = input.df.params.eta; + } + + PushBackWalkPolicy push_back_policy; + + walk crhmc_walk = walk(problem, p, input.df, input.f, params); + + typedef CrhmcRandomPointGenerator RandomPointGenerator; + + RandomPointGenerator::apply(problem, p, nburns, walk_len, randPoints, + push_back_policy, rng, F, f, params, crhmc_walk); + //crhmc_walk.disable_adaptive(); + randPoints.clear(); + RandomPointGenerator::apply(problem, p, rnum, walk_len, randPoints, + push_back_policy, rng, F, f, params, crhmc_walk, simdLen, raw_output); +} +#include "ode_solvers/ode_solvers.hpp" +template < + typename Polytope, + typename RNGType, + typename PointList, + typename NegativeGradientFunctor, + typename NegativeLogprobFunctor, + typename HessianFunctor, + typename CRHMCWalk, + int simdLen=1 +> +void execute_crhmc(Polytope &P, RNGType &rng, PointList &randPoints, + unsigned int const& walkL, unsigned int const& numpoints, + unsigned int const& nburns, NegativeGradientFunctor *F=NULL, + NegativeLogprobFunctor *f=NULL, HessianFunctor *h=NULL, bool raw_output= false){ +typedef typename Polytope::MT MatrixType; +typedef typename Polytope::PointType Point; +typedef typename Point::FT NT; +if(h!=NULL){ +typedef crhmc_input + < + MatrixType, + Point, + NegativeLogprobFunctor, + NegativeGradientFunctor, + HessianFunctor + > Input; +typedef crhmc_problem CrhmcProblem; +crhmc_sampling < + PointList, + Polytope, + RNGType, + CRHMCWalk, + NT, + Point, + NegativeGradientFunctor, + NegativeLogprobFunctor, + HessianFunctor, + ImplicitMidpointODESolver < + Point, + NT, + CrhmcProblem, + NegativeGradientFunctor, + simdLen + > +>(randPoints, P, rng, walkL, numpoints, nburns, *F, *f, *h, simdLen, raw_output); +}else{ + typedef crhmc_input + < + MatrixType, + Point, + NegativeLogprobFunctor, + NegativeGradientFunctor, + ZeroFunctor + > Input; + typedef crhmc_problem CrhmcProblem; + ZeroFunctor zerof; +crhmc_sampling < + PointList, + Polytope, + RNGType, + CRHMCWalk, + NT, + Point, + NegativeGradientFunctor, + NegativeLogprobFunctor, + ZeroFunctor, + ImplicitMidpointODESolver < + Point, + NT, + CrhmcProblem, + NegativeGradientFunctor, + simdLen + > +>(randPoints, P, rng, walkL, numpoints, nburns, *F, *f, zerof, simdLen, raw_output); +} +} +template +< + typename WalkTypePolicy, + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename NT, + typename Point +> +void exponential_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &c, + const NT &a, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef ExponentialRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, c, a, nburns, walk_len, randPoints, + push_back_policy, rng); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, c, a, rnum, walk_len, randPoints, + push_back_policy, rng); +} + + +template < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename NT, + typename Point + > +void exponential_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + WalkTypePolicy &WalkType, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &c, + const NT &a, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef ExponentialRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, c, a, nburns, walk_len, randPoints, + push_back_policy, rng, WalkType.param); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, c, a, rnum, walk_len, randPoints, + push_back_policy, rng, WalkType.param); +} + + +template +< + typename WalkTypePolicy, + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename NT, + typename Point +> +void exponential_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &c, + const NT &a, + const NT &eta, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef ExponentialRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, c, a, eta, nburns, walk_len, randPoints, + push_back_policy, rng); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, c, a, eta, rnum, walk_len, randPoints, + push_back_policy, rng); +} + + +template < + typename PointList, + typename Polytope, + typename RandomNumberGenerator, + typename WalkTypePolicy, + typename NT, + typename Point + > +void exponential_sampling(PointList &randPoints, + Polytope &P, + RandomNumberGenerator &rng, + WalkTypePolicy &WalkType, + const unsigned int &walk_len, + const unsigned int &rnum, + const Point &c, + const NT &a, + const NT &eta, + const Point &starting_point, + unsigned int const& nburns) +{ + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + PushBackWalkPolicy push_back_policy; + + Point p = starting_point; + + typedef ExponentialRandomPointGenerator RandomPointGenerator; + if (nburns > 0) { + RandomPointGenerator::apply(P, p, c, a, eta, nburns, walk_len, randPoints, + push_back_policy, rng, WalkType.param); + randPoints.clear(); + } + RandomPointGenerator::apply(P, p, c, a, eta, rnum, walk_len, randPoints, + push_back_policy, rng, WalkType.param); +} + + +#endif diff --git a/src/volesti/include/sampling/simplex.hpp b/src/volesti/include/sampling/simplex.hpp new file mode 100644 index 00000000..92534993 --- /dev/null +++ b/src/volesti/include/sampling/simplex.hpp @@ -0,0 +1,392 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018-2020 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// VolEsti is free software: you can redistribute it and/or modify it +// under the terms of the GNU Lesser General Public License as published by +// the Free Software Foundation, either version 3 of the License, or (at +// your option) any later version. +// +// VolEsti is distributed in the hope that it will be useful, but WITHOUT +// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +// for more details. +// +// See the file COPYING.LESSER for the text of the GNU Lesser General +// Public License. If you did not receive this file along with HeaDDaCHe, +// see . + +#ifndef SAMPLERS_SIMPLEX_HPP +#define SAMPLERS_SIMPLEX_HPP + +#ifndef isnan + using std::isnan; +#endif + +template +void Sam_Unit(unsigned int dim, + unsigned int num, + std::list &points, + double seed = std::numeric_limits::signaling_NaN()) +{ + + unsigned int j,i,x_rand,M=2147483647,pr,divisors,pointer; // M is the largest possible integer + std::vector x_vec; + std::vector y; + + boost::random::uniform_int_distribution<> uidist(1,M); + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed2 = seed; + rng.seed(rng_seed2); + } + //unsigned seed = std::chrono::system_clock::now().time_since_epoch().count(); + //RNGType rng(seed); + + if (dim<=60){ + + x_vec.assign(dim+1,0); + + for (i=0; i filter; + bool t1=true,t2=true; + pr=3*dim+1; + if(pr%2==0) pr+=1; + + while(t1){ + t2=true; + divisors=(int)floor(sqrt((NT)pr))+1; + for (i=3; i x_vec2; + NT Ti,sum; + + x_vec2.assign(dim+1,0.0); + + // Generate the number of points requested + for (i=0; i +void Sam_Canon_Unit(unsigned int dim, + unsigned int num, + std::list &points, + double seed = std::numeric_limits::signaling_NaN()) +{ + + unsigned int j,i,x_rand,M=2147483647,pointer; // M is the largest possible integer + std::vector y; + dim--; + boost::random::uniform_int_distribution<> uidist(1,M); + + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + if (!isnan(seed)) { + unsigned rng_seed2 = seed; + rng.seed(rng_seed2); + } + + std::vector x_vec2; + NT Ti,sum; + + x_vec2.assign(dim+1,0.0); + + // Generate the number of points requested + for (i=0; i +void Sam_arb_simplex(const Vpolytope &P, unsigned int num, PointList &points){ + + typedef typename Vpolytope::MT MT; + typedef typename Vpolytope::NT NT; + typedef typename Vpolytope::rngtype RNGType; + typedef typename Vpolytope::PolytopePoint Point; + + MT V = P.get_mat(); + std::vector vec_point; + + unsigned int n=V.rows(),j,i,k,x_rand,M=2147483647,pr,divisors,pointer; // M is the largest possible integer + unsigned int dim = V.cols(); + std::vector temp_p(dim, 0.0); + std::vector x_vec; + std::vector y; + + for (int i = 0; i < V.rows(); ++i) { + for (int j = 0; j < V.cols(); ++j) { + temp_p[j] = V(i,j); + } + vec_point.push_back(Point(dim, temp_p.begin(), temp_p.end())); + } + typename std::vector::iterator it_beg = vec_point.begin(); + + NT Xj; + Point p0=*it_beg; + + boost::random::uniform_int_distribution<> uidist(1,M); + unsigned seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(seed); + + if (dim<=60){ + + x_vec.assign(dim+2,0); x_vec[dim+1]=M; + + for (i=0; i filter; + x_vec.assign(dim+2,0); x_vec[dim+1]=M; + + bool t1=true,t2=true; + pr=3*dim+1; + if(pr%2==0) pr+=1; + + while(t1){ + t2=true; + divisors=(int)floor(sqrt((NT)pr))+1; + for (i=3; i x_vec2; + NT Ti,sum; + x_vec2.assign(dim+1,0.0); + + // Generate the number of points requested + for (i=0; i +struct GetDirection +{ + typedef typename Point::FT NT; + + template + inline static Point apply(unsigned int const& dim, + RandomNumberGenerator &rng, + bool normalize=true) + { + NT normal = NT(0); + Point p(dim); + NT* data = p.pointerToData(); + + if(normalize) + { + for (unsigned int i=0; i +struct GetDirection> +{ + template + inline static CorreMatrix apply(unsigned int const& dim, + RandomNumberGenerator &rng, + bool normalize=true) + { + typedef Eigen::Matrix MT; + typedef Eigen::Matrix VT; + + unsigned int n = std::ceil(std::sqrt(2*dim)); + MT mat = MT::Zero(n,n); + NT normal = NT(0), coeff; + + int i, j; + + if(normalize) + { + for(i = 0; i < n ; ++i) + { + for(j = 0; j < i; ++j) + { + coeff = rng.sample_ndist(); + mat(i,j) = coeff; + normal += coeff * coeff; + } + } + normal = NT(1)/std::sqrt(normal); + mat *= normal; + }else + { + for(i = 0; i < n ; ++i) + { + for(j = 0; j < i; ++j) + { + coeff = rng.sample_ndist(); + mat(i,j) = coeff; + } + } + } + return CorreMatrix(mat); + } +}; + +template +struct GetPointInDsphere +{ + template + inline static Point apply(unsigned int const& dim, + NT const& radius, + RandomNumberGenerator &rng) + { + Point p = GetDirection::apply(dim, rng); + NT U = rng.sample_urdist(); + U = std::pow(U, NT(1)/(NT(dim))); + p *= radius * U; + return p; + } +}; + +template +struct GetPointOnDsphere +{ + template + inline static Point apply(unsigned int const& dim, + NT const& radius, + RandomNumberGenerator &rng) + { + Point p = GetDirection::apply(dim, rng); + if (radius != 0) p *= radius; + return p; + } +}; + + + +#endif // SPHERE_HPP diff --git a/src/volesti/include/sos/NonSymmetricIPM.h b/src/volesti/include/sos/NonSymmetricIPM.h new file mode 100644 index 00000000..b5c5b95e --- /dev/null +++ b/src/volesti/include/sos/NonSymmetricIPM.h @@ -0,0 +1,333 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file +#ifndef SOS_NONSYMMETRICIPM_H +#define SOS_NONSYMMETRICIPM_H + +#include "barriers/LHSCB.h" +#include "barriers/ProductBarrier.h" +#include +#include +#include + + +template +class Instance { +public: + Constraints constraints; + LHSCB *barrier; +}; + +template +class DirectionDecomposition { +public: + DirectionDecomposition(Vector v, unsigned const n, unsigned const m) { + assert(v.rows() == m + n + 1 + n + 1); + y = v.block(0, 0, m, 1); + x = v.block(m, 0, n, 1); + tau = v.block(m + n, 0, 1, 1).sum(); + s = v.block(m + n + 1, 0, n, 1); + kappa = v.block(m + n + 1 + n, 0, 1, 1).sum(); + } + + friend std::ostream &operator<<(std::ostream &os, const DirectionDecomposition &dir); + + Vector y, x, s; + T kappa, tau; +}; + +template +class ErrorConstants { +public: + + ErrorConstants() {}; + + template + ErrorConstants(const ErrorConstants other) { + //Note: Boost Dependency + primal = boost::numeric_cast(other.primal); + dual = boost::numeric_cast(other.dual); + complementary = boost::numeric_cast(other.complementary); + } + + void set(Matrix A, Vector b, Vector c) { + //Constant are used to conform with the implementation by Papp & Yildiz. + primal = std::max(T(1), sqrt(pow(A.norm(), 2) + pow(b.norm(), 2))); + Matrix Id = Matrix::Identity(A.cols(), A.cols()); + dual = std::max(T(1), sqrt(pow(A.transpose().norm(), 2) + pow(Id.norm(), 2) + pow(c.norm(), 2))); + complementary = sqrt(pow(c.norm(), 2) + pow(b.norm(), 2) + 1); + } + + T primal; + T dual; + T complementary; +}; + +template +class NonSymmetricIPM { + + typedef Matrix Matrix; + typedef Vector Vector; + +public: + + NonSymmetricIPM(Matrix &A_, Vector &b_, Vector &c_, LHSCB *barrier_); + + NonSymmetricIPM(Instance &instance) : NonSymmetricIPM(instance.constraints.A, + instance.constraints.b, instance.constraints.c, + instance.barrier) { + _logger->set_level(spdlog::level::info); + }; + + NonSymmetricIPM(Instance &, std::string); + + + template + void cast_members_from(const NonSymmetricIPM &other) { + A_sparse = other.A_sparse.template cast(); + _basis_ker_A = other._basis_ker_A.template cast(); + _config = other._config; + + _num_predictor_steps = other._num_predictor_steps; + _num_corrector_steps = other._num_corrector_steps; + _param_step_length_predictor = boost::numeric_cast(other._param_step_length_predictor); + _step_length_predictor = boost::numeric_cast(other._step_length_predictor); + _step_length_corrector = boost::numeric_cast(other._step_length_corrector); + _epsilon = boost::numeric_cast(other._epsilon); + + _large_neighborhood = boost::numeric_cast(other._large_neighborhood); + _small_neighborhood = boost::numeric_cast(other._small_neighborhood); + //copy current solution + + x = other.x.template cast(); + y = other.y.template cast(); + s = other.s.template cast(); + kappa = boost::numeric_cast(other.kappa); + tau = boost::numeric_cast(other.tau); + + _last_predictor_direction = other._last_predictor_direction.template cast(); + + _err_consts = ErrorConstants(other._err_consts); + + //copy configuration; + + _use_line_search = other._use_line_search; + } + + template + NonSymmetricIPM *cast_with_product_barrier() { + + //cast initialisatino data + Eigen::Matrix A_ = A.template cast(); + Eigen::Matrix b_ = b.template cast(); + Eigen::Matrix c_ = c.template cast(); + + //TODO: High priority. Figure out how to undo the cast to the ProductBarrier. + ProductBarrier *barrier_ = static_cast *>(_barrier)->template cast(); + + assert(barrier_ != nullptr); + + NonSymmetricIPM *nonSymmetricIPM = new NonSymmetricIPM(A_, b_, c_, barrier_); + + //cast members + + nonSymmetricIPM->template cast_members_from(*this); + + //copy configuration; + + assert(nonSymmetricIPM->x.rows() == nonSymmetricIPM->s.rows()); + assert(nonSymmetricIPM->x.rows() == nonSymmetricIPM->c.rows()); + + return nonSymmetricIPM; + } + + IPMDouble calc_step_length_predictor() { + IPMDouble const epsilon = .5; + IPMDouble const eta = _large_neighborhood * pow(epsilon, _num_corrector_steps); + IPMDouble const k_x = eta + sqrt(2 * eta * eta + _barrier->concordance_parameter(x) + 1); + return _param_step_length_predictor / k_x; + } + + int run_solver(); + + enum Termination { + SUCCESS, + FAILURE + }; + + inline IPMDouble primal_error() { + return (A * x - tau * b).norm() / _err_consts.primal; + } + + inline IPMDouble dual_error() { + return (A.transpose() * y + s - tau * c).norm() / _err_consts.dual; + } + + inline IPMDouble primal_error_rescaled() { + return (A * x / tau - b).norm(); + } + + inline IPMDouble dual_error_rescaled() { + return (A.transpose() * y / tau + s / tau - c).norm(); + } + + inline IPMDouble duality_gap() { + return x.dot(s) / _barrier->concordance_parameter(x); + + } + + inline IPMDouble complementarity() { + return abs(c.dot(x) - b.dot(y)) / _err_consts.complementary; + } + + bool verify_solution(IPMDouble precision = 10e-5) { + if (not _barrier->in_interior(x)) { + return false; + } + if (primal_error_rescaled() > precision) { + return false; + } + if (dual_error_rescaled() > precision) { + return false; + } + + //TODO: Add duality gap check. In general we don't have strict duality as exploited below. + if (duality_gap() > precision) { + return false; + } + return true; + } + + Solution get_solution() { + Solution sol; + assert(tau != 0); + sol.x = x / tau; + sol.s = s / tau; + sol.centrality = centrality(); + sol.gap = mu(); + return sol; + } + + std::shared_ptr _logger; + std::shared_ptr _benchmark_logger; + + + //TODO:Figure out how to make these variables while being able to compile it. + + Matrix A; + Vector b; + Vector c; + + Eigen::SparseMatrix A_sparse; + + //Matrix whose columns are a basis of the kernel of A. Currently unused. + Matrix _basis_ker_A; + + Vector x; + Vector y; + Vector s; + + pt::ptree _config; + + unsigned _num_predictor_steps = 500; + unsigned _num_corrector_steps; + + IPMDouble _param_step_length_predictor = 0.02; + + IPMDouble _step_length_predictor; + IPMDouble _step_length_corrector; + + IPMDouble _epsilon = 10e-5; + + unsigned _total_num_line_steps; + + //Large neighborhood + IPMDouble _large_neighborhood; + //Small neighborhood + IPMDouble _small_neighborhood; + + bool _check_centrality_in_every_segment = true; + + bool _type_cast_if_unsuccessful = true; + + bool _use_line_search = true; + + IPMDouble kappa, tau; + + Vector _last_predictor_direction; + + ErrorConstants _err_consts; + + void initialize(); + +private: + + cxxtimer::Timer _predictor_timer; + cxxtimer::Timer _corrector_timer; + cxxtimer::Timer _andersen_sys_timer; + cxxtimer::Timer _centrality_timer; + + std::vector _custom_timers; + + cxxtimer::Timer _general_method_timer; + cxxtimer::Timer _total_runtime_timer; + + + void set_configuration_variables(); + + bool terminate(); + + bool terminate_successfully_wrapper(); + + bool terminate_successfully(); + + bool terminate_infeasible_wrapper(); + + bool terminate_infeasible(); + + + Vector _stored_x_centrality; + Vector _stored_s_centrality; + IPMDouble _stored_centrality_error; + + + IPMDouble mu(); + + Vector psi(IPMDouble t); + + LHSCB *_barrier; + + std::vector > + solve_andersen_andersen_subsystem(std::vector > &); + + Vector andersen_andersen_solve(Vector const); + + Vector solve(Matrix &, Vector const); + + Vector create_predictor_RHS(); + + Vector create_corrector_RHS(); + + Vector solve_predictor_system(); + + Vector solve_corrector_system(); + + IPMDouble centrality(); + + + void print_status(); + + void apply_update(Vector concat); + + Vector build_update_vector(); + + void test_gradient(); + + void test_hessian(); +}; + +#include "NonSymmetricIPM.hpp" + +#endif //SOS_NONSYMMETRICIPM_H diff --git a/src/volesti/include/sos/NonSymmetricIPM.hpp b/src/volesti/include/sos/NonSymmetricIPM.hpp new file mode 100644 index 00000000..a634c545 --- /dev/null +++ b/src/volesti/include/sos/NonSymmetricIPM.hpp @@ -0,0 +1,781 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include +#include "NonSymmetricIPM.h" +#include "spdlog/sinks/stdout_color_sinks.h" +#include "spdlog/fmt/ostr.h" +#include "barriers/ProductBarrier.h" +#include "barriers/SumBarrier.h" +#include "barriers/InterpolantDualSOSBarrier.h" + +template +std::ostream &operator<<(std::ostream &os, const DirectionDecomposition &dir) { + os << "x " << dir.x.transpose() << std::endl; + os << "s: " << dir.s.transpose() << std::endl; + os << "y: " << dir.y.transpose() << std::endl; + os << "kappa: " << dir.kappa << ", tau: " << dir.tau << std::endl; + return os; +} + + +template +NonSymmetricIPM::NonSymmetricIPM(Instance &instance, std::string config_json) : NonSymmetricIPM( + instance) { + pt::read_json(config_json, _config); + _config = _config.get_child("IPM"); + std::cout << "NonSymmetricIPM configuration..." << std::endl; + pt::write_json(std::cout, _config); + initialize(); + set_configuration_variables(); +} + +template +void NonSymmetricIPM::set_configuration_variables() { + _epsilon = _config.get("epsilon"); + _logger->info("epsilon set to {}", _epsilon); + _num_corrector_steps = _config.get("num_corrector_steps"); + _large_neighborhood = _config.get("large_neighborhood"); + _small_neighborhood = _config.get("small_neighborhood"); + _param_step_length_predictor = _config.get("scale_predictor_step"); + _step_length_predictor = calc_step_length_predictor(); + _step_length_corrector = _config.get("length_corrector_step"); + _check_centrality_in_every_segment = _config.get("check_centrality_in_every_segment"); + _type_cast_if_unsuccessful = _config.get("type_cast_if_unsuccessful"); + + _logger->info("Set log level to {}", spdlog::level::level_enum(_config.get("logger_level"))); + _logger->set_level(spdlog::level::level_enum(_config.get("logger_level"))); + + _use_line_search = _config.get("use_line_search"); +} + +template +Vector NonSymmetricIPM::solve(Matrix &M_, Vector const v_) { + Vector sol = M_.colPivHouseholderQr().solve(v_); + return sol; +} + +//TODO:For sparse systems we might create a dense matrix in the LLT decomposition. Implement separate solver for this case +//TODO:make sure all memory is preallocated. + +template +std::vector, Vector > > +NonSymmetricIPM::solve_andersen_andersen_subsystem( + std::vector > &v) { + + _custom_timers[8].start(); + _custom_timers[4].start(); + //TODO: double transposition. Figure out how to multiply solve from RHS. +// Matrix A_H_inv = LLT.solve(A.transpose()).transpose() / mu(); + Matrix A_H_inv = _barrier->llt_solve(x, A.transpose()).transpose() / mu(); + A_H_inv.eval(); + _custom_timers[4].stop(); + _custom_timers[5].start(); + + //TODO: Use better method to sparsify A. + Matrix A_H_inv_A_top = A_H_inv * A_sparse.transpose(); + A_H_inv_A_top.eval(); + + _custom_timers[5].stop(); + _custom_timers[6].start(); + Eigen::LLT A_H_inv_A_top_LLT = A_H_inv_A_top.llt(); + _custom_timers[6].stop(); + _custom_timers[8].stop(); + _custom_timers[9].start(); + + std::vector > results; + //TODO: might be possible to solve these system in batches! + //TODO: check whether Conjugate Gradient Method solves Normal Equations more efficiently. + for (unsigned i = 0; i < v.size(); i++) { + Vector &r1 = v[i].first; + Vector &r2 = v[i].second; + Vector r2_solve = A * _barrier->llt_solve(x, r2) / mu(); +// Vector new_s_intermediate = A_H_inv_A_top_LLT.matrixL().solve(-(r2_solve - r1)); +// Vector new_s = A_H_inv_A_top_LLT.matrixU().solve(new_s_intermediate); + Vector new_s = A_H_inv_A_top_LLT.solve(-(r2_solve - r1)); +// Vector new_t_intermediate = LLT.matrixL().solve((r2 + A.transpose() * new_s) / mu()); +// Vector new_t = LLT.matrixU().solve(new_t_intermediate); + Vector new_t = _barrier->llt_solve(x, (r2 + A.transpose() * new_s) / mu()); + results.emplace_back(std::pair(new_s, new_t)); + } + _custom_timers[9].stop(); + return results; +} + +template +Vector NonSymmetricIPM::andersen_andersen_solve(Vector const rhs) { + + _andersen_sys_timer.start(); + + const int m = A.rows(); + const int n = A.cols(); + + //TODO: figure out whether rescaling makes sense. + Vector const r_p = rhs.segment(0, m); + Vector const r_d = rhs.segment(m, n); + Vector const r_g = rhs.segment(m + n, 1); + Vector const r_xs = rhs.segment(m + n + 1, n); + Vector const r_tk = rhs.segment(m + n + 1 + n, 1); + + Matrix mu_H_x = mu() * _barrier->hessian(x); + IPMDouble mu_H_tau = mu() / (tau * tau); + + std::vector > new_rhs_vectors; + new_rhs_vectors.emplace_back(std::pair(b, -c)); + new_rhs_vectors.emplace_back(std::pair(r_p, r_d + r_xs)); + + std::vector > ret = solve_andersen_andersen_subsystem(new_rhs_vectors); + + Vector p = ret[0].first; + Vector q = ret[0].second; + + Vector u = ret[1].first; + Vector v = ret[1].second; + + Vector pq(m + n); + pq << p, q; + + Vector uv(m + n); + uv << u, v; + + SPDLOG_TRACE("{}", pq.segment(m, n).transpose()); + SPDLOG_TRACE("{}", q.transpose()); + + Vector rhs_uv(m + n); + rhs_uv << r_p, r_d + r_xs; + + IPMDouble d_tau = (r_g.sum() + r_tk.sum() - b.dot(u) + c.dot(v)) / (mu() / (tau * tau) + b.dot(p) - c.dot(q)); + + Vector d_yx = uv + d_tau * pq; + Vector d_x = d_yx.segment(m, n); + Vector d_s = r_xs - mu_H_x * d_x; + IPMDouble d_kappa = r_tk.sum() - mu_H_tau * d_tau; + + Vector d_tau_vec(1); + d_tau_vec(0) = d_tau; + Vector d_kappa_vec(1); + d_kappa_vec(0) = d_kappa; + + Vector sol(m + n + 1 + n + 1); + sol << d_yx, d_tau_vec, d_s, d_kappa_vec; + + _andersen_sys_timer.stop(); + + //Check for dual error in this solution and potentially run iterative refinement. + + Vector dual_err = -A.transpose() * d_yx.segment(0, y.rows()) + d_tau * c - d_s - r_d; + IPMDouble rel_dual_error = dual_err.norm() / r_d.norm(); + + _logger->debug("relative dual error is {}", rel_dual_error); + if (rel_dual_error > 10e-2) { + _logger->warn("relative dual error is {}", rel_dual_error); + } + return sol; +} + +template +Vector NonSymmetricIPM::build_update_vector() { + Vector concat(y.rows() + x.rows() + 1 + s.rows() + 1); + concat << y, x, tau * Matrix::Identity(1, 1), s, kappa * Matrix::Identity(1, 1); + return concat; +} + +template +void NonSymmetricIPM::apply_update(Vector concat) { + y = concat.block(0, 0, y.rows(), 1); + x = concat.block(y.rows(), 0, x.rows(), 1); + tau = concat.block(y.rows() + x.rows(), 0, 1, 1).sum(); + s = concat.block(y.rows() + x.rows() + 1, 0, s.rows(), 1); + kappa = concat.block(y.rows() + x.rows() + 1 + s.rows(), 0, 1, 1).sum(); +} + +//Heuristic for steplength. +template +IPMDouble calc_new_alpha(int num_steps, IPMDouble step_length) { + IPMDouble scale_fac = pow(2, num_steps) * step_length; + IPMDouble const threshold = .4; + if (scale_fac > threshold) { + IPMDouble ratio = log2(scale_fac / threshold); + scale_fac = 1. - (1 - threshold) / pow(2, ratio); + } + return scale_fac; +} + +template +int NonSymmetricIPM::run_solver() { + + _logger->info("Solver started."); + + _logger->trace("b: {}", b.transpose()); + _logger->trace("c: {}", c.transpose()); + _logger->trace("gradient: {}", _barrier->gradient(x).transpose()); + + print_status(); + + _total_num_line_steps = 0; + _total_runtime_timer.start(); + unsigned pred_iteration = 0; + for (; pred_iteration < _num_predictor_steps; ++pred_iteration) { + Vector vec_begin_predictor_direction = build_update_vector(); + _logger->debug("Begin predictor iteration {}", pred_iteration); + _predictor_timer.start(); + if (terminate_successfully_wrapper()) { + _logger->info("Interior point method terminated successfully with required proximity."); + break; + } + + if (terminate_infeasible_wrapper()) { + _logger->info("Interior point method terminated with infeasible solution."); + break; + } + +#ifndef NDEBUG + if (centrality() > _large_neighborhood) { + _logger->warn("Centrality at beginning of predictor step is {}, large neighborhood is {}", centrality(), + _large_neighborhood); + } +#endif + + _logger->trace("Solve predictor system..."); + Vector predictor_direction = solve_predictor_system(); + _logger->trace("Finished solving predictor system"); + + + DirectionDecomposition pred_dir(predictor_direction, x.rows(), y.rows()); + + //Check predictor direction + + IPMDouble err_primal = (A * pred_dir.x - b * pred_dir.tau + A * x - b * tau).norm() + / (A * x - b * tau).norm(); + IPMDouble err_dual = (-A.transpose() * pred_dir.y + c * pred_dir.tau - pred_dir.s + - A.transpose() * y + c * tau - s).norm() / (A.transpose() * y - c * tau + s).norm(); + IPMDouble err_opt = abs(b.dot(pred_dir.y) - c.dot(pred_dir.x) - pred_dir.kappa + + b.dot(y) - c.dot(x) - kappa) / abs(-b.dot(y) + c.dot(x) + kappa); + IPMDouble err_cent = (pred_dir.s + mu() * _barrier->hessian(x) * pred_dir.x + s).norm() / s.norm(); + IPMDouble err_cent2 = abs(pred_dir.kappa + mu() / (tau * tau) * pred_dir.tau + kappa) / kappa; + + IPMDouble err_sum = err_primal + err_dual + err_opt + err_cent + err_cent2; + + if (_logger->level() <= spdlog::level::debug) { + _logger->debug("Errors in predictor direction {} {} {} {} {}", err_primal, err_dual, err_opt, err_cent, + err_cent2); + } + + const IPMDouble ERR_THRESHOLD = .1; + if (err_sum > ERR_THRESHOLD) { + _logger->warn("Error in predictor direction too big. Terminate. (Error is {}. Threshold is {})", + err_sum, ERR_THRESHOLD); + _logger->info("Predictor direction: "); + std::cout << pred_dir.x.transpose() << std::endl << pred_dir.s.transpose() << std::endl; + //TODO: Iterative refinement. + return Termination::FAILURE; + } + + + Vector curr_vec = build_update_vector(); + Vector orig_vector = curr_vec; + unsigned num_line_steps = 0; + if (not _use_line_search) { + curr_vec += _step_length_predictor * predictor_direction; + apply_update(curr_vec); + + } else { + //find longest step length such that we remain in beta environment. + //TODO: find sophisticated way of computing this efficiently. Currently we do simple repeated squaring, + // irrespective of the barrier function. + + Vector fallback_vec = orig_vector; + curr_vec = orig_vector + calc_new_alpha(num_line_steps, _step_length_predictor) * predictor_direction; + apply_update(curr_vec); + while (kappa > 0 and tau > 0 and _barrier->in_interior(x) + and (not _check_centrality_in_every_segment or (centrality() < _large_neighborhood))) { + _logger->debug("Another iteration in line search..."); + fallback_vec = curr_vec; + curr_vec = orig_vector + calc_new_alpha(num_line_steps, _step_length_predictor) * predictor_direction; + num_line_steps++; + apply_update(curr_vec); + } + + //Dynamically adjust step length + if (num_line_steps == 0) { + _step_length_predictor /= 2; + } + + if (num_line_steps > 3) { + _step_length_predictor *= 2; + } + + if (num_line_steps == 0) { + _logger->info("Reason for stopping: "); + if (not _barrier->in_interior(x)) { + _logger->info("Not in interior"); + } else if (centrality() >= _large_neighborhood) { + _logger->info("Centrality {} is worse than neighborhood {}", centrality(), _large_neighborhood); + } + } + _logger->debug("Applied {} line steps in iteration {}", num_line_steps, pred_iteration); + apply_update(fallback_vec); + +#ifndef NDEBUG + if(not terminate_successfully() and num_line_steps == 0){ + _logger->warn("Could not perform a single predictor step"); + } +#endif + } + + _total_num_line_steps += num_line_steps; + _logger->info("End of predictor step {} with {} line steps and total num line steps {}:", + pred_iteration, num_line_steps, _total_num_line_steps); + if (_logger->level() <= spdlog::level::info) { + print_status(); + } + + _predictor_timer.stop(); + _corrector_timer.start(); + + for (unsigned corr_iteration = 0; corr_iteration < _num_corrector_steps; ++corr_iteration) { + //TODO: figure out if this is already as expensive as running another corrector step. (probably not, as the crrent value can be used for next predictor step if true). + + IPMDouble cur_centrality = centrality(); + if (cur_centrality < _small_neighborhood) { + _logger->info("Central enough to skip corrector iteration {}", corr_iteration); + break; + } + + Vector psi_full = psi(mu()); + + //TODO: Use Woodburry matrix identity for corrector step. + + Vector corrector_direction = solve_corrector_system(); + + DirectionDecomposition dir(corrector_direction, x.rows(), y.rows()); + Vector concat = build_update_vector(); + concat += _step_length_corrector * corrector_direction; + apply_update(concat); + + if (centrality() > cur_centrality / 2) { + //In this case we are too far away from the central path. Revert back to and shrink large neighborhood + _logger->info("Shrink large neighborhood and use stored iterate, " + "because centering step converges too slowly."); + apply_update(vec_begin_predictor_direction); + _large_neighborhood = _small_neighborhood + (_large_neighborhood - _small_neighborhood) / 2; + break; + } + + _logger->info("End of corrector step {} :", corr_iteration); + if (_logger->level() <= spdlog::level::info) { + print_status(); + } + + if (_logger->level() <= spdlog::level::debug) { + assert(kappa > 0); + assert(tau > 0); + assert(_barrier->in_interior(x)); + assert(centrality() < _large_neighborhood); + } + } + _corrector_timer.stop(); + } + + _total_runtime_timer.stop(); + _benchmark_logger->info("{} {} {} {} {}", (_barrier->getNumVariables() / 2 - 1) / 2, pred_iteration + 1, + _total_runtime_timer.count() / 1000., + _total_runtime_timer.count() / (1000. * _total_num_line_steps), + _epsilon); + return Termination::SUCCESS; +} + +template +NonSymmetricIPM::NonSymmetricIPM(Matrix &A_, Vector &b_, Vector &c_, LHSCB *barrier_) : + A(A_), b(b_), c(c_), kappa(1.), tau(1.), _barrier(barrier_) { + y = Matrix::Zero(A.rows(), 1); + + //TODO: use proper tolerance / reference. + A_sparse = A.sparseView(IPMDouble(10e-10), 1e-10); + + _stored_x_centrality.resize(c.rows()); + _stored_s_centrality.resize(c.rows()); + + _custom_timers.resize(10); + + _logger = spdlog::get("NonSymmetricIPM"); + + + if (_logger == nullptr) { + std::vector sinks; + sinks.push_back(std::make_shared()); + sinks.push_back(std::make_shared("logs/logfile.txt")); + _logger = std::make_shared("NonSymmetricIPM", begin(sinks), end(sinks)); + _logger->set_level(spdlog::level::info); + + sinks.push_back(std::make_shared("logs/benchmark.txt")); + _benchmark_logger = std::make_shared("", begin(sinks) + 2, end(sinks)); + } + + _logger->info("A has dimensions {} x {}", A.rows(), A.cols()); + _logger->info("The number of non-zero entries is {}", A_sparse.nonZeros()); + +} + +template +void NonSymmetricIPM::initialize() { + + //Rescale instance for stability/conditioning (See Papp & Yildiz paper) + + x = _barrier->initialize_x(); + s = _barrier->initialize_s(); + + _err_consts.set(A,b,c); + + IPMDouble scaling_delta_primal = 0; + for (int i = 0; i < A.rows(); i++) { + IPMDouble const row_ratio = (1. + abs(b(i))) / (1. + abs(A.row(i).sum())); + scaling_delta_primal = std::max(scaling_delta_primal, row_ratio); + } + + IPMDouble scaling_delta_dual = 0; + for (int i = 0; i < s.rows(); i++) { + IPMDouble const entry_ratio = (1 + abs(s(i))) / (1 + abs(c(i))); + scaling_delta_dual = std::max(scaling_delta_dual, entry_ratio); + } + + IPMDouble const scaling_delta = sqrt(scaling_delta_dual * scaling_delta_primal); + + _logger->debug("Norm of x is {} and norm of s is {} before rescaling.", x.norm(), s.norm()); + + x = _barrier->initialize_x(scaling_delta); + s = _barrier->initialize_s(scaling_delta); + + _logger->debug("Rescaled initial point by {}", scaling_delta); + _logger->debug("Norm of c is {}, Norm of b is {} and norm of A is {} ", c.norm(), b.norm(), A.norm()); + + assert(x.rows() == _barrier->getNumVariables()); + + //TODO: switch to dense operation based on number of nonzeros. +// Matrix QR = A.transpose().householderQr().householderQ(); + + Eigen::SparseQR, Eigen::COLAMDOrdering > QR_sparse; + + Eigen::SparseMatrix tmp_sparse = A_sparse.transpose(); + tmp_sparse.makeCompressed(); + QR_sparse.compute(tmp_sparse); + +// _basis_ker_A = QR.block(0, A.rows(), QR.rows(), QR.cols() - A.rows()); + _basis_ker_A = Matrix(QR_sparse.matrixQ()).block(0, A.rows(), QR_sparse.rows(), QR_sparse.cols() - A.rows()); + + _logger->trace("Matrix A is: \n {}", A); + _logger->trace("Basis of ker A is: \n {}", _basis_ker_A); + _logger->trace("Check correctness: \n {}", A * _basis_ker_A); + + _num_corrector_steps = 3; + _large_neighborhood = .99; + _small_neighborhood = 0.1; + + _step_length_predictor = calc_step_length_predictor(); + _step_length_corrector = 1; + + + +} + +template +Vector NonSymmetricIPM::create_predictor_RHS() { + Vector v(y.rows() + x.rows() + 1 + s.rows() + 1); + v << y, x, tau * Matrix::Identity(1, 1), s, kappa * Matrix::Identity(1, 1); + DirectionDecomposition cur_sol(v, x.rows(), y.rows()); + Vector v1 = -(A * x - b * tau); + Vector v2 = -(-A.transpose() * y + c * tau - s); + Vector v3 = -(b.dot(y) - c.dot(x) - kappa) * Matrix::Identity(1, 1); + Vector rhs(v1.rows() + v2.rows() + v3.rows() + s.rows() + 1); + rhs << v1, v2, v3, -s, -kappa * Matrix::Identity(1, 1); + return rhs; +} + +template +Vector NonSymmetricIPM::create_corrector_RHS() { + Vector v1 = Vector::Zero(A.rows() + A.cols() + 1); + Vector v2 = -psi(mu()); + Vector rhs(v1.rows() + s.rows() + 1); + rhs << v1, v2; + _logger->trace("Corrector RHS is \n {}", rhs.transpose()); + return rhs; +}; + +template +IPMDouble NonSymmetricIPM::mu() { + return (x.dot(s) + (tau * kappa)) / (_barrier->concordance_parameter(x) + 1); +}; + +template +Vector NonSymmetricIPM::psi(IPMDouble t) { + Vector v(s.rows()); + v = s + t * _barrier->gradient(x); + Vector v_aux = (kappa - t / tau) * Matrix::Identity(1, 1); + Vector res(s.rows() + 1); + res << v, v_aux; + return res; +} + +template +Vector NonSymmetricIPM::solve_predictor_system() { + Vector rhs = create_predictor_RHS(); + SPDLOG_LOGGER_DEBUG(_logger, "System RHS for predictor step is {}", rhs.transpose()); + Vector andersen_direction = andersen_andersen_solve(rhs); + return andersen_direction; +} + +template +Vector NonSymmetricIPM::solve_corrector_system() { + Vector rhs = create_corrector_RHS(); + SPDLOG_LOGGER_DEBUG(_logger, "Psi is {}", psi(mu()).transpose()); + SPDLOG_LOGGER_DEBUG(_logger, "Barrier is {}", _barrier->gradient(x).transpose()); + SPDLOG_LOGGER_DEBUG(_logger, "s is {}", s.transpose()); + SPDLOG_LOGGER_DEBUG(_logger, "System RHS for corrector step is {}", rhs.transpose()); + SPDLOG_LOGGER_TRACE(_logger, "Corrector matrix is: \n{}", _M); + + //TODO: check if iterative refinement makes sense + auto andersen_dir = andersen_andersen_solve(rhs); + + return andersen_dir; +} + +template +void NonSymmetricIPM::print_status() { + + std::string format_ = "{:<25}:{:20.2}"; + + Double const step_length_predictor = static_cast(_step_length_predictor); + Double const step_length_corrector = static_cast(_step_length_corrector); + + _logger->debug(format_, "alpha_predictor", step_length_predictor); + _logger->debug(format_, "alpha_corrector", step_length_corrector); + + Matrix aux(2, x.rows()); + aux.block(0, 0, 1, x.rows()) = x.transpose(); + aux.block(1, 0, 1, s.rows()) = s.transpose(); + + + if (_logger->level() <= spdlog::level::trace) { + _logger->trace("Current primal/dual x, s pair :\n{}", aux); + _logger->trace("Current primal/dual rescaled x, s pair :\n{}", aux / tau); + } + + _logger->info(format_, "kappa", static_cast(kappa)); + _logger->info(format_, "tau", static_cast(tau)); + + IPMDouble mu_ipm_scaled = mu() / (tau * tau); + Double const mu_scaled = static_cast(mu_ipm_scaled); + _logger->debug(format_, "mu scaled", mu_scaled); + + IPMDouble mu_ipm = mu(); + Double const mu_ = static_cast(mu_ipm); + _logger->info(format_, "mu", boost::numeric_cast(mu())); + + if (_logger->level() <= spdlog::level::debug) { + IPMDouble centrality_ipm_ = centrality(); + Double const centrality_ = static_cast(centrality_ipm_); + _logger->debug(format_, "centrality error", centrality_); + } + + IPMDouble duality_gap_ipm_ = kappa / tau; + Double duality_gap_ = static_cast(duality_gap_ipm_); + _logger->info(format_, "duality gap", duality_gap_); + + IPMDouble primal_inf_ipm_ = primal_error(); + Double primal_inf_ = static_cast(primal_inf_ipm_); + _logger->info(format_, "primal infeas. ", primal_inf_); + + IPMDouble primal_inf_unscaled_ipm_ = primal_error_rescaled(); + Double primal_inf_unscaled_ = static_cast(primal_inf_unscaled_ipm_); + _logger->debug(format_, "primal infeas. unscaled", primal_inf_unscaled_); + + IPMDouble dual_inf_ipm_ = dual_error(); + Double dual_inf_ = static_cast(dual_inf_ipm_); + _logger->info(format_, "dual infeas.", dual_inf_); + + IPMDouble dual_inf_unscaled_ipm_ = dual_error_rescaled(); + Double dual_inf_unscaled_ = static_cast(dual_inf_unscaled_ipm_); + _logger->debug(format_, "dual infeas. unscaled", dual_inf_unscaled_); + + _logger->trace("last predictor direction: {}", _last_predictor_direction.transpose()); + + _logger->info(format_, "Predictor time (s)", _predictor_timer.count() / 1000.); + _logger->info(format_, "Corrector time (s)", _corrector_timer.count() / 1000.); + + _logger->info(format_, "Total andersen time (s)", + _andersen_sys_timer.count() / 1000.); + _logger->info(format_, "Total runtime (s)", + _total_runtime_timer.count() / 1000.); + _logger->info(format_, "Calc centrality time (s)", + _centrality_timer.count() / 1000.); + _logger->info(format_, "Time checking interior(s)", + _barrier->_in_interior_timer.template count() / 1000.); + + + _logger->info(format_, "Time per step: ", + _total_runtime_timer.count() / (1000. * _total_num_line_steps)); + if (_logger->level() <= spdlog::level::debug) { + for (unsigned idx = 0; idx < _custom_timers.size(); idx++) { + std::string s = "Custom timer " + std::to_string(idx); + _logger->info(format_, s, + _custom_timers[idx].template count() / 1000.); + } + ProductBarrier *productBarrier = static_cast *>(_barrier); + SumBarrier *sumBarrier = static_cast *>(productBarrier->get_barriers()[0]); + InterpolantDualSOSBarrier *interpBarrier = static_cast *>(sumBarrier->get_barriers()[0]); + _logger->info("Runtimes for updating gradient/hessian/LLT"); + for (unsigned idx = 0; idx < interpBarrier->_custom_timers.size(); idx++) { + std::string s = "Custom timer " + std::to_string(idx); + _logger->info(format_, s, + interpBarrier->_custom_timers[idx].template count() / 1000.); + } + } + _logger->info("--------------------------------------------------------------------------------------"); + +} + +template +bool NonSymmetricIPM::terminate_successfully_wrapper() { +// Eigen::internal::set_is_malloc_allowed(true); + bool result = terminate_successfully(); +// Eigen::internal::set_is_malloc_allowed(false); + return result; +} + +template +bool NonSymmetricIPM::terminate_successfully() { + if (primal_error() > _epsilon) { + return false; + } + + if (dual_error() > _epsilon) { + return false; + } + + if (complementarity() > _epsilon){ + return false; + } + + return true; +} + +// Termination criteria taken from Skajaa - Ye "A Homogeneous Interior-Point Algorithm for +// Nonsymmetric Convex Conic Optimization" https://web.stanford.edu/~yyye/nonsymmhsdimp.pdf page 15. + +template +bool NonSymmetricIPM::terminate_infeasible_wrapper() { + +// Eigen::internal::set_is_malloc_allowed(true); + bool result = terminate_infeasible(); +// Eigen::internal::set_is_malloc_allowed(false); + return result; +} + +template +bool NonSymmetricIPM::terminate_infeasible() { + + //TODO: Figure out if initialization scaling (delta) should influence the termination criteria. + + //Primal feasibility + IPMDouble const ipm_1 = IPMDouble(1.); + IPMDouble const primal_error = (A * x - b * tau).template lpNorm(); + IPMDouble const A_norm = A.template lpNorm(); + IPMDouble const b_norm = b.template lpNorm(); + if (primal_error > _epsilon * std::max(ipm_1, A_norm + b_norm)) { + return false; + } + + IPMDouble const dual_error = (A.transpose() * y + s - c * tau).template lpNorm(); + IPMDouble const c_norm = c.template lpNorm(); + //Dual feasibility + if (dual_error > _epsilon * std::max(ipm_1, A_norm + c_norm)) { + return false; + } + + //Duality gap + if (abs(-c.dot(x) + b.dot(y) - kappa) + > + _epsilon * std::max(ipm_1, static_cast(c_norm + b_norm))) { + return false; + }; + + //tiny tau + if (tau > _epsilon * 10e-2 * std::max(ipm_1, kappa)) { + return false; + } + return true; + +} + +template +bool NonSymmetricIPM::terminate() { + return terminate_successfully_wrapper() or terminate_infeasible_wrapper(); +} + + +template +IPMDouble NonSymmetricIPM::centrality() { + + _centrality_timer.start(); + if (_stored_x_centrality == x and _stored_s_centrality == s) { + return _stored_centrality_error; + } + IPMDouble const mu_d = mu(); + _custom_timers[0].start(); + Vector const psi_vec = psi(mu_d); + + _logger->trace("Vector Psi is: {}", psi_vec.transpose()); + + _custom_timers[0].stop(); + _custom_timers[2].start(); + IPMDouble tau_kappa_entry = tau * psi_vec.segment(psi_vec.rows() - 1, 1).sum(); + + Vector LLT_sol = _barrier->llt_L_solve(x, psi_vec.segment(0, psi_vec.rows() - 1)); + Vector err_L(psi_vec.rows()); + err_L << LLT_sol, tau_kappa_entry; + + if (_logger->level() <= spdlog::level::debug) { + ProductBarrier *pb = static_cast * >(_barrier); + auto &segs = pb->get_segments(); + Vector seg_norms(segs.size() + 1); + for (int i = 0; i < segs.size(); i++) { + seg_norms(i) = err_L.segment(segs[i].first, segs[i].second - segs[i].first).norm(); + } + seg_norms(segs.size()) = tau_kappa_entry; + seg_norms /= mu_d; + _logger->info("Segments error norms are: {}", seg_norms.transpose()); + } + + if (_logger->level() <= spdlog::level::trace) { + _logger->trace("Linear system error vector is {}", err_L.transpose()); + } + _custom_timers[2].stop(); + + IPMDouble centr_err_L = err_L.norm() / mu_d; + + _stored_x_centrality = x; + _stored_s_centrality = s; + _stored_centrality_error = centr_err_L; + _centrality_timer.stop(); + return centr_err_L; +} + +template +void NonSymmetricIPM::test_hessian() { + for (int i = 0; i < x.rows(); i++) { + Vector dir = Vector::Zero(x.rows()); + dir(i) = 10e-4; + _logger->debug("Test hessian at: {}", x.transpose()); + _logger->debug("with update {}", dir.transpose()); + Matrix M(2, x.rows()); + M.block(0, 0, 1, x.rows()) = _barrier->gradient(x + dir).transpose() + - _barrier->gradient(x).transpose(); + M.block(1, 0, 1, x.rows()) = (_barrier->hessian(x) * dir).transpose(); + _logger->debug("Comparison of gradient and hessian for {}: \n{}", dir.transpose(), M); + } + return; +}; + diff --git a/src/volesti/include/sos/README.md b/src/volesti/include/sos/README.md new file mode 100644 index 00000000..6c9eef4c --- /dev/null +++ b/src/volesti/include/sos/README.md @@ -0,0 +1,104 @@ +## Sum of Squares optimization + +This subproject implements the algorithm(s) in [1,2,3]. +Supplementary material describing the mathematical theory behind the code can be found [here](http://personal.lse.ac.uk/natura/gsoc2020/supplementary.pdf) + +#### Usage the SOS-solver for Polynomial Envelope problems + +For precise computation of Chebyshev Points and Lagrange Polynomials boost::multiprecision is used. Boost is also +used for the Property Tree in instances and configuration as well as for typecasting templated classes. The boost headers in this project do not contain the needed header files. Please provide the link to the boost files (version 1.67 or higher) via thet `-DBOOST_DIR` flag. Also [spdlog](https://github.com/gabime/spdlog) is used. Please provide link to installed package via `-DSPDLOG_DIR` flag. (The link also provides manuals for installation with various package managers.) +Navigate to the SOS envelope example and and compile: + +``` +cd examples/EnvelopeProblemSOS +cmake -DCMAKE_BUILD_TYPE=Release_double -DBOOST_DIR=your_boost_include_directory -DSPDLOG_DIR=your_spdlog_include_dir . +make +``` + +Run + +``` +./EnvelopeProblemSOS +``` + +See an example plot below. +![image](plot_saved.png "Example envelope") + +To parse a custom file invoke with added file argument: + +``` +./NonSymmetricConicOptimization file.json +``` + +where `file.json` has format + +``` json +{ + "max_degree": 30, + "num_variables": 1, + "polynomials": [ + [1,-1, 3, -4, 7], + [0.5,2, 8, -3, 5] + ] +} + +``` + +Each array in "polynomials" stands for a polynomial. The entries of its array a are the coefficients of the first length(a) coefficients in standard monomial basis or Chebyshev basis. The basis choice can be adjusted in the configuration json file. + + +#### Implemented + +* The generic implementation of the algorithm in [1] with barrier methods for the following cones: + * Non-negative orthant + * SDP cone + * Dual of SOS cone with following bases: + * monomial + * interpolant +* Generation of Lagrange polynomials for Chebyshev points of the second kind +* Tool that approximates the polynomial lower envelope of any (stable to degree approx. 100) set of univariate polynomials +on the interval [-1,1]. +* Visualisation via Matplotlib plot. +* Support for several floating-point precision data types for the IPM +(Primitive Floating Point and boost:multiprecision floats) +* Added multivariate support +* Added multithreading. +* Added LAPACK and BLAS support +* Added MKL Support (system-dependent configuration necessary) +* Dynamically switch between float -> double -> long double -> multiprecision, when Matrices become ill-conditioned. + Note that this feature currenlty only works for the InterpolantDualSOSBarrier, SumBarrier and ProductBarrier. +* Dynamic step-length for predictor step and size of large neighborhood. +* Input and configuration files in JSON format + +#### Roadmap + +* Add switch to use sparse computation (via counting nonzeros in A or other existing funcionality) +* Parameter tuning +* Choose Method for QR Decompositions dynamically. Benchmarks can be found [here](https://eigen.tuxfamily.org/dox/group__DenseDecompositionBenchmark.html). +* Use Eigen funcionality to avoid dynamical heap memory is allocation. +* Implement Lagrange polynomials more efficiently (or find C++ library with sufficient precision.) +* Benchmarking with alfonso, SOSTOOLS, MOSEK (SDP and the new Nonsymmetric cone solver), SeDuMi +* Implementation/inspiration from [4]. In particular interesting are: + * Positive definite rescaling matrix for added stability and monotonicity + * Combining Predictor- and Corrector steps. +* Higher order corrector steps as in [1] +* Write more tests +* Test different central path neighborhoods, e.g. measuring \psi in \infty norm. +* Interfaces for R (and Python) +* Store the gradient and Hessian of the previous iterate (for failing next step in predictor direction) + * Even better: For all the line steps, compute the new predictor/corrector direction. This is cheap, + as we already have gradient and hessian, but might drastically improve performance. +* Fix SDP solver. It returns feasible solutions but no optimal solutions yet. + + +#### References + +[1] A. Skajaa and Y. Ye, [A homogeneous interior-point algorithm for nonsymmetric convex conic optimization](https://web.stanford.edu/~yyye/nonsymmhsdimp.pdf), Mathematical Programming Ser. A, 150 (2015), pp. 391-422. + +[2] D. Papp and S. Yildiz, On “A homogeneous interior-point algorithm for nonsymmetric convex conic optimization”. https://arxiv.org/abs/1712.00492 + +[3] D. Papp and S. Yildiz, [Sum-of-squares optimization without semidefinite programming](https://arxiv.org/abs/1712.01792). SIAM Journal on Optimization 29(1), 2019, pp. 822-851. + +[4] R. Badenbroek and J. Dahl, [An Algorithm for Nonsymmetric Conic Optimization Inspired by MOSEK](https://arxiv.org/pdf/2003.01546.pdf). https://arxiv.org/pdf/2003.01546.pdf + + diff --git a/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.h b/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.h new file mode 100644 index 00000000..d4ff9b7d --- /dev/null +++ b/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.h @@ -0,0 +1,49 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file +#ifndef SOS_DUALSOSCONESTANDARDBARRIER_H +#define SOS_DUALSOSCONESTANDARDBARRIER_H + +#include "LHSCB.h" + +//Implementation for Standard Monomial Basis for the Dual SOS cone. +template +class DualSOSConeStandardBarrier : public LHSCB { + + using LHSCB = LHSCB; + + typedef Vector Vector; + typedef Matrix Matrix; + +public: + DualSOSConeStandardBarrier() : LHSCB() {}; + + DualSOSConeStandardBarrier(unsigned max_polynomial_degree_) : _max_polynomial_degree(max_polynomial_degree_) { + }; + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + //Matrix inverse_hessian(Vector x) override; + bool in_interior(Vector x) override; + + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + Vector initialize_s() override; + + //Lambda is the linear operator used in Proposition 1.1 in Papp & Yildiz "SOS + //without semidefinite programming. In particular, for the monomial basis it takes + //the form listed in section 3.1 of the same paper. + Matrix Lambda(Vector x); + +private: + unsigned _max_polynomial_degree; +}; + +#include "DualSOSConeStandardBarrier.hpp" + +#endif //SOS_DUALSOSCONESTANDARDBARRIER_H diff --git a/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.hpp b/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.hpp new file mode 100644 index 00000000..7e4f5106 --- /dev/null +++ b/src/volesti/include/sos/barriers/DualSOSConeStandardBarrier.hpp @@ -0,0 +1,81 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include "DualSOSConeStandardBarrier.h" + + +template +Vector DualSOSConeStandardBarrier::gradient(Vector x) { + assert(in_interior(x)); + Matrix X = Lambda(x); + Matrix Z = X.inverse(); + Vector g(x.rows()); + for (int i = 0; i < g.rows(); ++i) { + Matrix E_i = Matrix::Zero(X.rows(), X.cols()); + for (int j = 0; j <= i; ++j) { + E_i(j, i - j) = 1; + } + g(i) = -Z.cwiseProduct(E_i).sum(); + } + return g; +} + +template +Matrix DualSOSConeStandardBarrier::hessian(Vector x) { + assert(in_interior(x)); + Matrix X = Lambda(x); + Matrix Z = X.inverse(); + Matrix H = Matrix::Zero(Z.rows(), Z.cols()); + for (int u = 0; u < H.rows(); ++u) { + for (int v = 0; v < H.cols(); ++v) { + IPMDouble H_uv = 0; + for (int a = 0; a <= u; ++a) { + for (int k = 0; k <= v; ++k) { + H_uv += Z(a, u - a) + Z(k, v - k); + } + } + } + } + return H; +} + +template +bool DualSOSConeStandardBarrier::in_interior(Vector x) { + Matrix X = Lambda(x); + CustomLLT llt_check; + llt_check.compute(X); + return llt_check.info() != Eigen::NumericalIssue; +} + +template +IPMDouble DualSOSConeStandardBarrier::concordance_parameter(Vector x) { + return x.rows(); +} + +template +Vector DualSOSConeStandardBarrier::initialize_x() { + //TODO: find centered initialization + return Vector(); +} + +template +Vector DualSOSConeStandardBarrier::initialize_s() { + //TODO: find centered initialization + return Vector(); +} + +template +Matrix DualSOSConeStandardBarrier::Lambda(Vector x) { + assert(x.rows() == _max_polynomial_degree + 1); + Matrix M(_max_polynomial_degree + 1, _max_polynomial_degree + 1); + for (unsigned i = 0; i < _max_polynomial_degree + 1; ++i) { + for (unsigned j = 0; j < _max_polynomial_degree + 1; ++j) { + M(i, j) = x(i + j); + } + } + return M; +} + diff --git a/src/volesti/include/sos/barriers/FullSpaceBarrier.h b/src/volesti/include/sos/barriers/FullSpaceBarrier.h new file mode 100644 index 00000000..c8221a88 --- /dev/null +++ b/src/volesti/include/sos/barriers/FullSpaceBarrier.h @@ -0,0 +1,46 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_FULLSPACEBARRIER_H +#define SOS_FULLSPACEBARRIER_H + +#include "LHSCB.h" + +template +class FullSpaceBarrier final : public LHSCB{ + + using LHSCB = LHSCB; + + typedef Vector Vector; + typedef Matrix Matrix; + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + Matrix inverse_hessian(Vector x) override; + + bool in_interior(Vector x) override; + + Matrix llt_solve(Vector x, const Matrix &rhs) override; + Vector llt_L_solve(Vector x, Vector rhs) override; + + + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + + Vector initialize_s() override; + +public: + FullSpaceBarrier(unsigned num_variables_) { + this->_num_variables = num_variables_; + } +}; + +#include "FullSpaceBarrier.hpp" + +#endif //SOS_FULLSPACEBARRIER_H diff --git a/src/volesti/include/sos/barriers/FullSpaceBarrier.hpp b/src/volesti/include/sos/barriers/FullSpaceBarrier.hpp new file mode 100644 index 00000000..3e0f29cb --- /dev/null +++ b/src/volesti/include/sos/barriers/FullSpaceBarrier.hpp @@ -0,0 +1,66 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include "FullSpaceBarrier.h" + +//This barrier function exists for convenience. +//An instance should be reformulated instead of using the barrier. + +template +Vector FullSpaceBarrier::gradient(Vector) { + return Vector::Zero(this->_num_variables); +} + +template +Matrix FullSpaceBarrier::hessian(Vector) { + return Matrix::Zero(this->_num_variables, this->_num_variables); +} + +template +Matrix FullSpaceBarrier::inverse_hessian(Vector) { + return Matrix::Zero(this->_num_variables, this->_num_variables); +} + +template +bool FullSpaceBarrier::in_interior(Vector) { + return true; +} + +template +Matrix FullSpaceBarrier::llt_solve(Vector x, const Matrix &rhs) { + if(rhs.norm() > 1e-10){ + spdlog::warn("Exit because RHS of matrix is"); + std::cout << rhs << std::endl; + assert(false); + } + return Matrix::Zero(rhs.rows(),rhs.cols()); +} + +template +Vector FullSpaceBarrier::llt_L_solve(Vector x, Vector rhs) { + if(rhs.norm() > 1e-10){ + spdlog::warn("Exit because RHS of matrix is"); + std::cout << rhs << std::endl; + assert(false); + } + return Vector::Zero(x.rows()); +} + + +template +IPMDouble FullSpaceBarrier::concordance_parameter(Vector) { + return 0; +} + +template +Vector FullSpaceBarrier::initialize_x() { + return Vector::Zero(this->_num_variables); +} + +template +Vector FullSpaceBarrier::initialize_s() { + return Vector::Zero(this->_num_variables); +} diff --git a/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.h b/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.h new file mode 100644 index 00000000..76781e52 --- /dev/null +++ b/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.h @@ -0,0 +1,113 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_INTERPOLANTDUALSOSBARRIER_H +#define SOS_INTERPOLANTDUALSOSBARRIER_H + +#include "LHSCB.h" + +template +class InterpolantDualSOSBarrier : public LHSCB { + + using LHSCB = LHSCB; + + typedef Vector Vector; + typedef Matrix Matrix; + +public: + InterpolantDualSOSBarrier() : LHSCB() {}; + + InterpolantDualSOSBarrier(unsigned max_polynomial_degree_, unsigned num_variables_ = 1) : + InterpolantDualSOSBarrier(max_polynomial_degree_, Vector::Ones(1), num_variables_) {}; + + InterpolantDualSOSBarrier(unsigned max_polynomial_degree_, Vector poly_g, unsigned num_variable_symbols_ = 1); + + template + InterpolantDualSOSBarrier * cast(){ + InterpolantDualSOSBarrier * int_bar = new InterpolantDualSOSBarrier(); + int_bar->_custom_timers.resize(10); + int_bar->_max_polynomial_degree = _max_polynomial_degree; + int_bar->_num_variable_symbols = _num_variable_symbols; + int_bar->_unisolvent_basis = _unisolvent_basis; + int_bar->_intermediate_matrix = _intermediate_matrix.template cast(); + int_bar->_preintermediate_matrix = _preintermediate_matrix.template cast(); +// int_bar->_intermediate_LLT = _intermediate_LLT.template cast(); + int_bar->_Q = _Q.template cast(); + int_bar->_V = _V.template cast(); + int_bar->_L = _L; + int_bar->_U = _U; + int_bar->_g = _g.template cast(); + int_bar->_g_g_transpose = _g_g_transpose.template cast(); + int_bar->_P = _P.template cast(); + int_bar->use_low_rank_updates = use_low_rank_updates; + + return int_bar; + }; + + bool update_gradient_hessian_LLT(Vector x, bool check_interior_only = false); + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + Eigen::LLT llt(Vector x, bool symmetrize = 0) override; + + Matrix inverse_hessian(Vector x) override; + + bool in_interior(Vector x) override; + + //TODO: better solution for implementation concordance parameter; + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + + Vector initialize_s() override; + + std::vector > &get_basis() { + return _unisolvent_basis; + } + + Matrix get_P() { + return _P; + } + + void configure(pt::ptree & config); + + unsigned _max_polynomial_degree; + unsigned _num_variable_symbols; + //Turn _unisolvent_basis into Matrix + std::vector > _unisolvent_basis; + Matrix _intermediate_matrix; + Matrix _preintermediate_matrix; + CustomLLT _intermediate_LLT; + Matrix _Q; + Matrix _V; + unsigned _L, _U; + + //Weighted polynomials + Vector _g; + Matrix _g_g_transpose; + Matrix _P; + + //Usage of Woodburry matrix identity and simple heuristic for choice of variables + //to update. + bool use_low_rank_updates = true; + +private: + + + void compute_V_transpose_V(); + + void construct_univariate(Vector poly_g); + void construct_bivariate(Vector poly_g); + void construct_multivariate(Vector poly_g); + + +}; + +#include "InterpolantDualSOSBarrier.hpp" + +#endif //SOS_INTERPOLANTDUALSOSBARRIER_H diff --git a/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.hpp b/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.hpp new file mode 100644 index 00000000..c08046de --- /dev/null +++ b/src/volesti/include/sos/barriers/InterpolantDualSOSBarrier.hpp @@ -0,0 +1,493 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include "InterpolantDualSOSBarrier.h" +#include +#include "Padua/padua.h" + +template +InterpolantDualSOSBarrier::InterpolantDualSOSBarrier( + unsigned max_polynomial_degree_, Vector poly_g,unsigned num_variable_symbols_) + : _max_polynomial_degree(max_polynomial_degree_), _num_variable_symbols(num_variable_symbols_) { + + //TODO: Check if still true for multivariate case. + assert(poly_g.rows() <= max_polynomial_degree_ + 1); + + this->_custom_timers.resize(10); + + //poly_g.rows() is degree + 1 of the polynomial g; + //TODO: setting _L using the number of rows only works for univariate polynomials + // + + //For now no weights for multivariate case. + assert(_num_variable_symbols == 1 or poly_g == Vector::Ones(poly_g.rows())); + + //TODO: check if type cast is safe. + _L = static_cast(boost::math::binomial_coefficient( + _num_variable_symbols + _max_polynomial_degree + 1 - (unsigned) poly_g.rows(), _num_variable_symbols)); + _U = static_cast(boost::math::binomial_coefficient( + 2 * _max_polynomial_degree + _num_variable_symbols, _num_variable_symbols)); + + _preintermediate_matrix = Matrix(_U, _L); + _intermediate_matrix = Matrix(_L, _L); + _intermediate_LLT = CustomLLT(_L); + _V = Matrix(_L, _U); + _Q = Matrix(_V.cols(), _V.cols()); + + this->_num_variables = _U; + _unisolvent_basis.resize(_U); + + if (_num_variable_symbols == 1) { + construct_univariate(poly_g); + } else if (_num_variable_symbols == 2) { + construct_bivariate(poly_g); + } else { + construct_multivariate(poly_g); + } + +}; + +template +void InterpolantDualSOSBarrier::construct_univariate(Vector poly_g) { + for (unsigned i = 0; i < _unisolvent_basis.size(); ++i) { + BoostDouble cos_i = boost::multiprecision::cos(i * boost::math::constants::pi() / (_U - 1)); + InterpolantDouble dummy_ipm; + InterpolantDouble cos_val = static_cast(cos_i); + _unisolvent_basis[i].push_back(cos_val); + } + + //TODO: Figure out how choice of P could influence condition / stability of maps. + + //Use monomial standard basis to orthogonalize + + //_P is used in the Interior Point Method. Therefore we need to convert the multi-precision + // floating-point into the IPM floating point precision + + //TODO: This interpolant Matrix only needs to be found once and can then be reused; + + //Alternative approach of finding _P via Chebyshev basis + + //TODO:Make this library more precise + + //Computing _g should be done with transformation matrix. + _g = Vector::Zero(_U); + for (int p = 0; p < _U; ++p) { + _g(p) = poly_g(0); + for (int i = 1; i < poly_g.rows(); i++) { + _g(p) += poly_g(i) * pow(_unisolvent_basis[p][0], i).template convert_to(); + } + } + _g_g_transpose = _g * _g.transpose(); + + //TODO: Option to compute cheb_P via InterpolantDouble; + + //TODO: Figure out whether orthogonalization could be done in double precision to speed up initialisation. + spdlog::info("Construct orthogonal interpolant point Matrix P..."); + cxxtimer::Timer orth_timer; + orth_timer.start(); + _P = orthogonal_P_Matrix_library.get(_L,_U); + orth_timer.stop(); + std::cout << "Orthogonalization done in " << orth_timer.count() / 1000. + << " seconds." << std::endl; +} + +//Untested +template +void InterpolantDualSOSBarrier::construct_bivariate(Vector poly_g) { + + //For now no weighted polynomials + assert(poly_g == Vector::Ones(1)); + unsigned const corrected_d = 2 * _max_polynomial_degree + 1 - poly_g.size(); + unsigned const corrected_d_plus_1 = corrected_d + 1; + + double *pd_pts = padua::padua_points(_max_polynomial_degree + 1); + + for (int i = 0; i < _U; i++) { + _unisolvent_basis[i] = {pd_pts[2 * i], pd_pts[2 * i + 1]}; + } + + //Set weight vector _g; + //TODO: do properly for weighted case. + _g = Vector::Ones(_U); + _g_g_transpose = _g * _g.transpose(); + + std::cout << "Construct Matrix P" << std::endl; + + _P.resize(_U, _L); + + //TODO: Make following loops more efficient OR find mathematical theory that simplifies expressions. + unsigned col_idx = 0; + for (int i = 0; i <= _max_polynomial_degree; i++) { + for (int j = 0; j + i <= _max_polynomial_degree; j++) { + Eigen::VectorXd vec_i = Eigen::VectorXd::Zero(i + 1); + vec_i(i) = 1; + Eigen::VectorXd vec_j = Eigen::VectorXd::Zero(j + 1); + vec_j(j) = 1; + ChebTools::ChebyshevExpansion cheb_i(vec_i); + ChebTools::ChebyshevExpansion cheb_j(vec_j); + for (int k = 0; k < _U; k++) { + double first_eval = cheb_i.y_recurrence(static_cast(_unisolvent_basis[k][0])); + double second_eval = cheb_j.y_recurrence(static_cast(_unisolvent_basis[k][1])); + _P(k, col_idx) = first_eval * second_eval; + } + col_idx++; + } + } + + std::cout << "P before orthogonolisation: " << std::endl << _P << std::endl; + + Matrix P_ortho = _P.householderQr().householderQ(); + P_ortho.colwise().hnormalized(); + _P = P_ortho.block(0, 0, _U, _L); + //end +} + +//TODO: test +template +void InterpolantDualSOSBarrier::construct_multivariate(Vector poly_g) { + assert(poly_g == Vector::Ones(1)); + + //Set weight vector _g; + //TODO: do properly for weighted case. + _g = Vector::Ones(_U); + _g_g_transpose = _g * _g.transpose(); + + _P.resize(_U, _L); + + //just for testing. + + int num_candidates = 1; + for (int i = 2 * _max_polynomial_degree + 1 + 1; + i <= 2 * _max_polynomial_degree + _num_variable_symbols + 1; i++) { + num_candidates *= i; + } + + //generate Fekete candidates + + + std::vector > candidates; + std::vector comb_bound; + for (int i = 2 * _max_polynomial_degree + 1; i <= 2 * _max_polynomial_degree + _num_variable_symbols; i++) { + comb_bound.push_back(i); + } + AllCombinationTuple comb_tuple(comb_bound); + + do { + std::vector &cheb_v = comb_tuple.get_combination(); + std::vector cand; + for (int i = 0; i < cheb_v.size(); i++) { + cand.push_back(cos((double) cheb_v[i] * boost::math::constants::pi() / (double) comb_bound[i])); + } + candidates.push_back(cand); + } while (comb_tuple.next()); + + for (auto cand : candidates) { + for (auto c : cand) { + std::cout << c << " "; + } + std::cout << std::endl; + } + + assert(num_candidates == candidates.size()); + + //generate Fekete polynomials + + Matrix candidate_matrix(_U, candidates.size()); + + DegreeTuple dt(_num_variable_symbols, 2 * _max_polynomial_degree); + unsigned tup_idx = 0; + do { + //Compute chebyshev polynomial evaluation + std::vector &tup = dt.get_tuple(); + for (int i = 0; i < candidates.size(); i++) { + double cheb_eval = 1.; + for (int j = 0; j < candidates[i].size(); j++) { + Eigen::VectorXd vec_j = Eigen::VectorXd::Zero(tup[j] + 1); + vec_j(tup[j]) = 1; + ChebTools::ChebyshevExpansion cheb_j(vec_j); + double eval_j = cheb_j.y_recurrence(static_cast(candidates[i][j])); + cheb_eval *= eval_j; + } + candidate_matrix(tup_idx, i) = cheb_eval; + } + tup_idx++; + } while (dt.next_valid()); + + + assert(tup_idx == _U); + +// std::cout << "Candidate matrix is \n"; +// std::cout << candidate_matrix << std::endl; + + + Matrix col_permutated_cand_matrix = candidate_matrix * candidate_matrix.colPivHouseholderQr().colsPermutation(); +// std::cout << "And after permutation \n"; +// std::cout << col_permutated_cand_matrix << std::endl; + _P = col_permutated_cand_matrix.block(0, 0, _U, _U).transpose(); +} + +//For profiling purposes +template +void InterpolantDualSOSBarrier::compute_V_transpose_V() { + _Q.noalias() = _V.transpose() * _V; +} + +template +void InterpolantDualSOSBarrier::configure(pt::ptree & config){ + if(config.find("use_low_rank_updates") != config.not_found()){ + use_low_rank_updates = config.get("use_low_rank_updates"); + } +} + +template +bool InterpolantDualSOSBarrier::update_gradient_hessian_LLT(Vector x, bool check_interior_only) { + + Matrix Q; + CustomLLT LLT; + IPMDouble stored_gx_norm; + Vector new_stored_x; + +// if(!this->_stored_gradients.empty()){ +// //This is to test whether lower ran updates make sense +// +// Vector stored_x = this->_stored_gradients[0].first; +// stored_gx_norm = _g.cwiseProduct(stored_x).norm(); +// Matrix aux(2, x.rows()); +// IPMDouble relative_norm = stored_x.norm()/x.norm(); +//// std::cout << "relative norm is " << relative_norm << std::endl; +// aux << stored_x.transpose(), x.transpose() * relative_norm; +//// std::cout << "aux is\n" << aux << std::endl; +// Vector rel = stored_x.cwiseProduct(x.cwiseInverse()); +// std::vector relative_vec(rel.data(), rel.data() + rel.rows()); +// sort(relative_vec.begin(), relative_vec.end()); +// IPMDouble mean = relative_vec[relative_vec.size() / 2]; +// +// std::cout << "Relative mean vector is: \n"; +// +// unsigned small_variation_count_05 = 0; +// unsigned small_variation_count_01 = 0; +// for(IPMDouble i : relative_vec){ +// i/=mean; +// std::cout << i << ", "; +// if(abs(i-1) < .05){ +// small_variation_count_05++; +// } +// if(abs(i-1) < .01){ +// small_variation_count_01++; +// } +// } +// std::cout << std::endl; +// std::cout << "small variation ratio: " << double(small_variation_count_01) / relative_vec.size() << std::endl; +// std::cout << "small variation ratio: " << double(small_variation_count_05) / relative_vec.size() << std::endl; +// } + + bool do_exact_computation = true; + + if (use_low_rank_updates and not this->_stored_gradients.empty()) { + do_exact_computation = false; + this->_custom_timers[9].start(); + Vector stored_x = this->_stored_gradients[0].first; + stored_gx_norm = _g.cwiseProduct(stored_x).norm(); + Matrix aux(2, x.rows()); + aux << stored_x.transpose(), x.transpose(); + std::cout << "aux is\n" << aux << std::endl; + Vector stored_scaled_gx = _g.cwiseProduct(stored_x).normalized(); + //TODO: Find best scaling to minimize number of adjusted variables. + Vector scaled_gx = _g.cwiseProduct(x).normalized(); + Vector relative = scaled_gx - stored_scaled_gx; + Vector relative_abs = relative.cwiseAbs(); +// TODO: Check if adding diagonal here makes sense. This corresponds to the gradient and the term occurs for the Sherman-Morrison update. + + std::vector relative_vec(relative_abs.data(), relative_abs.data() + relative_abs.rows()); + std::vector relative_vec_sorted = sort_indexes(relative_vec); + std::reverse(relative_vec_sorted.begin(), relative_vec_sorted.end()); + + Q = _Q; + LLT = _intermediate_LLT; + assert(_U == relative_vec_sorted.size()); + + new_stored_x = this->_stored_gradients[0].first; + for (int i = 0; i < _U; i++) { + + //rank one updates + + unsigned idx = relative_vec_sorted[i]; + + //TODO: find good threshold + if (relative_abs(idx) < 1e-10 * relative_abs(relative_vec_sorted[0])) { + std::cout << "break after " << i << " of " << _U << " updates" + << std::endl; + break; + } + + //Q update + new_stored_x(idx) = x(idx); + + this->_custom_timers[7].start(); + Vector P_idx = _P.row(idx).transpose(); + Vector tmp = LLT.solve(P_idx); + Vector rank_one_factor = _P * tmp; + Matrix Q_update = + -rank_one_factor * rank_one_factor.transpose() / + (1 / (stored_gx_norm * relative(idx)) + tmp.dot(P_idx)); +// std::cout << "Q_update normed is \n" << Q_update << std::endl; + + Q += Q_update; + this->_custom_timers[7].stop(); + + //L update + this->_custom_timers[8].start(); + LLT.rankUpdate(P_idx, stored_gx_norm * relative(idx)); + if (LLT.info() == Eigen::NumericalIssue) { + //This means we should either change the order in which we apply the rank-1 updates + //or revert back to exact computation. Currently we just jump to exact computation. + + do_exact_computation = true; + break; + } + this->_custom_timers[8].stop(); + } + this->_custom_timers[9].stop(); + + if (not do_exact_computation) { +// std::cout << "Old matrix LLT \n" << LLT.matrixL().toDenseMatrix() << std::endl; + _intermediate_LLT.copy_and_scale(LLT, sqrt(_g.cwiseProduct(x).norm() / stored_gx_norm)); +// std::cout << "Newly copied matrix \n" +// << _intermediate_LLT.matrixL().toDenseMatrix() / sqrt(_g.cwiseProduct(x).norm() / stored_gx_norm) << std::endl; + _Q = Q * stored_gx_norm / _g.cwiseProduct(x).norm(); + } + } + + if (do_exact_computation) { + new_stored_x = x; + + if(this->_stored_intermediate_LLT.empty()){ + this->_stored_intermediate_LLT.resize(1); + this->_stored_intermediate_LLT[0].first = Vector::Zero(new_stored_x.rows()); + } + + if(this->_stored_intermediate_LLT[0].first == new_stored_x){ + _intermediate_LLT = this->_stored_intermediate_LLT[0].second; + } else { + + this->_custom_timers[1].start(); + _preintermediate_matrix.noalias() = _g.cwiseProduct(x).asDiagonal() * _P; + _intermediate_matrix.noalias() = _P.transpose() * _preintermediate_matrix; + this->_custom_timers[1].stop(); + this->_custom_timers[2].start(); + _intermediate_LLT.compute(_intermediate_matrix); + this->_custom_timers[2].stop(); + + if (_intermediate_LLT.info() == Eigen::NumericalIssue) { + return false; + } else { + this->_stored_intermediate_LLT[0].first = new_stored_x; + this->_stored_intermediate_LLT[0].second = _intermediate_LLT; + } + } + + if (check_interior_only) { + return true; + } + + this->_custom_timers[3].start(); + _V.noalias() = _intermediate_LLT.matrixL().solve(_P.transpose()); + this->_custom_timers[3].stop(); + + //Experiments showed that using the triangularView instead would slow down the program. + //So we use the full Matrix _V to compute its product with the transpose. + + this->_custom_timers[4].start(); + compute_V_transpose_V(); + this->_custom_timers[4].stop(); + } + + //TODO: store hessian as self-adjoint + if (this->_stored_hessians.empty()) { + this->_stored_hessians.resize(1); + } + this->_stored_hessians[0].first = new_stored_x; + this->_stored_hessians[0].second.noalias() = _g_g_transpose.cwiseProduct(_Q.cwiseProduct(_Q)); + + if (this->_stored_gradients.empty()) { + this->_stored_gradients.resize(1); + } + this->_stored_gradients[0].first = new_stored_x; + this->_stored_gradients[0].second.noalias() = -_Q.diagonal().cwiseProduct(_g); + + if (this->_stored_LLT.empty()) { + this->_stored_LLT.resize(1); + } + + this->_stored_LLT[0].first = new_stored_x; + this->_custom_timers[5].start(); + this->_stored_LLT[0].second.compute(this->_stored_hessians[0].second.template selfadjointView()); + this->_custom_timers[5].stop(); + + return true; +} + +template +Vector InterpolantDualSOSBarrier::gradient(Vector x) { + auto *grad_ptr = this->find_gradient(x); + if (grad_ptr) { + return *grad_ptr; + } + update_gradient_hessian_LLT(x); + return this->_stored_gradients[0].second; +} + +template +Matrix InterpolantDualSOSBarrier::hessian(Vector x) { + auto *hess_ptr = this->find_hessian(x); + if (hess_ptr) { + return *hess_ptr; + } + update_gradient_hessian_LLT(x); + return this->_stored_hessians[0].second; +} + +template +Eigen::LLT > InterpolantDualSOSBarrier::llt(Vector x, bool) { + auto *llt_ptr = this->find_LLT(x); + if (llt_ptr) { + return *llt_ptr; + } + update_gradient_hessian_LLT(x); + return this->_stored_LLT[0].second; +} + +//Should not be invoked as it is slow. +template +Matrix InterpolantDualSOSBarrier::inverse_hessian(Vector x) { + Matrix L_inv = llt(x).matrixL().toDenseMatrix().inverse(); + Matrix inv = L_inv.transpose() * L_inv; + return inv; +} + +template +bool InterpolantDualSOSBarrier::in_interior(Vector x) { + //The computational effort to calculate whether x is in the interior is nearly as high + //as computing gradient and hessian. Therefore we just calculate them here as well. + bool check_interior_only = true; + return update_gradient_hessian_LLT(x, check_interior_only); +} + +template +IPMDouble InterpolantDualSOSBarrier::concordance_parameter(Vector) { + return _L; +} + +template +Vector InterpolantDualSOSBarrier::initialize_x() { + return Vector::Ones(_U); +} + +template +Vector InterpolantDualSOSBarrier::initialize_s() { + return -gradient(initialize_x()); +} + diff --git a/src/volesti/include/sos/barriers/LHSCB.h b/src/volesti/include/sos/barriers/LHSCB.h new file mode 100644 index 00000000..dd19d099 --- /dev/null +++ b/src/volesti/include/sos/barriers/LHSCB.h @@ -0,0 +1,89 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_LHSCB_H +#define SOS_LHSCB_H + +#include +#include +#include "utils.h" + +template +class LHSCB { + typedef Vector Vector; + typedef Matrix Matrix; +public: + LHSCB() : _num_variables(0) {}; + + template + void cast_members_from(const LHSCB & other){ + _num_variables = other._num_variables; + } + + template + LHSCB * cast(){ + LHSCB * lhscb; + (*lhscb).template cast_members_from(*this); + return lhscb; + } + + virtual ~LHSCB() {}; + + virtual Vector gradient(Vector x) = 0; + + virtual Matrix hessian(Vector x) = 0; + + virtual Eigen::LLT llt(Vector x, bool symmetrize = 0); + + virtual Matrix llt_solve(Vector x, const Matrix &rhs); + + virtual Vector llt_L_solve(Vector x, Vector rhs); + + Vector *find_gradient(Vector x); + + Matrix *find_hessian(Vector x); + + Eigen::LLT *find_LLT(Vector x); + + virtual Matrix inverse_hessian(Vector x); + + virtual bool in_interior(Vector x) = 0; + + virtual IPMDouble concordance_parameter(Vector x) = 0; + + virtual Vector initialize_x(IPMDouble parameter) { + return parameter * initialize_x(); + } + + //TODO: figure out if initializing the dual is in general just - 1/mu * g(x) (i.e. whether this is in the dual cone) + virtual Vector initialize_s(IPMDouble parameter) { + return initialize_s() / parameter; + } + + virtual Vector initialize_x() = 0; + + virtual Vector initialize_s() = 0; + + cxxtimer::Timer _in_interior_timer; + std::vector _custom_timers; + + unsigned _num_variables; + + +protected: + + std::vector > _stored_gradients; + std::vector > _stored_hessians; + std::vector > > _stored_LLT; + std::vector > > _stored_intermediate_LLT; + +public: + unsigned int getNumVariables() const; +}; + +#include "LHSCB.hpp" + +#endif //SOS_LHSCB_H diff --git a/src/volesti/include/sos/barriers/LHSCB.hpp b/src/volesti/include/sos/barriers/LHSCB.hpp new file mode 100644 index 00000000..bf209a92 --- /dev/null +++ b/src/volesti/include/sos/barriers/LHSCB.hpp @@ -0,0 +1,89 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include +#include "LHSCB.h" + +template +Matrix LHSCB::inverse_hessian(Vector x) { + Eigen::LLT LLT = llt(x); + Matrix L_inv = LLT.matrixL().toDenseMatrix().inverse(); + return L_inv.transpose() * L_inv; +} + +template +unsigned int LHSCB::getNumVariables() const { + return _num_variables; +} + +//TODO: use short queue instead. +template +Vector *LHSCB::find_gradient(Vector x) { + for (int i = _stored_gradients.size() - 1; i >= 0; i--) { + if (x == _stored_gradients[i].first) { + return &_stored_gradients[i].second; + } + } + return nullptr; +} + +template +Matrix *LHSCB::find_hessian(Vector x) { + for (int i = _stored_hessians.size() - 1; i >= 0; i--) { + if (x == _stored_hessians[i].first) { + return &_stored_hessians[i].second; + } + } + return nullptr; +} + +template +Eigen::LLT > *LHSCB::find_LLT(Vector x) { + for (int i = _stored_LLT.size() - 1; i >= 0; i--) { + if (x == _stored_LLT[i].first) { + return &_stored_LLT[i].second; + } + } + return nullptr; +} + +template +Eigen::LLT > LHSCB::llt(Vector x, bool symmetrize) { + + Eigen::LLT *llt_ptr = nullptr; + + if (not symmetrize) { + llt_ptr = find_LLT(x); + } + + if (llt_ptr) { + return *llt_ptr; + } + + if (_stored_LLT.empty()) { + _stored_LLT.resize(1); + } + + _stored_LLT[0].first = x; + if (not symmetrize) { + _stored_LLT[0].second = Eigen::LLT(hessian(x).llt()); + } else { + Matrix hess_tmp = hessian(x); + _stored_LLT[0].second = Eigen::LLT(((hess_tmp+ hess_tmp.transpose()) / 2).llt()); + } + + return _stored_LLT[0].second; +} + +template +Matrix LHSCB::llt_solve(Vector x, const Matrix &rhs) { + return llt(x).solve(rhs); +} + +template +Vector LHSCB::llt_L_solve(Vector x, Vector rhs) { + return llt(x).matrixL().solve(rhs); +} diff --git a/src/volesti/include/sos/barriers/LPStandardBarrier.h b/src/volesti/include/sos/barriers/LPStandardBarrier.h new file mode 100644 index 00000000..a0de7a48 --- /dev/null +++ b/src/volesti/include/sos/barriers/LPStandardBarrier.h @@ -0,0 +1,47 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_LPSTANDARDBARRIER_H +#define SOS_LPSTANDARDBARRIER_H + +#include "LHSCB.h" + +template +class LPStandardBarrier final : public LHSCB { + using LHSCB = LHSCB; + + typedef Vector Vector; + typedef Matrix Matrix; + +public: + LPStandardBarrier() : LHSCB() {}; + + LPStandardBarrier(unsigned num_variables_) { + this->_num_variables = num_variables_; + }; + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + Matrix inverse_hessian(Vector x) override; + + bool in_interior(Vector x) override; + + //TODO: better solution for concordance parameter; + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + + Vector initialize_s() override; + +private: + +}; + +#include "LPStandardBarrier.hpp" + +#endif //SOS_LPSTANDARDBARRIER_H diff --git a/src/volesti/include/sos/barriers/LPStandardBarrier.hpp b/src/volesti/include/sos/barriers/LPStandardBarrier.hpp new file mode 100644 index 00000000..9a6815a9 --- /dev/null +++ b/src/volesti/include/sos/barriers/LPStandardBarrier.hpp @@ -0,0 +1,43 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include "LPStandardBarrier.h" + +template +bool LPStandardBarrier::in_interior(Vector x) { + return (x.minCoeff() > 0); +} + +template +Vector LPStandardBarrier::gradient(Vector x) { + assert(in_interior(x)); + return -x.array().inverse(); +} + +template +Matrix LPStandardBarrier::hessian(Vector x) { + return x.array().pow(2).inverse().matrix().asDiagonal(); +} + +template +Matrix LPStandardBarrier::inverse_hessian(Vector x) { + return hessian(x).inverse(); +} + +template +IPMDouble LPStandardBarrier::concordance_parameter(Vector x) { + return x.rows(); +} + +template +Vector LPStandardBarrier::initialize_x() { + return Vector::Ones(this->_num_variables); +} + +template +Vector LPStandardBarrier::initialize_s() { + return Vector::Ones(this->_num_variables); +} diff --git a/src/volesti/include/sos/barriers/ProductBarrier.h b/src/volesti/include/sos/barriers/ProductBarrier.h new file mode 100644 index 00000000..be5e4329 --- /dev/null +++ b/src/volesti/include/sos/barriers/ProductBarrier.h @@ -0,0 +1,113 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_PRODUCTBARRIER_H +#define SOS_PRODUCTBARRIER_H + +#include "LHSCB.h" +#include "SumBarrier.h" +#include "InterpolantDualSOSBarrier.h" + + +template +class ProductBarrier : public LHSCB { + + typedef Vector Vector; + typedef Matrix Matrix; + + typedef Vector (LHSCB::*VectorFunc)(Vector); + typedef Matrix (LHSCB::*MatrixFunc)(Vector); + typedef Vector (LHSCB::*VoidFunc)(); + +public: + ProductBarrier(unsigned num_threads = 1) : LHSCB() { + _num_threads = num_threads; + } + + ProductBarrier(std::vector *> barriers_, std::vector num_variables_, unsigned num_threads = 1) { + assert(barriers_.size() == num_variables_.size()); + _num_threads = num_threads; + for (unsigned j = 0; j < barriers_.size(); ++j) { + _barriers.push_back(barriers_[j]); + _num_vars_per_barrier.push_back(num_variables_[j]); + } + } + + template + ProductBarrier * cast(){ + ProductBarrier * pb = new ProductBarrier(); + pb->_num_threads = _num_threads; + pb->_segments = _segments; + for(LHSCB * barrier : _barriers){ + //TODO: High priority. Figure out how to undo the cast to the SumBarrier. + LHSCB * new_barr; + if(dynamic_cast*>(barrier)){ + new_barr = static_cast*>(barrier)->template cast(); + } else { + if(dynamic_cast*>(barrier)){ + new_barr = static_cast*>(barrier)->template cast(); + } else { + exit(1); + } + } + pb->add_barrier(new_barr); + } + return pb; + }; + + void add_barrier(LHSCB *lhscb) { + _barriers.push_back(lhscb); + _num_vars_per_barrier.push_back(lhscb->getNumVariables()); + this->_num_variables += lhscb->getNumVariables(); + } + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + +// Matrix inverse_hessian(Vector x) override; + bool in_interior(Vector x) override; + + //TODO: better solution for concordance parameter; + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + + Vector initialize_s() override; + + Eigen::LLT llt(Vector x, bool symmetrize = 0) override; + + Matrix llt_solve(Vector x, const Matrix &rhs) override; + + Vector llt_L_solve(Vector x, Vector rhs) override; + + Matrix inverse_hessian(Vector x) override; + + void update_segments(); + std::vector > & get_segments(){ + return _segments; + }; + + Vector evaluate(Vector x, VectorFunc func); + Matrix evaluate(Vector x, MatrixFunc func); + Vector evaluate(VoidFunc func); + + std::vector *> & get_barriers(){ + return _barriers; + } + + std::vector *> _barriers; + std::vector > _segments; + std::vector _num_vars_per_barrier; + unsigned _num_threads; + +private: + +}; + +#include "ProductBarrier.hpp" + +#endif //SOS_PRODUCTBARRIER_H diff --git a/src/volesti/include/sos/barriers/ProductBarrier.hpp b/src/volesti/include/sos/barriers/ProductBarrier.hpp new file mode 100644 index 00000000..722a0c47 --- /dev/null +++ b/src/volesti/include/sos/barriers/ProductBarrier.hpp @@ -0,0 +1,173 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + + +#include "ProductBarrier.h" + +//TODO: Figure out how to compute llt decomposition with diagonal blocks. There +// does not seem to be a straightforward way with Eigen. +template +Eigen::LLT > ProductBarrier::llt(Vector x, bool symmetrize) { + return LHSCB::llt(x, symmetrize); +} + +//TODO: figure out how these methods can be even more abstracted. +template +Vector ProductBarrier::llt_L_solve(Vector x, Vector rhs) { + update_segments(); + Vector product_llt_solve(this->_num_variables); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair & seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector x_seg = x.segment(seg.first, seg.second - seg.first); + Vector rhs_seg = rhs.segment(seg.first, seg.second - seg.first); + Vector lls_solve_seg = rhs_seg.nonZeros() ? barrier->llt_L_solve(x_seg, rhs_seg) + : Vector::Zero(rhs_seg.rows()); + product_llt_solve.segment(seg.first , seg.second - seg.first) = lls_solve_seg; + } + return product_llt_solve; +} + +template +Matrix ProductBarrier::llt_solve(Vector x, const Matrix &rhs) { + update_segments(); + Matrix product_llt_solve = Matrix::Zero(rhs.rows(), rhs.cols()); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair & seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector x_seg = x.segment(seg.first, seg.second - seg.first); + Matrix rhs_block = rhs.block(seg.first, 0, seg.second-seg.first, rhs.cols()); + Matrix lls_solve_block = barrier->llt_solve(x_seg, rhs_block); + product_llt_solve.block(seg.first, 0, seg.second - seg.first, rhs.cols()) = lls_solve_block; + } + return product_llt_solve; +} + +//Should not be used, as immense storage is needed. +template +Matrix ProductBarrier::hessian(Vector x) { + return evaluate(x, &LHSCB::hessian); +} + +template +bool ProductBarrier::in_interior(Vector x) { + this->_in_interior_timer.start(); + update_segments(); + std::vector in_interior_vec(_barriers.size()); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair & seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector x_seg = x.segment(seg.first, seg.second - seg.first); + in_interior_vec[i] = barrier->in_interior(x_seg); + } + this->_in_interior_timer.stop(); + return std::all_of(in_interior_vec.begin(), in_interior_vec.end(), [](bool b){return b;}); +} + +template +IPMDouble ProductBarrier::concordance_parameter(Vector x) { + unsigned idx = 0; + IPMDouble concordance_par = 0; + for (unsigned i = 0; i < _barriers.size(); ++i) { + LHSCB *barrier = _barriers[i]; + unsigned num_variables = _num_vars_per_barrier[i]; + Vector v_segment = x.segment(idx, num_variables); + concordance_par += barrier->concordance_parameter(v_segment); + idx += num_variables; + } + return concordance_par; +} + +template +Vector ProductBarrier::initialize_x() { + return evaluate(&LHSCB::initialize_x); +} + +template +Vector ProductBarrier::initialize_s() { + return evaluate(&LHSCB::initialize_s); +} + +template +Matrix ProductBarrier::inverse_hessian(Vector x) { + return evaluate(x, &LHSCB::inverse_hessian); +} + +template +Vector ProductBarrier::gradient(Vector x) { + return evaluate(x, &LHSCB::gradient); +} + +template +void ProductBarrier::update_segments() { + _segments.resize(0); + unsigned idx = 0; + std::vector > segments; + for (unsigned i = 0; i < _barriers.size(); ++i) { + unsigned num_variables = _num_vars_per_barrier[i]; + _segments.push_back(std::pair(idx, idx + num_variables)); + idx += num_variables; + } +} + +template +Vector ProductBarrier::evaluate(Vector x, VectorFunc func) { + update_segments(); + Vector product_vector(this->_num_variables); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair &seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector x_seg = x.segment(seg.first, seg.second - seg.first); + Vector vec_seg = (barrier->*func)(x_seg); + product_vector.segment(seg.first, seg.second - seg.first) = vec_seg; + } + return product_vector; +} + +template +Matrix ProductBarrier::evaluate(Vector x, MatrixFunc func) { + update_segments(); + Matrix product_matrix = Matrix::Zero(this->_num_variables, this->_num_variables); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair &seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector x_seg = x.segment(seg.first, seg.second - seg.first); + Matrix matrix_block = (barrier->*func)(x_seg); + product_matrix.block(seg.first, seg.first, seg.second - seg.first, seg.second - seg.first) = matrix_block; + } + return product_matrix; +} + +template +Vector ProductBarrier::evaluate(VoidFunc func) { + update_segments(); + Vector product_vector(this->_num_variables); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (unsigned i = 0; i < _barriers.size(); ++i) { + std::pair &seg = _segments[i]; + LHSCB *barrier = _barriers[i]; + Vector vec_seg = (barrier->*func)(); + product_vector.segment(seg.first, seg.second - seg.first) = vec_seg; + } + return product_vector; +} diff --git a/src/volesti/include/sos/barriers/SumBarrier.h b/src/volesti/include/sos/barriers/SumBarrier.h new file mode 100644 index 00000000..7252cab5 --- /dev/null +++ b/src/volesti/include/sos/barriers/SumBarrier.h @@ -0,0 +1,59 @@ +// +// Created by test Bento Natura on 30/07/2020. +// + +#ifndef SOS_SUMBARRIER_H +#define SOS_SUMBARRIER_H + +#include "LHSCB.h" +#include "InterpolantDualSOSBarrier.h" + +//corresponds to intersection of cones. +template +class SumBarrier : public LHSCB { + + typedef Vector Vector; + typedef Matrix Matrix; + +public: + SumBarrier(unsigned num_variables_); + + SumBarrier(std::vector *> barriers_, unsigned num_variables_); + + template + SumBarrier * cast(){ + SumBarrier * sb = new SumBarrier(this->_num_variables); + for(LHSCB * barrier : _barriers){ + //TODO: High priority. Figure out how to undo the cast to the InterpolantBarrier. + InterpolantDualSOSBarrier * new_barr = static_cast*>(barrier)->template cast(); +// LHSCB * new_barr = (*barrier).template cast(); + sb->add_barrier(new_barr); + } + return sb; + }; + + void add_barrier(LHSCB *lhscb); + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + bool in_interior(Vector x) override; + + //TODO: better solution for concordance parameter; + IPMDouble concordance_parameter(Vector x) override; + + //TODO: verify that initialisation works for both primal and dual side in general. + Vector initialize_x() override; + + Vector initialize_s() override; + + std::vector *> & get_barriers(){ + return _barriers; + } +private: + std::vector *> _barriers; +}; + +#include "SumBarrier.hpp" +#endif //SOS_SUMBARRIER_H diff --git a/src/volesti/include/sos/barriers/SumBarrier.hpp b/src/volesti/include/sos/barriers/SumBarrier.hpp new file mode 100644 index 00000000..40cb48ba --- /dev/null +++ b/src/volesti/include/sos/barriers/SumBarrier.hpp @@ -0,0 +1,101 @@ +// +// Created by test Bento Natura on 30/07/2020. +// + +#include "SumBarrier.h" + +template +SumBarrier::SumBarrier(unsigned num_variables_) : LHSCB() { + this->_num_variables = num_variables_; +} + +template +SumBarrier::SumBarrier(std::vector *> barriers_, unsigned num_variables_) { + this->_num_variables = num_variables_; + for (unsigned j = 0; j < barriers_.size(); ++j) { + assert(barriers_[j]->getNumVariables() == num_variables_); + _barriers.push_back(barriers_[j]); + } +} + +template +void SumBarrier::add_barrier(LHSCB *lhscb) { + _barriers.push_back(lhscb); +} + +template +Vector SumBarrier::gradient(Vector x) { + Vector grad_vec = Vector::Zero(x.rows()); + std::vector barrier_vec(_barriers.size()); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (int i = 0; i < _barriers.size(); i++){ + barrier_vec[i] = _barriers[i]->gradient(x); + } + for(int i = 0; i < _barriers.size(); i++){ + grad_vec += barrier_vec[i]; + } + return grad_vec; +} + +template +Matrix SumBarrier::hessian(Vector x) { + Matrix hess_mat = Matrix::Zero(x.rows(), x.rows()); + std::vector barrier_vec(_barriers.size()); +#ifdef PARALLELIZE_BARRIERS +#pragma omp parallel for +#endif + for (int i = 0; i < _barriers.size(); i++){ + barrier_vec[i] = _barriers[i]->hessian(x); + } + for(int i = 0; i < _barriers.size(); i++){ + hess_mat += barrier_vec[i]; + } + + return hess_mat; +} + +template +bool SumBarrier::in_interior(Vector x) { + std::vector barrier_vec(_barriers.size()); +#pragma omp parallel for + for (int i = 0; i < _barriers.size(); i++){ + barrier_vec[i] = _barriers[i]->in_interior(x); + } + return std::all_of(barrier_vec.begin(), barrier_vec.end(), [](bool b){return b;}); +} + +//TODO: better solution for concordance parameter; +template +IPMDouble SumBarrier::concordance_parameter(Vector x) { + IPMDouble conc_par = 0.; + for ( + auto barrier + : _barriers) { + conc_par += barrier-> + concordance_parameter(x); + } + return conc_par; +} + +//TODO: verify that initialisation works for both primal and dual side in general. +template +Vector SumBarrier::initialize_x() { + assert(!_barriers.empty()); + Vector rs = _barriers[0]->initialize_x(); + return rs; + +} + +template +Vector SumBarrier::initialize_s() { + Vector s_init = Vector::Zero(this->_num_variables); + for ( + auto barrier + : _barriers) { + s_init += barrier->initialize_s(); + } + return s_init; +} + diff --git a/src/volesti/include/sos/barriers/ZeroSpaceBarrier.h b/src/volesti/include/sos/barriers/ZeroSpaceBarrier.h new file mode 100644 index 00000000..e2a11929 --- /dev/null +++ b/src/volesti/include/sos/barriers/ZeroSpaceBarrier.h @@ -0,0 +1,44 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_ZEROSPACEBARRIER_H +#define SOS_ZEROSPACEBARRIER_H + +#include "LHSCB.h" + +//This class models the 0-cone. Should not be used, so reformulate your +//instance. +template +class ZeroSpaceBarrier final : public LHSCB { + + using LHSCB = LHSCB; + + typedef Vector Vector; + typedef Matrix Matrix; + + Vector gradient(Vector x) override; + + Matrix hessian(Vector x) override; + + Matrix inverse_hessian(Vector x) override; + + bool in_interior(Vector x) override; + + IPMDouble concordance_parameter(Vector x) override; + + Vector initialize_x() override; + + Vector initialize_s() override; + +public: + ZeroSpaceBarrier(unsigned num_variables_) { + this->_num_variables = num_variables_; + } +}; + +#include "ZeroSpaceBarrier.hpp" + +#endif //SOS_ZEROSPACEBARRIER_H diff --git a/src/volesti/include/sos/barriers/ZeroSpaceBarrier.hpp b/src/volesti/include/sos/barriers/ZeroSpaceBarrier.hpp new file mode 100644 index 00000000..1381fbcd --- /dev/null +++ b/src/volesti/include/sos/barriers/ZeroSpaceBarrier.hpp @@ -0,0 +1,46 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#include "ZeroSpaceBarrier.h" + +template +Vector ZeroSpaceBarrier::gradient(Vector x) { + //should not be used + assert(false); + return -std::numeric_limits::infinity() * Vector::Ones(x.cols()); +} + +template +Matrix ZeroSpaceBarrier::hessian(Vector x) { + //should not be used + assert(false); + return std::numeric_limits::infinity() * Matrix::Identity(x.cols(), x.cols()); +} + +template +Matrix ZeroSpaceBarrier::inverse_hessian(Vector) { + return Matrix::Zero(this->_num_variables, this->_num_variables); +} + +template +bool ZeroSpaceBarrier::in_interior(Vector) { + return true; +} + +template +IPMDouble ZeroSpaceBarrier::concordance_parameter(Vector) { + return 0; +} + +template +Vector ZeroSpaceBarrier::initialize_x() { + return Vector::Zero(this->_num_variables); +} + +template +Vector ZeroSpaceBarrier::initialize_s() { + return Vector::Zero(this->_num_variables); +} diff --git a/src/volesti/include/sos/config/config.json b/src/volesti/include/sos/config/config.json new file mode 100644 index 00000000..f3b150f5 --- /dev/null +++ b/src/volesti/include/sos/config/config.json @@ -0,0 +1,21 @@ +{ + "IPM": { + "epsilon": 1e-6, + "num_corrector_steps": 4, + "large_neighborhood": 5, + "small_neighborhood": 0.05, + "scale_predictor_step": 1, + "length_corrector_step": 1, + "logger_level": 2, + "use_line_search": true, + "check_centrality_in_every_segment": true, + "type_cast_if_unsuccessful": true + }, + "InterpolantBarrier": { + "use_low_rank_updates": false + }, + + "use_weighted_polynomials": true, + "input_in_interpolant_basis": true, + "plot_threshold": 20 +} diff --git a/src/volesti/include/sos/gsoc_history.md b/src/volesti/include/sos/gsoc_history.md new file mode 100644 index 00000000..df4354de --- /dev/null +++ b/src/volesti/include/sos/gsoc_history.md @@ -0,0 +1,15 @@ +#### Weekly Progress + +* Week 1: Implementation of [1,2]. +* Week 2: Testing implementation with LP and SDP barrier, Debugging, Refactoring. +* Week 3: Implementation of Interpolant-Barrier, Focus on speeding up implementation with dedicated libraries. +* Week 4: Implementation of Toy Problem: Polynomial Envelope. +* Week 5: Debugging, Refactoring Code. +* Week 6: Debugging, Refactoring Code, Speeding up initialisation of "Polynomial Envelope" Problem + (non-trivial: properties of Chebyshev polynomials, Clenshaw-Curtis algorithm) + * Week 7: Runtime speedups, more efficient (including experimentally) implementation. + * Week 8: Further speedup, Big code refactoring, Add Polynomial minimization, Create Benchmarks with alfonso. + * Week 9: Writing of documentation / report, more refactoring, add multivariate support. + * Week 10: Improve multivariate support, Added multithreading. + * Week 11: Added low-rank updates to both Hessian and intermediate Cholesky decomposition, templating algorithm. + * Week 12: MKL Support, JSON parsing of configuration, templating algorithm diff --git a/src/volesti/include/sos/include/cxxtimer.hpp b/src/volesti/include/sos/include/cxxtimer.hpp new file mode 100644 index 00000000..c3f0b722 --- /dev/null +++ b/src/volesti/include/sos/include/cxxtimer.hpp @@ -0,0 +1,184 @@ +/* + +MIT License + +Copyright (c) 2017 André L. Maravilha + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +*/ + +#ifndef CXX_TIMER_HPP +#define CXX_TIMER_HPP + +#include + + +namespace cxxtimer { + +/** + * This class works as a stopwatch. + */ +class Timer { + +public: + + /** + * Constructor. + * + * @param start + * If true, the timer is started just after construction. + * Otherwise, it will not be automatically started. + */ + Timer(bool start = false); + + /** + * Copy constructor. + * + * @param other + * The object to be copied. + */ + Timer(const Timer& other) = default; + + /** + * Transfer constructor. + * + * @param other + * The object to be transfered. + */ + Timer(Timer&& other) = default; + + /** + * Destructor. + */ + virtual ~Timer() = default; + + /** + * Assignment operator by copy. + * + * @param other + * The object to be copied. + * + * @return A reference to this object. + */ + Timer& operator=(const Timer& other) = default; + + /** + * Assignment operator by transfer. + * + * @param other + * The object to be transferred. + * + * @return A reference to this object. + */ + Timer& operator=(Timer&& other) = default; + + /** + * Start/resume the timer. + */ + void start(); + + /** + * Stop/pause the timer. + */ + void stop(); + + /** + * Reset the timer. + */ + void reset(); + + /** + * Return the elapsed time. + * + * @param duration_t + * The duration type used to return the time elapsed. If not + * specified, it returns the time as represented by + * std::chrono::milliseconds. + * + * @return The elapsed time. + */ + template + typename duration_t::rep count() const; + +private: + + bool started_; + bool paused_; + std::chrono::steady_clock::time_point reference_; + std::chrono::duration accumulated_; +}; + +} + + +inline cxxtimer::Timer::Timer(bool start) : + started_(false), paused_(false), + reference_(std::chrono::steady_clock::now()), + accumulated_(std::chrono::duration(0)) { + if (start) { + this->start(); + } +} + +inline void cxxtimer::Timer::start() { + if (!started_) { + started_ = true; + paused_ = false; + accumulated_ = std::chrono::duration(0); + reference_ = std::chrono::steady_clock::now(); + } else if (paused_) { + reference_ = std::chrono::steady_clock::now(); + paused_ = false; + } +} + +inline void cxxtimer::Timer::stop() { + if (started_ && !paused_) { + std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now(); + accumulated_ = accumulated_ + std::chrono::duration_cast< std::chrono::duration >(now - reference_); + paused_ = true; + } +} + +inline void cxxtimer::Timer::reset() { + if (started_) { + started_ = false; + paused_ = false; + reference_ = std::chrono::steady_clock::now(); + accumulated_ = std::chrono::duration(0); + } +} + +template +typename duration_t::rep cxxtimer::Timer::count() const { + if (started_) { + if (paused_) { + return std::chrono::duration_cast(accumulated_).count(); + } else { + return std::chrono::duration_cast( + accumulated_ + (std::chrono::steady_clock::now() - reference_)).count(); + } + } else { + return duration_t(0).count(); + } +} + + +#endif diff --git a/src/volesti/include/sos/include/matplotlib-cpp/matplotlibcpp.h b/src/volesti/include/sos/include/matplotlib-cpp/matplotlibcpp.h new file mode 100644 index 00000000..4b2b241f --- /dev/null +++ b/src/volesti/include/sos/include/matplotlib-cpp/matplotlibcpp.h @@ -0,0 +1,2366 @@ +//The MIT License (MIT) +// +//Copyright (c) 2014 Benno Evers +// +//Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +//in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +//furnished to do so, subject to the following conditions: +// +//The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +//THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +//IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +//FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +//LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +//OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +//SOFTWARE. + +//Credit to https://github.com/lava/matplotlib-cpp/ + +#pragma once + +// Python headers must be included before any system headers, since +// they define _POSIX_C_SOURCE +#include + +#include +#include +#include +#include +#include +#include +#include +#include // requires c++11 support +#include + +#ifndef WITHOUT_NUMPY +# define NPY_NO_DEPRECATED_API NPY_1_7_API_VERSION +# include + +# ifdef WITH_OPENCV +# include +# endif // WITH_OPENCV + +/* + * A bunch of constants were removed in OpenCV 4 in favour of enum classes, so + * define the ones we need here. + */ +# if CV_MAJOR_VERSION > 3 +# define CV_BGR2RGB cv::COLOR_BGR2RGB +# define CV_BGRA2RGBA cv::COLOR_BGRA2RGBA +# endif +#endif // WITHOUT_NUMPY + +#if PY_MAJOR_VERSION >= 3 +# define PyString_FromString PyUnicode_FromString +# define PyInt_FromLong PyLong_FromLong +# define PyString_FromString PyUnicode_FromString +#endif + + +namespace matplotlibcpp { +namespace detail { + +static std::string s_backend; + +struct _interpreter { + PyObject *s_python_function_show; + PyObject *s_python_function_close; + PyObject *s_python_function_draw; + PyObject *s_python_function_pause; + PyObject *s_python_function_save; + PyObject *s_python_function_figure; + PyObject *s_python_function_fignum_exists; + PyObject *s_python_function_plot; + PyObject *s_python_function_quiver; + PyObject *s_python_function_semilogx; + PyObject *s_python_function_semilogy; + PyObject *s_python_function_loglog; + PyObject *s_python_function_fill; + PyObject *s_python_function_fill_between; + PyObject *s_python_function_hist; + PyObject *s_python_function_imshow; + PyObject *s_python_function_scatter; + PyObject *s_python_function_boxplot; + PyObject *s_python_function_subplot; + PyObject *s_python_function_subplot2grid; + PyObject *s_python_function_legend; + PyObject *s_python_function_xlim; + PyObject *s_python_function_ion; + PyObject *s_python_function_ginput; + PyObject *s_python_function_ylim; + PyObject *s_python_function_title; + PyObject *s_python_function_axis; + PyObject *s_python_function_axvline; + PyObject *s_python_function_xlabel; + PyObject *s_python_function_ylabel; + PyObject *s_python_function_gca; + PyObject *s_python_function_xticks; + PyObject *s_python_function_yticks; + PyObject *s_python_function_tick_params; + PyObject *s_python_function_grid; + PyObject *s_python_function_clf; + PyObject *s_python_function_errorbar; + PyObject *s_python_function_annotate; + PyObject *s_python_function_tight_layout; + PyObject *s_python_colormap; + PyObject *s_python_empty_tuple; + PyObject *s_python_function_stem; + PyObject *s_python_function_xkcd; + PyObject *s_python_function_text; + PyObject *s_python_function_suptitle; + PyObject *s_python_function_bar; + PyObject *s_python_function_colorbar; + PyObject *s_python_function_subplots_adjust; + + + /* For now, _interpreter is implemented as a singleton since its currently not possible to have + multiple independent embedded python interpreters without patching the python source code + or starting a separate process for each. + http://bytes.com/topic/python/answers/793370-multiple-independent-python-interpreters-c-c-program + */ + + static _interpreter& get() { + static _interpreter ctx; + return ctx; + } + + PyObject* safe_import(PyObject* module, std::string fname) { + PyObject* fn = PyObject_GetAttrString(module, fname.c_str()); + + if (!fn) + throw std::runtime_error(std::string("Couldn't find required function: ") + fname); + + if (!PyFunction_Check(fn)) + throw std::runtime_error(fname + std::string(" is unexpectedly not a PyFunction.")); + + return fn; + } + +private: + +#ifndef WITHOUT_NUMPY +# if PY_MAJOR_VERSION >= 3 + + void *import_numpy() { + import_array(); // initialize C-API + return NULL; + } + +# else + + void import_numpy() { + import_array(); // initialize C-API + } + +# endif +#endif + + _interpreter() { + + // optional but recommended +#if PY_MAJOR_VERSION >= 3 + wchar_t name[] = L"plotting"; +#else + char name[] = "plotting"; +#endif + Py_SetProgramName(name); + Py_Initialize(); + +#ifndef WITHOUT_NUMPY + import_numpy(); // initialize numpy C-API +#endif + + PyObject* matplotlibname = PyString_FromString("matplotlib"); + PyObject* pyplotname = PyString_FromString("matplotlib.pyplot"); + PyObject* cmname = PyString_FromString("matplotlib.cm"); + PyObject* pylabname = PyString_FromString("pylab"); + if (!pyplotname || !pylabname || !matplotlibname || !cmname) { + throw std::runtime_error("couldnt create string"); + } + + PyObject* matplotlib = PyImport_Import(matplotlibname); + Py_DECREF(matplotlibname); + if (!matplotlib) { + PyErr_Print(); + throw std::runtime_error("Error loading module matplotlib!"); + } + + // matplotlib.use() must be called *before* pylab, matplotlib.pyplot, + // or matplotlib.backends is imported for the first time + if (!s_backend.empty()) { + PyObject_CallMethod(matplotlib, const_cast("use"), const_cast("s"), s_backend.c_str()); + } + + PyObject* pymod = PyImport_Import(pyplotname); + Py_DECREF(pyplotname); + if (!pymod) { throw std::runtime_error("Error loading module matplotlib.pyplot!"); } + + s_python_colormap = PyImport_Import(cmname); + Py_DECREF(cmname); + if (!s_python_colormap) { throw std::runtime_error("Error loading module matplotlib.cm!"); } + + PyObject* pylabmod = PyImport_Import(pylabname); + Py_DECREF(pylabname); + if (!pylabmod) { throw std::runtime_error("Error loading module pylab!"); } + + s_python_function_show = safe_import(pymod, "show"); + s_python_function_close = safe_import(pymod, "close"); + s_python_function_draw = safe_import(pymod, "draw"); + s_python_function_pause = safe_import(pymod, "pause"); + s_python_function_figure = safe_import(pymod, "figure"); + s_python_function_fignum_exists = safe_import(pymod, "fignum_exists"); + s_python_function_plot = safe_import(pymod, "plot"); + s_python_function_quiver = safe_import(pymod, "quiver"); + s_python_function_semilogx = safe_import(pymod, "semilogx"); + s_python_function_semilogy = safe_import(pymod, "semilogy"); + s_python_function_loglog = safe_import(pymod, "loglog"); + s_python_function_fill = safe_import(pymod, "fill"); + s_python_function_fill_between = safe_import(pymod, "fill_between"); + s_python_function_hist = safe_import(pymod,"hist"); + s_python_function_scatter = safe_import(pymod,"scatter"); + s_python_function_boxplot = safe_import(pymod,"boxplot"); + s_python_function_subplot = safe_import(pymod, "subplot"); + s_python_function_subplot2grid = safe_import(pymod, "subplot2grid"); + s_python_function_legend = safe_import(pymod, "legend"); + s_python_function_ylim = safe_import(pymod, "ylim"); + s_python_function_title = safe_import(pymod, "title"); + s_python_function_axis = safe_import(pymod, "axis"); + s_python_function_axvline = safe_import(pymod, "axvline"); + s_python_function_xlabel = safe_import(pymod, "xlabel"); + s_python_function_ylabel = safe_import(pymod, "ylabel"); + s_python_function_gca = safe_import(pymod, "gca"); + s_python_function_xticks = safe_import(pymod, "xticks"); + s_python_function_yticks = safe_import(pymod, "yticks"); + s_python_function_tick_params = safe_import(pymod, "tick_params"); + s_python_function_grid = safe_import(pymod, "grid"); + s_python_function_xlim = safe_import(pymod, "xlim"); + s_python_function_ion = safe_import(pymod, "ion"); + s_python_function_ginput = safe_import(pymod, "ginput"); + s_python_function_save = safe_import(pylabmod, "savefig"); + s_python_function_annotate = safe_import(pymod,"annotate"); + s_python_function_clf = safe_import(pymod, "clf"); + s_python_function_errorbar = safe_import(pymod, "errorbar"); + s_python_function_tight_layout = safe_import(pymod, "tight_layout"); + s_python_function_stem = safe_import(pymod, "stem"); + s_python_function_xkcd = safe_import(pymod, "xkcd"); + s_python_function_text = safe_import(pymod, "text"); + s_python_function_suptitle = safe_import(pymod, "suptitle"); + s_python_function_bar = safe_import(pymod,"bar"); + s_python_function_colorbar = PyObject_GetAttrString(pymod, "colorbar"); + s_python_function_subplots_adjust = safe_import(pymod,"subplots_adjust"); +#ifndef WITHOUT_NUMPY + s_python_function_imshow = safe_import(pymod, "imshow"); +#endif + s_python_empty_tuple = PyTuple_New(0); + } + + ~_interpreter() { + Py_Finalize(); + } +}; + +} // end namespace detail + +/// Select the backend +/// +/// **NOTE:** This must be called before the first plot command to have +/// any effect. +/// +/// Mainly useful to select the non-interactive 'Agg' backend when running +/// matplotlibcpp in headless mode, for example on a machine with no display. +/// +/// See also: https://matplotlib.org/2.0.2/api/matplotlib_configuration_api.html#matplotlib.use +inline void backend(const std::string& name) +{ + detail::s_backend = name; +} + +inline bool annotate(std::string annotation, double x, double y) +{ + detail::_interpreter::get(); + + PyObject * xy = PyTuple_New(2); + PyObject * str = PyString_FromString(annotation.c_str()); + + PyTuple_SetItem(xy,0,PyFloat_FromDouble(x)); + PyTuple_SetItem(xy,1,PyFloat_FromDouble(y)); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "xy", xy); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, str); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_annotate, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + + if(res) Py_DECREF(res); + + return res; +} + +namespace detail { + +#ifndef WITHOUT_NUMPY +// Type selector for numpy array conversion +template struct select_npy_type { const static NPY_TYPES type = NPY_NOTYPE; }; //Default +template <> struct select_npy_type { const static NPY_TYPES type = NPY_DOUBLE; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_FLOAT; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_BOOL; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_INT8; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_SHORT; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_INT; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_INT64; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_UINT8; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_USHORT; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_ULONG; }; +template <> struct select_npy_type { const static NPY_TYPES type = NPY_UINT64; }; + +// Sanity checks; comment them out or change the numpy type below if you're compiling on +// a platform where they don't apply +//static_assert(sizeof(long long) == 8); +//template <> struct select_npy_type { const static NPY_TYPES type = NPY_INT64; }; +//static_assert(sizeof(unsigned long long) == 8); +//template <> struct select_npy_type { const static NPY_TYPES type = NPY_UINT64; }; +// TODO: add int, long, etc. + +template +PyObject* get_array(const std::vector& v) +{ + npy_intp vsize = v.size(); + NPY_TYPES type = select_npy_type::type; + if (type == NPY_NOTYPE) { + size_t memsize = v.size()*sizeof(double); + double* dp = static_cast(::malloc(memsize)); + for (size_t i=0; i(varray), NPY_ARRAY_OWNDATA); + return varray; + } + + PyObject* varray = PyArray_SimpleNewFromData(1, &vsize, type, (void*)(v.data())); + return varray; +} + + +template +PyObject* get_2darray(const std::vector<::std::vector>& v) +{ + if (v.size() < 1) throw std::runtime_error("get_2d_array v too small"); + + npy_intp vsize[2] = {static_cast(v.size()), + static_cast(v[0].size())}; + + PyArrayObject *varray = + (PyArrayObject *)PyArray_SimpleNew(2, vsize, NPY_DOUBLE); + + double *vd_begin = static_cast(PyArray_DATA(varray)); + + for (const ::std::vector &v_row : v) { + if (v_row.size() != static_cast(vsize[1])) + throw std::runtime_error("Missmatched array size"); + std::copy(v_row.begin(), v_row.end(), vd_begin); + vd_begin += vsize[1]; + } + + return reinterpret_cast(varray); +} + +#else // fallback if we don't have numpy: copy every element of the given vector + +template +PyObject* get_array(const std::vector& v) +{ + PyObject* list = PyList_New(v.size()); + for(size_t i = 0; i < v.size(); ++i) { + PyList_SetItem(list, i, PyFloat_FromDouble(v.at(i))); + } + return list; +} + +#endif // WITHOUT_NUMPY + +// sometimes, for labels and such, we need string arrays +inline PyObject * get_array(const std::vector& strings) +{ + PyObject* list = PyList_New(strings.size()); + for (std::size_t i = 0; i < strings.size(); ++i) { + PyList_SetItem(list, i, PyString_FromString(strings[i].c_str())); + } + return list; +} + +// not all matplotlib need 2d arrays, some prefer lists of lists +template +PyObject* get_listlist(const std::vector>& ll) +{ + PyObject* listlist = PyList_New(ll.size()); + for (std::size_t i = 0; i < ll.size(); ++i) { + PyList_SetItem(listlist, i, get_array(ll[i])); + } + return listlist; +} + +} // namespace detail + +/// Plot a line through the given x and y data points.. +/// +/// See: https://matplotlib.org/3.2.1/api/_as_gen/matplotlib.pyplot.plot.html +template +bool plot(const std::vector &x, const std::vector &y, const std::map& keywords) +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + // using numpy arrays + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + // construct positional args + PyObject* args = PyTuple_New(2); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, yarray); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_plot, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +// TODO - it should be possible to make this work by implementing +// a non-numpy alternative for `detail::get_2darray()`. +#ifndef WITHOUT_NUMPY +template +void plot_surface(const std::vector<::std::vector> &x, + const std::vector<::std::vector> &y, + const std::vector<::std::vector> &z, + const std::map &keywords = + std::map()) +{ + detail::_interpreter::get(); + + // We lazily load the modules here the first time this function is called + // because I'm not sure that we can assume "matplotlib installed" implies + // "mpl_toolkits installed" on all platforms, and we don't want to require + // it for people who don't need 3d plots. + static PyObject *mpl_toolkitsmod = nullptr, *axis3dmod = nullptr; + if (!mpl_toolkitsmod) { + detail::_interpreter::get(); + + PyObject* mpl_toolkits = PyString_FromString("mpl_toolkits"); + PyObject* axis3d = PyString_FromString("mpl_toolkits.mplot3d"); + if (!mpl_toolkits || !axis3d) { throw std::runtime_error("couldnt create string"); } + + mpl_toolkitsmod = PyImport_Import(mpl_toolkits); + Py_DECREF(mpl_toolkits); + if (!mpl_toolkitsmod) { throw std::runtime_error("Error loading module mpl_toolkits!"); } + + axis3dmod = PyImport_Import(axis3d); + Py_DECREF(axis3d); + if (!axis3dmod) { throw std::runtime_error("Error loading module mpl_toolkits.mplot3d!"); } + } + + assert(x.size() == y.size()); + assert(y.size() == z.size()); + + // using numpy arrays + PyObject *xarray = detail::get_2darray(x); + PyObject *yarray = detail::get_2darray(y); + PyObject *zarray = detail::get_2darray(z); + + // construct positional args + PyObject *args = PyTuple_New(3); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, yarray); + PyTuple_SetItem(args, 2, zarray); + + // Build up the kw args. + PyObject *kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "rstride", PyInt_FromLong(1)); + PyDict_SetItemString(kwargs, "cstride", PyInt_FromLong(1)); + + PyObject *python_colormap_coolwarm = PyObject_GetAttrString( + detail::_interpreter::get().s_python_colormap, "coolwarm"); + + PyDict_SetItemString(kwargs, "cmap", python_colormap_coolwarm); + + for (std::map::const_iterator it = keywords.begin(); + it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), + PyString_FromString(it->second.c_str())); + } + + + PyObject *fig = + PyObject_CallObject(detail::_interpreter::get().s_python_function_figure, + detail::_interpreter::get().s_python_empty_tuple); + if (!fig) throw std::runtime_error("Call to figure() failed."); + + PyObject *gca_kwargs = PyDict_New(); + PyDict_SetItemString(gca_kwargs, "projection", PyString_FromString("3d")); + + PyObject *gca = PyObject_GetAttrString(fig, "gca"); + if (!gca) throw std::runtime_error("No gca"); + Py_INCREF(gca); + PyObject *axis = PyObject_Call( + gca, detail::_interpreter::get().s_python_empty_tuple, gca_kwargs); + + if (!axis) throw std::runtime_error("No axis"); + Py_INCREF(axis); + + Py_DECREF(gca); + Py_DECREF(gca_kwargs); + + PyObject *plot_surface = PyObject_GetAttrString(axis, "plot_surface"); + if (!plot_surface) throw std::runtime_error("No surface"); + Py_INCREF(plot_surface); + PyObject *res = PyObject_Call(plot_surface, args, kwargs); + if (!res) throw std::runtime_error("failed surface"); + Py_DECREF(plot_surface); + + Py_DECREF(axis); + Py_DECREF(args); + Py_DECREF(kwargs); + if (res) Py_DECREF(res); +} +#endif // WITHOUT_NUMPY + +template +void plot3(const std::vector &x, + const std::vector &y, + const std::vector &z, + const std::map &keywords = + std::map()) +{ + detail::_interpreter::get(); + + // Same as with plot_surface: We lazily load the modules here the first time + // this function is called because I'm not sure that we can assume "matplotlib + // installed" implies "mpl_toolkits installed" on all platforms, and we don't + // want to require it for people who don't need 3d plots. + static PyObject *mpl_toolkitsmod = nullptr, *axis3dmod = nullptr; + if (!mpl_toolkitsmod) { + detail::_interpreter::get(); + + PyObject* mpl_toolkits = PyString_FromString("mpl_toolkits"); + PyObject* axis3d = PyString_FromString("mpl_toolkits.mplot3d"); + if (!mpl_toolkits || !axis3d) { throw std::runtime_error("couldnt create string"); } + + mpl_toolkitsmod = PyImport_Import(mpl_toolkits); + Py_DECREF(mpl_toolkits); + if (!mpl_toolkitsmod) { throw std::runtime_error("Error loading module mpl_toolkits!"); } + + axis3dmod = PyImport_Import(axis3d); + Py_DECREF(axis3d); + if (!axis3dmod) { throw std::runtime_error("Error loading module mpl_toolkits.mplot3d!"); } + } + + assert(x.size() == y.size()); + assert(y.size() == z.size()); + + PyObject *xarray = detail::get_array(x); + PyObject *yarray = detail::get_array(y); + PyObject *zarray = detail::get_array(z); + + // construct positional args + PyObject *args = PyTuple_New(3); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, yarray); + PyTuple_SetItem(args, 2, zarray); + + // Build up the kw args. + PyObject *kwargs = PyDict_New(); + + for (std::map::const_iterator it = keywords.begin(); + it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), + PyString_FromString(it->second.c_str())); + } + + PyObject *fig = + PyObject_CallObject(detail::_interpreter::get().s_python_function_figure, + detail::_interpreter::get().s_python_empty_tuple); + if (!fig) throw std::runtime_error("Call to figure() failed."); + + PyObject *gca_kwargs = PyDict_New(); + PyDict_SetItemString(gca_kwargs, "projection", PyString_FromString("3d")); + + PyObject *gca = PyObject_GetAttrString(fig, "gca"); + if (!gca) throw std::runtime_error("No gca"); + Py_INCREF(gca); + PyObject *axis = PyObject_Call( + gca, detail::_interpreter::get().s_python_empty_tuple, gca_kwargs); + + if (!axis) throw std::runtime_error("No axis"); + Py_INCREF(axis); + + Py_DECREF(gca); + Py_DECREF(gca_kwargs); + + PyObject *plot3 = PyObject_GetAttrString(axis, "plot"); + if (!plot3) throw std::runtime_error("No 3D line plot"); + Py_INCREF(plot3); + PyObject *res = PyObject_Call(plot3, args, kwargs); + if (!res) throw std::runtime_error("Failed 3D line plot"); + Py_DECREF(plot3); + + Py_DECREF(axis); + Py_DECREF(args); + Py_DECREF(kwargs); + if (res) Py_DECREF(res); +} + +template +bool stem(const std::vector &x, const std::vector &y, const std::map& keywords) +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + // using numpy arrays + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + // construct positional args + PyObject* args = PyTuple_New(2); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, yarray); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for (std::map::const_iterator it = + keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), + PyString_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call( + detail::_interpreter::get().s_python_function_stem, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if (res) + Py_DECREF(res); + + return res; +} + +template< typename Numeric > +bool fill(const std::vector& x, const std::vector& y, const std::map& keywords) +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + // using numpy arrays + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + // construct positional args + PyObject* args = PyTuple_New(2); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, yarray); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_fill, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + + if (res) Py_DECREF(res); + + return res; +} + +template< typename Numeric > +bool fill_between(const std::vector& x, const std::vector& y1, const std::vector& y2, const std::map& keywords) +{ + assert(x.size() == y1.size()); + assert(x.size() == y2.size()); + + detail::_interpreter::get(); + + // using numpy arrays + PyObject* xarray = detail::get_array(x); + PyObject* y1array = detail::get_array(y1); + PyObject* y2array = detail::get_array(y2); + + // construct positional args + PyObject* args = PyTuple_New(3); + PyTuple_SetItem(args, 0, xarray); + PyTuple_SetItem(args, 1, y1array); + PyTuple_SetItem(args, 2, y2array); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_fill_between, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +template< typename Numeric> +bool hist(const std::vector& y, long bins=10,std::string color="b", + double alpha=1.0, bool cumulative=false) +{ + detail::_interpreter::get(); + + PyObject* yarray = detail::get_array(y); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "bins", PyLong_FromLong(bins)); + PyDict_SetItemString(kwargs, "color", PyString_FromString(color.c_str())); + PyDict_SetItemString(kwargs, "alpha", PyFloat_FromDouble(alpha)); + PyDict_SetItemString(kwargs, "cumulative", cumulative ? Py_True : Py_False); + + PyObject* plot_args = PyTuple_New(1); + + PyTuple_SetItem(plot_args, 0, yarray); + + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_hist, plot_args, kwargs); + + + Py_DECREF(plot_args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +#ifndef WITHOUT_NUMPY +namespace detail { + +inline void imshow(void *ptr, const NPY_TYPES type, const int rows, const int columns, const int colors, const std::map &keywords, PyObject** out) +{ + assert(type == NPY_UINT8 || type == NPY_FLOAT); + assert(colors == 1 || colors == 3 || colors == 4); + + detail::_interpreter::get(); + + // construct args + npy_intp dims[3] = { rows, columns, colors }; + PyObject *args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyArray_SimpleNewFromData(colors == 1 ? 2 : 3, dims, type, ptr)); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject *res = PyObject_Call(detail::_interpreter::get().s_python_function_imshow, args, kwargs); + Py_DECREF(args); + Py_DECREF(kwargs); + if (!res) + throw std::runtime_error("Call to imshow() failed"); + if (out) + *out = res; + else + Py_DECREF(res); +} + +} // namespace detail + +inline void imshow(const unsigned char *ptr, const int rows, const int columns, const int colors, const std::map &keywords = {}, PyObject** out = nullptr) +{ + detail::imshow((void *) ptr, NPY_UINT8, rows, columns, colors, keywords, out); +} + +inline void imshow(const float *ptr, const int rows, const int columns, const int colors, const std::map &keywords = {}, PyObject** out = nullptr) +{ + detail::imshow((void *) ptr, NPY_FLOAT, rows, columns, colors, keywords, out); +} + +#ifdef WITH_OPENCV +void imshow(const cv::Mat &image, const std::map &keywords = {}) +{ + // Convert underlying type of matrix, if needed + cv::Mat image2; + NPY_TYPES npy_type = NPY_UINT8; + switch (image.type() & CV_MAT_DEPTH_MASK) { + case CV_8U: + image2 = image; + break; + case CV_32F: + image2 = image; + npy_type = NPY_FLOAT; + break; + default: + image.convertTo(image2, CV_MAKETYPE(CV_8U, image.channels())); + } + + // If color image, convert from BGR to RGB + switch (image2.channels()) { + case 3: + cv::cvtColor(image2, image2, CV_BGR2RGB); + break; + case 4: + cv::cvtColor(image2, image2, CV_BGRA2RGBA); + } + + detail::imshow(image2.data, npy_type, image2.rows, image2.cols, image2.channels(), keywords); +} +#endif // WITH_OPENCV +#endif // WITHOUT_NUMPY + +template +bool scatter(const std::vector& x, + const std::vector& y, + const double s=1.0, // The marker size in points**2 + const std::map & keywords = {}) +{ + detail::_interpreter::get(); + + assert(x.size() == y.size()); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "s", PyLong_FromLong(s)); + for (const auto& it : keywords) + { + PyDict_SetItemString(kwargs, it.first.c_str(), PyString_FromString(it.second.c_str())); + } + + PyObject* plot_args = PyTuple_New(2); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_scatter, plot_args, kwargs); + + Py_DECREF(plot_args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +template +bool boxplot(const std::vector>& data, + const std::vector& labels = {}, + const std::map & keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* listlist = detail::get_listlist(data); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, listlist); + + PyObject* kwargs = PyDict_New(); + + // kwargs needs the labels, if there are (the correct number of) labels + if (!labels.empty() && labels.size() == data.size()) { + PyDict_SetItemString(kwargs, "labels", detail::get_array(labels)); + } + + // take care of the remaining keywords + for (const auto& it : keywords) + { + PyDict_SetItemString(kwargs, it.first.c_str(), PyString_FromString(it.second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_boxplot, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + + if(res) Py_DECREF(res); + + return res; +} + +template +bool boxplot(const std::vector& data, + const std::map & keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* vector = detail::get_array(data); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, vector); + + PyObject* kwargs = PyDict_New(); + for (const auto& it : keywords) + { + PyDict_SetItemString(kwargs, it.first.c_str(), PyString_FromString(it.second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_boxplot, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + + if(res) Py_DECREF(res); + + return res; +} + +template +bool bar(const std::vector & x, + const std::vector & y, + std::string ec = "black", + std::string ls = "-", + double lw = 1.0, + const std::map & keywords = {}) +{ + detail::_interpreter::get(); + + PyObject * xarray = detail::get_array(x); + PyObject * yarray = detail::get_array(y); + + PyObject * kwargs = PyDict_New(); + + PyDict_SetItemString(kwargs, "ec", PyString_FromString(ec.c_str())); + PyDict_SetItemString(kwargs, "ls", PyString_FromString(ls.c_str())); + PyDict_SetItemString(kwargs, "lw", PyFloat_FromDouble(lw)); + + for (std::map::const_iterator it = + keywords.begin(); + it != keywords.end(); + ++it) { + PyDict_SetItemString( + kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject * plot_args = PyTuple_New(2); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + + PyObject * res = PyObject_Call( + detail::_interpreter::get().s_python_function_bar, plot_args, kwargs); + + Py_DECREF(plot_args); + Py_DECREF(kwargs); + if (res) Py_DECREF(res); + + return res; +} + +template +bool bar(const std::vector & y, + std::string ec = "black", + std::string ls = "-", + double lw = 1.0, + const std::map & keywords = {}) +{ + using T = typename std::remove_reference::type::value_type; + + detail::_interpreter::get(); + + std::vector x; + for (std::size_t i = 0; i < y.size(); i++) { x.push_back(i); } + + return bar(x, y, ec, ls, lw, keywords); +} + +inline bool subplots_adjust(const std::map& keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + for (std::map::const_iterator it = + keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), + PyFloat_FromDouble(it->second)); + } + + + PyObject* plot_args = PyTuple_New(0); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_subplots_adjust, plot_args, kwargs); + + Py_DECREF(plot_args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +template< typename Numeric> +bool named_hist(std::string label,const std::vector& y, long bins=10, std::string color="b", double alpha=1.0) +{ + detail::_interpreter::get(); + + PyObject* yarray = detail::get_array(y); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(label.c_str())); + PyDict_SetItemString(kwargs, "bins", PyLong_FromLong(bins)); + PyDict_SetItemString(kwargs, "color", PyString_FromString(color.c_str())); + PyDict_SetItemString(kwargs, "alpha", PyFloat_FromDouble(alpha)); + + + PyObject* plot_args = PyTuple_New(1); + PyTuple_SetItem(plot_args, 0, yarray); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_hist, plot_args, kwargs); + + Py_DECREF(plot_args); + Py_DECREF(kwargs); + if(res) Py_DECREF(res); + + return res; +} + +template +bool plot(const std::vector& x, const std::vector& y, const std::string& s = "") +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(s.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_plot, plot_args); + + Py_DECREF(plot_args); + if(res) Py_DECREF(res); + + return res; +} + +template +bool quiver(const std::vector& x, const std::vector& y, const std::vector& u, const std::vector& w, const std::map& keywords = {}) +{ + assert(x.size() == y.size() && x.size() == u.size() && u.size() == w.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + PyObject* uarray = detail::get_array(u); + PyObject* warray = detail::get_array(w); + + PyObject* plot_args = PyTuple_New(4); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, uarray); + PyTuple_SetItem(plot_args, 3, warray); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call( + detail::_interpreter::get().s_python_function_quiver, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) + Py_DECREF(res); + + return res; +} + +template +bool stem(const std::vector& x, const std::vector& y, const std::string& s = "") +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(s.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_stem, plot_args); + + Py_DECREF(plot_args); + if (res) + Py_DECREF(res); + + return res; +} + +template +bool semilogx(const std::vector& x, const std::vector& y, const std::string& s = "") +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(s.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_semilogx, plot_args); + + Py_DECREF(plot_args); + if(res) Py_DECREF(res); + + return res; +} + +template +bool semilogy(const std::vector& x, const std::vector& y, const std::string& s = "") +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(s.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_semilogy, plot_args); + + Py_DECREF(plot_args); + if(res) Py_DECREF(res); + + return res; +} + +template +bool loglog(const std::vector& x, const std::vector& y, const std::string& s = "") +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(s.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_loglog, plot_args); + + Py_DECREF(plot_args); + if(res) Py_DECREF(res); + + return res; +} + +template +bool errorbar(const std::vector &x, const std::vector &y, const std::vector &yerr, const std::map &keywords = {}) +{ + assert(x.size() == y.size()); + + detail::_interpreter::get(); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + PyObject* yerrarray = detail::get_array(yerr); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + PyDict_SetItemString(kwargs, "yerr", yerrarray); + + PyObject *plot_args = PyTuple_New(2); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + + PyObject *res = PyObject_Call(detail::_interpreter::get().s_python_function_errorbar, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + + if (res) + Py_DECREF(res); + else + throw std::runtime_error("Call to errorbar() failed."); + + return res; +} + +template +bool named_plot(const std::string& name, const std::vector& y, const std::string& format = "") +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(2); + + PyTuple_SetItem(plot_args, 0, yarray); + PyTuple_SetItem(plot_args, 1, pystring); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_plot, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) Py_DECREF(res); + + return res; +} + +template +bool named_plot(const std::string& name, const std::vector& x, const std::vector& y, const std::string& format = "") +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_plot, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) Py_DECREF(res); + + return res; +} + +template +bool named_semilogx(const std::string& name, const std::vector& x, const std::vector& y, const std::string& format = "") +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_semilogx, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) Py_DECREF(res); + + return res; +} + +template +bool named_semilogy(const std::string& name, const std::vector& x, const std::vector& y, const std::string& format = "") +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_semilogy, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) Py_DECREF(res); + + return res; +} + +template +bool named_loglog(const std::string& name, const std::vector& x, const std::vector& y, const std::string& format = "") +{ + detail::_interpreter::get(); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_loglog, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + if (res) Py_DECREF(res); + + return res; +} + +template +bool plot(const std::vector& y, const std::string& format = "") +{ + std::vector x(y.size()); + for(size_t i=0; i +bool plot(const std::vector& y, const std::map& keywords) +{ + std::vector x(y.size()); + for(size_t i=0; i +bool stem(const std::vector& y, const std::string& format = "") +{ + std::vector x(y.size()); + for (size_t i = 0; i < x.size(); ++i) x.at(i) = i; + return stem(x, y, format); +} + +template +void text(Numeric x, Numeric y, const std::string& s = "") +{ + detail::_interpreter::get(); + + PyObject* args = PyTuple_New(3); + PyTuple_SetItem(args, 0, PyFloat_FromDouble(x)); + PyTuple_SetItem(args, 1, PyFloat_FromDouble(y)); + PyTuple_SetItem(args, 2, PyString_FromString(s.c_str())); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_text, args); + if(!res) throw std::runtime_error("Call to text() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void colorbar(PyObject* mappable = NULL, const std::map& keywords = {}) +{ + if (mappable == NULL) + throw std::runtime_error("Must call colorbar with PyObject* returned from an image, contour, surface, etc."); + + detail::_interpreter::get(); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, mappable); + + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyFloat_FromDouble(it->second)); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_colorbar, args, kwargs); + if(!res) throw std::runtime_error("Call to colorbar() failed."); + + Py_DECREF(args); + Py_DECREF(kwargs); + Py_DECREF(res); +} + + +inline long figure(long number = -1) +{ + detail::_interpreter::get(); + + PyObject *res; + if (number == -1) + res = PyObject_CallObject(detail::_interpreter::get().s_python_function_figure, detail::_interpreter::get().s_python_empty_tuple); + else { + assert(number > 0); + + // Make sure interpreter is initialised + detail::_interpreter::get(); + + PyObject *args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyLong_FromLong(number)); + res = PyObject_CallObject(detail::_interpreter::get().s_python_function_figure, args); + Py_DECREF(args); + } + + if(!res) throw std::runtime_error("Call to figure() failed."); + + PyObject* num = PyObject_GetAttrString(res, "number"); + if (!num) throw std::runtime_error("Could not get number attribute of figure object"); + const long figureNumber = PyLong_AsLong(num); + + Py_DECREF(num); + Py_DECREF(res); + + return figureNumber; +} + +inline bool fignum_exists(long number) +{ + detail::_interpreter::get(); + + PyObject *args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyLong_FromLong(number)); + PyObject *res = PyObject_CallObject(detail::_interpreter::get().s_python_function_fignum_exists, args); + if(!res) throw std::runtime_error("Call to fignum_exists() failed."); + + bool ret = PyObject_IsTrue(res); + Py_DECREF(res); + Py_DECREF(args); + + return ret; +} + +inline void figure_size(size_t w, size_t h) +{ + detail::_interpreter::get(); + + const size_t dpi = 100; + PyObject* size = PyTuple_New(2); + PyTuple_SetItem(size, 0, PyFloat_FromDouble((double)w / dpi)); + PyTuple_SetItem(size, 1, PyFloat_FromDouble((double)h / dpi)); + + PyObject* kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "figsize", size); + PyDict_SetItemString(kwargs, "dpi", PyLong_FromSize_t(dpi)); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_figure, + detail::_interpreter::get().s_python_empty_tuple, kwargs); + + Py_DECREF(kwargs); + + if(!res) throw std::runtime_error("Call to figure_size() failed."); + Py_DECREF(res); +} + +inline void legend() +{ + detail::_interpreter::get(); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_legend, detail::_interpreter::get().s_python_empty_tuple); + if(!res) throw std::runtime_error("Call to legend() failed."); + + Py_DECREF(res); +} + +template +void ylim(Numeric left, Numeric right) +{ + detail::_interpreter::get(); + + PyObject* list = PyList_New(2); + PyList_SetItem(list, 0, PyFloat_FromDouble(left)); + PyList_SetItem(list, 1, PyFloat_FromDouble(right)); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, list); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_ylim, args); + if(!res) throw std::runtime_error("Call to ylim() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +template +void xlim(Numeric left, Numeric right) +{ + detail::_interpreter::get(); + + PyObject* list = PyList_New(2); + PyList_SetItem(list, 0, PyFloat_FromDouble(left)); + PyList_SetItem(list, 1, PyFloat_FromDouble(right)); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, list); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_xlim, args); + if(!res) throw std::runtime_error("Call to xlim() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + + +inline double* xlim() +{ + detail::_interpreter::get(); + + PyObject* args = PyTuple_New(0); + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_xlim, args); + PyObject* left = PyTuple_GetItem(res,0); + PyObject* right = PyTuple_GetItem(res,1); + + double* arr = new double[2]; + arr[0] = PyFloat_AsDouble(left); + arr[1] = PyFloat_AsDouble(right); + + if(!res) throw std::runtime_error("Call to xlim() failed."); + + Py_DECREF(res); + return arr; +} + + +inline double* ylim() +{ + detail::_interpreter::get(); + + PyObject* args = PyTuple_New(0); + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_ylim, args); + PyObject* left = PyTuple_GetItem(res,0); + PyObject* right = PyTuple_GetItem(res,1); + + double* arr = new double[2]; + arr[0] = PyFloat_AsDouble(left); + arr[1] = PyFloat_AsDouble(right); + + if(!res) throw std::runtime_error("Call to ylim() failed."); + + Py_DECREF(res); + return arr; +} + +template +inline void xticks(const std::vector &ticks, const std::vector &labels = {}, const std::map& keywords = {}) +{ + assert(labels.size() == 0 || ticks.size() == labels.size()); + + detail::_interpreter::get(); + + // using numpy array + PyObject* ticksarray = detail::get_array(ticks); + + PyObject* args; + if(labels.size() == 0) { + // construct positional args + args = PyTuple_New(1); + PyTuple_SetItem(args, 0, ticksarray); + } else { + // make tuple of tick labels + PyObject* labelstuple = PyTuple_New(labels.size()); + for (size_t i = 0; i < labels.size(); i++) + PyTuple_SetItem(labelstuple, i, PyUnicode_FromString(labels[i].c_str())); + + // construct positional args + args = PyTuple_New(2); + PyTuple_SetItem(args, 0, ticksarray); + PyTuple_SetItem(args, 1, labelstuple); + } + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_xticks, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if(!res) throw std::runtime_error("Call to xticks() failed"); + + Py_DECREF(res); +} + +template +inline void xticks(const std::vector &ticks, const std::map& keywords) +{ + xticks(ticks, {}, keywords); +} + +template +inline void yticks(const std::vector &ticks, const std::vector &labels = {}, const std::map& keywords = {}) +{ + assert(labels.size() == 0 || ticks.size() == labels.size()); + + detail::_interpreter::get(); + + // using numpy array + PyObject* ticksarray = detail::get_array(ticks); + + PyObject* args; + if(labels.size() == 0) { + // construct positional args + args = PyTuple_New(1); + PyTuple_SetItem(args, 0, ticksarray); + } else { + // make tuple of tick labels + PyObject* labelstuple = PyTuple_New(labels.size()); + for (size_t i = 0; i < labels.size(); i++) + PyTuple_SetItem(labelstuple, i, PyUnicode_FromString(labels[i].c_str())); + + // construct positional args + args = PyTuple_New(2); + PyTuple_SetItem(args, 0, ticksarray); + PyTuple_SetItem(args, 1, labelstuple); + } + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_yticks, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if(!res) throw std::runtime_error("Call to yticks() failed"); + + Py_DECREF(res); +} + +template +inline void yticks(const std::vector &ticks, const std::map& keywords) +{ + yticks(ticks, {}, keywords); +} + +inline void tick_params(const std::map& keywords, const std::string axis = "both") +{ + detail::_interpreter::get(); + + // construct positional args + PyObject* args; + args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyString_FromString(axis.c_str())); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for (std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_tick_params, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + if (!res) throw std::runtime_error("Call to tick_params() failed"); + + Py_DECREF(res); +} + +inline void subplot(long nrows, long ncols, long plot_number) +{ + detail::_interpreter::get(); + + // construct positional args + PyObject* args = PyTuple_New(3); + PyTuple_SetItem(args, 0, PyFloat_FromDouble(nrows)); + PyTuple_SetItem(args, 1, PyFloat_FromDouble(ncols)); + PyTuple_SetItem(args, 2, PyFloat_FromDouble(plot_number)); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_subplot, args); + if(!res) throw std::runtime_error("Call to subplot() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void subplot2grid(long nrows, long ncols, long rowid=0, long colid=0, long rowspan=1, long colspan=1) +{ + detail::_interpreter::get(); + + PyObject* shape = PyTuple_New(2); + PyTuple_SetItem(shape, 0, PyLong_FromLong(nrows)); + PyTuple_SetItem(shape, 1, PyLong_FromLong(ncols)); + + PyObject* loc = PyTuple_New(2); + PyTuple_SetItem(loc, 0, PyLong_FromLong(rowid)); + PyTuple_SetItem(loc, 1, PyLong_FromLong(colid)); + + PyObject* args = PyTuple_New(4); + PyTuple_SetItem(args, 0, shape); + PyTuple_SetItem(args, 1, loc); + PyTuple_SetItem(args, 2, PyLong_FromLong(rowspan)); + PyTuple_SetItem(args, 3, PyLong_FromLong(colspan)); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_subplot2grid, args); + if(!res) throw std::runtime_error("Call to subplot2grid() failed."); + + Py_DECREF(shape); + Py_DECREF(loc); + Py_DECREF(args); + Py_DECREF(res); +} + +inline void title(const std::string &titlestr, const std::map &keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* pytitlestr = PyString_FromString(titlestr.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pytitlestr); + + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_title, args, kwargs); + if(!res) throw std::runtime_error("Call to title() failed."); + + Py_DECREF(args); + Py_DECREF(kwargs); + Py_DECREF(res); +} + +inline void suptitle(const std::string &suptitlestr, const std::map &keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* pysuptitlestr = PyString_FromString(suptitlestr.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pysuptitlestr); + + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_suptitle, args, kwargs); + if(!res) throw std::runtime_error("Call to suptitle() failed."); + + Py_DECREF(args); + Py_DECREF(kwargs); + Py_DECREF(res); +} + +inline void axis(const std::string &axisstr) +{ + detail::_interpreter::get(); + + PyObject* str = PyString_FromString(axisstr.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, str); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_axis, args); + if(!res) throw std::runtime_error("Call to title() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void axvline(double x, double ymin = 0., double ymax = 1., const std::map& keywords_string = std::map(), + const std::map& keywords_double = std::map()) +{ + detail::_interpreter::get(); + + // construct positional args + PyObject* args = PyTuple_New(3); + PyTuple_SetItem(args, 0, PyFloat_FromDouble(x)); + PyTuple_SetItem(args, 1, PyFloat_FromDouble(ymin)); + PyTuple_SetItem(args, 2, PyFloat_FromDouble(ymax)); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords_string.begin(); it != keywords_string.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyString_FromString(it->second.c_str())); + } + + for(std::map::const_iterator it = keywords_double.begin(); it != keywords_double.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyFloat_FromDouble(it->second)); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_axvline, args, kwargs); + + Py_DECREF(args); + Py_DECREF(kwargs); + + if(res) Py_DECREF(res); +} + +inline void xlabel(const std::string &str, const std::map &keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* pystr = PyString_FromString(str.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pystr); + + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_xlabel, args, kwargs); + if(!res) throw std::runtime_error("Call to xlabel() failed."); + + Py_DECREF(args); + Py_DECREF(kwargs); + Py_DECREF(res); +} + +inline void ylabel(const std::string &str, const std::map& keywords = {}) +{ + detail::_interpreter::get(); + + PyObject* pystr = PyString_FromString(str.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pystr); + + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_ylabel, args, kwargs); + if(!res) throw std::runtime_error("Call to ylabel() failed."); + + Py_DECREF(args); + Py_DECREF(kwargs); + Py_DECREF(res); +} + +inline void set_zlabel(const std::string &str, const std::map& keywords = {}) +{ + detail::_interpreter::get(); + + // Same as with plot_surface: We lazily load the modules here the first time + // this function is called because I'm not sure that we can assume "matplotlib + // installed" implies "mpl_toolkits installed" on all platforms, and we don't + // want to require it for people who don't need 3d plots. + static PyObject *mpl_toolkitsmod = nullptr, *axis3dmod = nullptr; + if (!mpl_toolkitsmod) { + PyObject* mpl_toolkits = PyString_FromString("mpl_toolkits"); + PyObject* axis3d = PyString_FromString("mpl_toolkits.mplot3d"); + if (!mpl_toolkits || !axis3d) { throw std::runtime_error("couldnt create string"); } + + mpl_toolkitsmod = PyImport_Import(mpl_toolkits); + Py_DECREF(mpl_toolkits); + if (!mpl_toolkitsmod) { throw std::runtime_error("Error loading module mpl_toolkits!"); } + + axis3dmod = PyImport_Import(axis3d); + Py_DECREF(axis3d); + if (!axis3dmod) { throw std::runtime_error("Error loading module mpl_toolkits.mplot3d!"); } + } + + PyObject* pystr = PyString_FromString(str.c_str()); + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pystr); + + PyObject* kwargs = PyDict_New(); + for (auto it = keywords.begin(); it != keywords.end(); ++it) { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject *ax = + PyObject_CallObject(detail::_interpreter::get().s_python_function_gca, + detail::_interpreter::get().s_python_empty_tuple); + if (!ax) throw std::runtime_error("Call to gca() failed."); + Py_INCREF(ax); + + PyObject *zlabel = PyObject_GetAttrString(ax, "set_zlabel"); + if (!zlabel) throw std::runtime_error("Attribute set_zlabel not found."); + Py_INCREF(zlabel); + + PyObject *res = PyObject_Call(zlabel, args, kwargs); + if (!res) throw std::runtime_error("Call to set_zlabel() failed."); + Py_DECREF(zlabel); + + Py_DECREF(ax); + Py_DECREF(args); + Py_DECREF(kwargs); + if (res) Py_DECREF(res); +} + +inline void grid(bool flag) +{ + detail::_interpreter::get(); + + PyObject* pyflag = flag ? Py_True : Py_False; + Py_INCREF(pyflag); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pyflag); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_grid, args); + if(!res) throw std::runtime_error("Call to grid() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void show(const bool block = true) +{ + detail::_interpreter::get(); + + PyObject* res; + if(block) + { + res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_show, + detail::_interpreter::get().s_python_empty_tuple); + } + else + { + PyObject *kwargs = PyDict_New(); + PyDict_SetItemString(kwargs, "block", Py_False); + res = PyObject_Call( detail::_interpreter::get().s_python_function_show, detail::_interpreter::get().s_python_empty_tuple, kwargs); + Py_DECREF(kwargs); + } + + + if (!res) throw std::runtime_error("Call to show() failed."); + + Py_DECREF(res); +} + +inline void close() +{ + detail::_interpreter::get(); + + PyObject* res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_close, + detail::_interpreter::get().s_python_empty_tuple); + + if (!res) throw std::runtime_error("Call to close() failed."); + + Py_DECREF(res); +} + +inline void xkcd() { + detail::_interpreter::get(); + + PyObject* res; + PyObject *kwargs = PyDict_New(); + + res = PyObject_Call(detail::_interpreter::get().s_python_function_xkcd, + detail::_interpreter::get().s_python_empty_tuple, kwargs); + + Py_DECREF(kwargs); + + if (!res) + throw std::runtime_error("Call to show() failed."); + + Py_DECREF(res); +} + +inline void draw() +{ + detail::_interpreter::get(); + + PyObject* res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_draw, + detail::_interpreter::get().s_python_empty_tuple); + + if (!res) throw std::runtime_error("Call to draw() failed."); + + Py_DECREF(res); +} + +template +inline void pause(Numeric interval) +{ + detail::_interpreter::get(); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyFloat_FromDouble(interval)); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_pause, args); + if(!res) throw std::runtime_error("Call to pause() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void save(const std::string& filename) +{ + detail::_interpreter::get(); + + PyObject* pyfilename = PyString_FromString(filename.c_str()); + + PyObject* args = PyTuple_New(1); + PyTuple_SetItem(args, 0, pyfilename); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_save, args); + if (!res) throw std::runtime_error("Call to save() failed."); + + Py_DECREF(args); + Py_DECREF(res); +} + +inline void clf() { + detail::_interpreter::get(); + + PyObject *res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_clf, + detail::_interpreter::get().s_python_empty_tuple); + + if (!res) throw std::runtime_error("Call to clf() failed."); + + Py_DECREF(res); +} + +inline void ion() { + detail::_interpreter::get(); + + PyObject *res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_ion, + detail::_interpreter::get().s_python_empty_tuple); + + if (!res) throw std::runtime_error("Call to ion() failed."); + + Py_DECREF(res); +} + +inline std::vector> ginput(const int numClicks = 1, const std::map& keywords = {}) +{ + detail::_interpreter::get(); + + PyObject *args = PyTuple_New(1); + PyTuple_SetItem(args, 0, PyLong_FromLong(numClicks)); + + // construct keyword args + PyObject* kwargs = PyDict_New(); + for(std::map::const_iterator it = keywords.begin(); it != keywords.end(); ++it) + { + PyDict_SetItemString(kwargs, it->first.c_str(), PyUnicode_FromString(it->second.c_str())); + } + + PyObject* res = PyObject_Call( + detail::_interpreter::get().s_python_function_ginput, args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(args); + if (!res) throw std::runtime_error("Call to ginput() failed."); + + const size_t len = PyList_Size(res); + std::vector> out; + out.reserve(len); + for (size_t i = 0; i < len; i++) { + PyObject *current = PyList_GetItem(res, i); + std::array position; + position[0] = PyFloat_AsDouble(PyTuple_GetItem(current, 0)); + position[1] = PyFloat_AsDouble(PyTuple_GetItem(current, 1)); + out.push_back(position); + } + Py_DECREF(res); + + return out; +} + +// Actually, is there any reason not to call this automatically for every plot? +inline void tight_layout() { + detail::_interpreter::get(); + + PyObject *res = PyObject_CallObject( + detail::_interpreter::get().s_python_function_tight_layout, + detail::_interpreter::get().s_python_empty_tuple); + + if (!res) throw std::runtime_error("Call to tight_layout() failed."); + + Py_DECREF(res); +} + +// Support for variadic plot() and initializer lists: + +namespace detail { + +template +using is_function = typename std::is_function>>::type; + +template +struct is_callable_impl; + +template +struct is_callable_impl +{ + typedef is_function type; +}; // a non-object is callable iff it is a function + +template +struct is_callable_impl +{ + struct Fallback { void operator()(); }; + struct Derived : T, Fallback { }; + + template struct Check; + + template + static std::true_type test( ... ); // use a variadic function to make sure (1) it accepts everything and (2) its always the worst match + + template + static std::false_type test( Check* ); + +public: + typedef decltype(test(nullptr)) type; + typedef decltype(&Fallback::operator()) dtype; + static constexpr bool value = type::value; +}; // an object is callable iff it defines operator() + +template +struct is_callable +{ + // dispatch to is_callable_impl or is_callable_impl depending on whether T is of class type or not + typedef typename is_callable_impl::value, T>::type type; +}; + +template +struct plot_impl { }; + +template<> +struct plot_impl +{ + template + bool operator()(const IterableX& x, const IterableY& y, const std::string& format) + { + // 2-phase lookup for distance, begin, end + using std::distance; + using std::begin; + using std::end; + + auto xs = distance(begin(x), end(x)); + auto ys = distance(begin(y), end(y)); + assert(xs == ys && "x and y data must have the same number of elements!"); + + PyObject* xlist = PyList_New(xs); + PyObject* ylist = PyList_New(ys); + PyObject* pystring = PyString_FromString(format.c_str()); + + auto itx = begin(x), ity = begin(y); + for(size_t i = 0; i < xs; ++i) { + PyList_SetItem(xlist, i, PyFloat_FromDouble(*itx++)); + PyList_SetItem(ylist, i, PyFloat_FromDouble(*ity++)); + } + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xlist); + PyTuple_SetItem(plot_args, 1, ylist); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_CallObject(detail::_interpreter::get().s_python_function_plot, plot_args); + + Py_DECREF(plot_args); + if(res) Py_DECREF(res); + + return res; + } +}; + +template<> +struct plot_impl +{ + template + bool operator()(const Iterable& ticks, const Callable& f, const std::string& format) + { + if(begin(ticks) == end(ticks)) return true; + + // We could use additional meta-programming to deduce the correct element type of y, + // but all values have to be convertible to double anyways + std::vector y; + for(auto x : ticks) y.push_back(f(x)); + return plot_impl()(ticks,y,format); + } +}; + +} // end namespace detail + +// recursion stop for the above +template +bool plot() { return true; } + +template +bool plot(const A& a, const B& b, const std::string& format, Args... args) +{ + return detail::plot_impl::type>()(a,b,format) && plot(args...); +} + +/* + * This group of plot() functions is needed to support initializer lists, i.e. calling + * plot( {1,2,3,4} ) + */ +inline bool plot(const std::vector& x, const std::vector& y, const std::string& format = "") { + return plot(x,y,format); +} + +inline bool plot(const std::vector& y, const std::string& format = "") { + return plot(y,format); +} + +inline bool plot(const std::vector& x, const std::vector& y, const std::map& keywords) { + return plot(x,y,keywords); +} + +/* + * This class allows dynamic plots, ie changing the plotted data without clearing and re-plotting + */ +class Plot +{ +public: + // default initialization with plot label, some data and format + template + Plot(const std::string& name, const std::vector& x, const std::vector& y, const std::string& format = "") { + detail::_interpreter::get(); + + assert(x.size() == y.size()); + + PyObject* kwargs = PyDict_New(); + if(name != "") + PyDict_SetItemString(kwargs, "label", PyString_FromString(name.c_str())); + + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* pystring = PyString_FromString(format.c_str()); + + PyObject* plot_args = PyTuple_New(3); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + PyTuple_SetItem(plot_args, 2, pystring); + + PyObject* res = PyObject_Call(detail::_interpreter::get().s_python_function_plot, plot_args, kwargs); + + Py_DECREF(kwargs); + Py_DECREF(plot_args); + + if(res) + { + line= PyList_GetItem(res, 0); + + if(line) + set_data_fct = PyObject_GetAttrString(line,"set_data"); + else + Py_DECREF(line); + Py_DECREF(res); + } + } + + // shorter initialization with name or format only + // basically calls line, = plot([], []) + Plot(const std::string& name = "", const std::string& format = "") + : Plot(name, std::vector(), std::vector(), format) {} + + template + bool update(const std::vector& x, const std::vector& y) { + assert(x.size() == y.size()); + if(set_data_fct) + { + PyObject* xarray = detail::get_array(x); + PyObject* yarray = detail::get_array(y); + + PyObject* plot_args = PyTuple_New(2); + PyTuple_SetItem(plot_args, 0, xarray); + PyTuple_SetItem(plot_args, 1, yarray); + + PyObject* res = PyObject_CallObject(set_data_fct, plot_args); + if (res) Py_DECREF(res); + return res; + } + return false; + } + + // clears the plot but keep it available + bool clear() { + return update(std::vector(), std::vector()); + } + + // definitely remove this line + void remove() { + if(line) + { + auto remove_fct = PyObject_GetAttrString(line,"remove"); + PyObject* args = PyTuple_New(0); + PyObject* res = PyObject_CallObject(remove_fct, args); + if (res) Py_DECREF(res); + } + decref(); + } + + ~Plot() { + decref(); + } +private: + + void decref() { + if(line) + Py_DECREF(line); + if(set_data_fct) + Py_DECREF(set_data_fct); + } + + + PyObject* line = nullptr; + PyObject* set_data_fct = nullptr; +}; + +} // end namespace matplotlibcpp diff --git a/src/volesti/include/sos/plot_saved.png b/src/volesti/include/sos/plot_saved.png new file mode 100644 index 0000000000000000000000000000000000000000..4bf495f659687be7af402820c5598b7192bace40 GIT binary patch literal 132753 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 KYuQ&UAN(I|K4R(s literal 0 HcmV?d00001 diff --git a/src/volesti/include/sos/utils.cpp b/src/volesti/include/sos/utils.cpp new file mode 100644 index 00000000..bb1265e7 --- /dev/null +++ b/src/volesti/include/sos/utils.cpp @@ -0,0 +1,6 @@ +#include "utils.h" + +std::string getEnvVar(std::string const &key) { + char *val = getenv(key.c_str()); + return val == NULL ? std::string("") : std::string(val); +} \ No newline at end of file diff --git a/src/volesti/include/sos/utils.h b/src/volesti/include/sos/utils.h new file mode 100644 index 00000000..09285ec9 --- /dev/null +++ b/src/volesti/include/sos/utils.h @@ -0,0 +1,331 @@ +// VolEsti (volume computation and sampling library) +// +// Copyright (c) 2020 Bento Natura +// +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SOS_UTILS_H +#define SOS_UTILS_H + +//Preprocessor directive allows us to forbid Eigen to allocate memory. Temporariliy helps to debug where allocation might slow down the program. +#define EIGEN_RUNTIME_NO_MALLOC + +//Note MKL Macro is set in CmakeLists file. + +//#define EIGEN_USE_MKL_ALL +//#define EIGEN_USE_BLAS +//#define EIGEN_USE_LAPACKE + +#include +#include + +#include + +#include +#include +#include +#include "spdlog/spdlog.h" +#include +#include +#include + +#include "ChebTools/ChebTools.h" + + +#ifndef DIGITS_PRECISION +#define DIGITS_PRECISION 50 +#endif + +typedef boost::multiprecision::cpp_dec_float mp_backend; +typedef boost::multiprecision::number BoostDouble; +//typedef boost::multiprecision::number > BoostDouble; +typedef BoostDouble InterpolantDouble; + +typedef long double long_double; + +#ifdef IPM_DOUBLE +typedef IPM_DOUBLE Double; +#else +typedef double Double; +#endif + +//Change typedef here to use different double type in interior point method. +#ifdef IPM_USE_DOUBLE +//typedef Double IPMDouble; +#else +typedef BoostDouble IPMDouble; +#endif + +typedef Eigen::Matrix BoostMatrix; +typedef Eigen::Matrix BoostVector; + +typedef Eigen::Matrix InterpolantMatrix; +typedef Eigen::Matrix InterpolantVector; + +typedef Eigen::Matrix DoubleMatrix; +typedef Eigen::Matrix DoubleVector; + +//Note: Boost Dependency +namespace pt = boost::property_tree; + +template +using Matrix = Eigen::Matrix; + +template +using Vector = Eigen::Matrix; + +//Stack the columns of a square m x m matrix to a vector of length m x m. +template +Vector StackMatrixToVector(Matrix M) { + assert(M.rows() == M.cols()); + Eigen::Map> x(M.data(), M.rows() * M.cols(), 1); + return x; +} + +//Unstack vector +template +Matrix UnstackVectorToMatrix(Vector v, unsigned matrix_dimension) { + assert(v.rows() == matrix_dimension * matrix_dimension); + Eigen::Map > M(v.data(), matrix_dimension, matrix_dimension); + return M; +} + +//template +//class Constraints; + +template +class Solution { +public: + template + Solution cast() { + Solution sol; + sol.x = x.template cast(); + sol.s = s.template cast(); + //Note: Boost Dependency. + sol.centrality = boost::numeric_cast(centrality); + sol.gap= boost::numeric_cast(gap); + return sol; + } + + Vector x; + Vector s; + T centrality; + T gap; +}; + +//TODO: Need full row rank matrices for IPM. Also, is preprocessing A, e.g. row-echelon form useful? +template +class Constraints { +public: + Matrix A; + Vector b; + Vector c; + + Constraints() {}; + + Constraints(Matrix A_, Vector b_, Vector c_) : A(A_), b(b_), c(c_) {}; + + void print() { + std::cout << "Constraints are as follows. Constraint matrix is A: " << std::endl; + std::cout << A << std::endl; + std::cout << "Objective c is " << std::endl; + std::cout << c.transpose() << std::endl; + std::cout << "RHS b is " << std::endl; + std::cout << b.transpose() << std::endl; + } + + //TODO: use optional argument to indicate sparseness. + Constraints dual_system() { + std::cout << "Create dual system... " << std::endl; + Constraints dual_constraints; +// dual_constraints.c = A.colPivHouseholderQr().solve(b); + + //TODO: use proper tolerance / reference. + + Eigen::SparseMatrix A_top_sparse = A.transpose().sparseView(0, 1e-10); + Eigen::SparseMatrix A_sparse = A.sparseView(0, 1e-10); + + A_top_sparse.makeCompressed(); + A_sparse.makeCompressed(); + + Eigen::SparseQR, Eigen::COLAMDOrdering > QR_top_sparse; + Eigen::SparseQR, Eigen::COLAMDOrdering > QR_sparse; + + QR_top_sparse.compute(A_top_sparse); + QR_sparse.compute(A_sparse); + + dual_constraints.c = QR_sparse.solve(b); + + Matrix QR_from_sparse(QR_top_sparse.matrixQ()); +// Matrix QR = A.transpose().householderQr().householderQ(); + + dual_constraints.A = QR_from_sparse.block(0, A.rows(), QR_from_sparse.rows(), + QR_from_sparse.cols() - A.rows()).transpose(); + dual_constraints.b = dual_constraints.A * c; + + std::cout << "Done." << std::endl; + + //TODO: use different measure to calculate centrality error + assert((dual_constraints.A * A.transpose()).norm() < 10e-5); + return dual_constraints; + } +}; + +//Naive implementation +class DegreeTuple { +public: + DegreeTuple(const int num_vars, const unsigned max_degree_) { + max_degree = max_degree_; + v.resize(num_vars, 0); + } + + bool next() { + for (int i = v.size() - 1; i >= 0; i--) { + if (v[i] < max_degree) { + v[i]++; + return true; + } + v[i] = 0; + } + return false; + } + + bool valid() { + int sum = 0; + for (auto el : v) { + sum += el; + } + return sum <= max_degree; + } + + bool next_valid() { + if (!next()) { + return false; + } + while (not valid()) { + if (not next()) { + return false; + } + } + return true; + } + + std::vector &get_tuple() { + return v; + } + + void print_tuple() { + for (auto el : v) { + std::cout << el << " "; + } + std::cout << std::endl; + } + +private: + unsigned max_degree; + std::vector v; +}; + +class AllCombinationTuple { +public: + AllCombinationTuple(std::vector const bounds_) { + + bounds = bounds_; + v.resize(bounds.size(), 0); + } + + bool next() { + for (int i = v.size() - 1; i >= 0; i--) { + if (bounds[i] > v[i]) { + v[i]++; + return true; + } + v[i] = 0; + } + return false; + } + + std::vector &get_combination() { + return v; + } + + std::vector bounds; + std::vector v; +}; + + +//Implementation copied from https://stackoverflow.com/questions/1577475/c-sorting-and-keeping-track-of-indexes + +template +std::vector sort_indexes(const std::vector &v) { + + // initialize original index locations + std::vector idx(v.size()); + std::iota(idx.begin(), idx.end(), 0); + + // sort indexes based on comparing values in v + // using std::stable_sort instead of std::sort + // to avoid unnecessary index re-orderings + // when v contains elements of equal values + stable_sort(idx.begin(), idx.end(), + [&v](size_t i1, size_t i2) { return v[i1] < v[i2]; }); + + return idx; +} + +//Access environment variables + +std::string getEnvVar(std::string const &key); + +template +class OrthogonaPMatrixLibrary { +private: + std::map, Matrix > matrices; + + void build(int L, int U) { + Eigen::MatrixXd cheb_P = ChebTools::u_matrix_library.get(U - 1).block(0, 0, U, L); + Matrix P_tmp = cheb_P.cast(); + Matrix P_ortho = P_tmp.householderQr().householderQ(); + P_ortho.colwise().hnormalized(); + matrices[std::pair(L, U)] = P_ortho.block(0, 0, U, L); + } + +public: + /// Get the \f$\mathbf{U}\f$ matrix of degree N + const Matrix &get(int L, int U) { + auto it = matrices.find(std::pair(L, U)); + if (it != matrices.end()) { + return it->second; + } else { + build(L, U); + return matrices.find(std::pair(L, U))->second; + } + } +}; + +template +static OrthogonaPMatrixLibrary orthogonal_P_Matrix_library; + + +template class CustomLLT : public Eigen::LLT<_MatrixType, _UpLo> { +public: + + typedef _MatrixType MatrixType; + typedef typename MatrixType::Scalar Scalar; + + CustomLLT<_MatrixType, _UpLo>() : Eigen::LLT<_MatrixType, _UpLo>() {}; + + CustomLLT<_MatrixType, _UpLo>(unsigned int n) : Eigen::LLT<_MatrixType, _UpLo>(n) {}; + + void copy_and_scale(const CustomLLT<_MatrixType, _UpLo> & other, Scalar scalar){ + this->m_matrix = other.m_matrix * scalar; + this->m_l1_norm = other.m_l1_norm; + this->m_isInitialized = other.m_isInitialized; + this->m_info = other.m_info; + } +}; + +#endif //SOS_UTILS_H + + + diff --git a/src/volesti/include/volume/copulas.h b/src/volesti/include/volume/copulas.h new file mode 100644 index 00000000..11a89492 --- /dev/null +++ b/src/volesti/include/volume/copulas.h @@ -0,0 +1,244 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2018 Vissarion Fisikopoulos, Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// VolEsti is free software: you can redistribute it and/or modify it +// under the terms of the GNU Lesser General Public License as published by +// the Free Software Foundation, either version 3 of the License, or (at +// your option) any later version. +// +// VolEsti is distributed in the hope that it will be useful, but WITHOUT +// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +// for more details. +// +// See the file COPYING.LESSER for the text of the GNU Lesser General +// Public License. If you did not receive this file along with HeaDDaCHe, +// see . + +#ifndef COPULAS_H +#define COPULAS_H + + +template +std::vector > twoParHypFam(const int dim, + const int num, + const int num_slices, + const std::vector &pl1, const std::vector &pl2, + double seed = std::numeric_limits::signaling_NaN()) +{ + + int i,j,col,row; + std::vector vec1,vec2,Zs1,Zs2; + NT sum1,sum2; + std::list points; + typename std::list::iterator rpit; + std::pair< std::vector,std::vector > result; + Point p; + + Sam_Canon_Unit (dim, num, points, seed); + + std::vector > Matrix(num_slices); + std::vector > pos_Matrix(num_slices); + for (i=0; i +std::vector > hypfam_ellfam(int dim, + int num, + int num_slices, + std::vector pl, + ellipsoid G, + double seed = std::numeric_limits::signaling_NaN()) +{ + + int i,j,col,row; + std::vector vec1,vec2,Zs1,Cs; + NT sum1,sum2; + std::list points; + typename std::list::iterator rpit; + std::pair< std::vector,std::vector > result; + Point p; + + Sam_Canon_Unit (dim, num, points, seed); + + std::vector > Matrix(num_slices); + std::vector > pos_Matrix(num_slices); + for (i=0; i +#include +#include + +// From rosetta code at http://rosettacode.org/wiki/Combinations#C.2B.2B +// We made some adjustments to vectorize the output +// Compute all the N combinations from N elements +inline std::vector< std::vector > comb(int N, int K) +{ + std::string bitmask(K, 1); // K leading 1's + bitmask.resize(N, 0); // N-K trailing 0's + std::vector iter_comb(K,0); + std::vector > combs; + int count; + + // print integers and permute bitmask + do { + count = 0; + for (int i = 0; i < N; ++i) // [0..N-1] integers + { + if (bitmask[i]){ + iter_comb[count] = i; + count++; + } + } + combs.push_back(iter_comb); + } while (std::prev_permutation(bitmask.begin(), bitmask.end())); + return combs; +} + + +template +NT exact_zonotope_vol(const Polytope &ZP){ + + typedef typename Polytope::MT MT; + typedef std::vector< std::vector >::iterator IntMatIt; + typedef std::vector::iterator IntIt; + + int n = ZP.dimension(), col, k = ZP.num_of_generators(); + MT V1 = ZP.get_mat().transpose(), SubV(n,n), V(n, 2*k); + V << V1, -V1; + NT vol = 0.0; + + std::vector< std::vector > combs = comb(2*k, n); + IntMatIt iter_combs; + IntIt it; + + iter_combs = combs.begin(); + for ( ; iter_combs!=combs.end(); ++iter_combs) { + it = (*iter_combs).begin(); + col = 0; + // construct all the nxn submatrices + for ( ; it!=(*iter_combs).end(); ++it, ++col) { + SubV.col(col) = V.col(*it); + } + vol += std::abs(SubV.determinant()); + } + return vol; +} + +template +NT vol_Ali(std::vector &plane, const NT &zit, const unsigned int dim) { + + unsigned int i, J = 0, K = 0, k; + std::vector Y(dim, 0.0), X(dim, 0.0), a(dim + 1, 0.0); + + for (i = 0; i < dim; i++) { + + if (plane[i] + zit < 0) { + X[J] = plane[i] + zit; + J++; + } else { + Y[K] = plane[i] + zit; + K++; + } + } + + a[0] = 1.0; + if (J>0) { + for (i = 0; i < J; i++) { + for (k = 0; k < K; k++) { + a[k+1] = (Y[k] * a[k+1] - X[i] * a[k]) / (Y[k] - X[i]); + } + } + } + return a[K]; +} + +#endif diff --git a/src/volesti/include/volume/math_helpers.hpp b/src/volesti/include/volume/math_helpers.hpp new file mode 100644 index 00000000..11e2742b --- /dev/null +++ b/src/volesti/include/volume/math_helpers.hpp @@ -0,0 +1,46 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis +// Copyright (c) 2021 Vaibhav Thakkar + +// Contributed and/or modified by Vaibhav Thakkar, as part of Google Summer of Code 2021 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef MATH_HELPERS_HPP +#define MATH_HELPERS_HPP + +#include +#include + + +//An implementation of Welford's algorithm for mean and variance. +template +std::pair get_mean_variance(std::vector& vec) +{ + NT mean = 0; + NT M2 = 0; + NT variance = 0; + NT delta; + + unsigned int i=0; + for (auto vecit = vec.begin(); vecit!=vec.end(); vecit++, i++) + { + delta = *vecit - mean; + mean += delta / (i + 1); + M2 += delta * (*vecit - mean); + variance = M2 / (i + 1); + } + return std::pair (mean, variance); +} + + +template +static NT log_gamma_function(NT x) +{ + if (x <= NT(100)) return std::log(tgamma(x)); + return (std::log(x - NT(1)) + log_gamma_function(x - NT(1))); +} + +#endif // MATH_HELPERS_HPP \ No newline at end of file diff --git a/src/volesti/include/volume/rotating.hpp b/src/volesti/include/volume/rotating.hpp new file mode 100644 index 00000000..64c2198b --- /dev/null +++ b/src/volesti/include/volume/rotating.hpp @@ -0,0 +1,64 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2019 Vissarion Fisikopoulos +// Copyright (c) 2019 Apostolos Chalkis + + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef ROTATING_H +#define ROTATING_H + +#include + +template +MT rotating(Polytope &P){ + + typedef boost::mt19937 RNGType; + unsigned rng_seed = std::chrono::system_clock::now().time_since_epoch().count(); + RNGType rng(rng_seed); + boost::random::uniform_real_distribution<> urdist(-1.0, 1.0); + unsigned int n = P.dimension(); + + // pick a random rotation + MT R(n,n); + for (int i = 0; i < n; ++i) { + for (int j = 0; j < n; ++j) { + R(i,j) = urdist(rng); + } + } + + Eigen::JacobiSVD svd(R, Eigen::ComputeFullU | Eigen::ComputeFullV); + + // apply rotation to the polytope P + P.linear_transformIt(svd.matrixU()); + + return svd.matrixU().inverse(); +} + + +template +MT rotating(Polytope &P, unsigned seed){ + + typedef boost::mt19937 RNGType; + RNGType rng(seed); + boost::random::uniform_real_distribution<> urdist(-1.0, 1.0); + unsigned int n = P.dimension(); + + // pick a random rotation + MT R(n,n); + for (int i = 0; i < n; ++i) { + for (int j = 0; j < n; ++j) { + R(i,j) = urdist(rng); + } + } + + Eigen::JacobiSVD svd(R, Eigen::ComputeFullU | Eigen::ComputeFullV); + + // apply rotation to the polytope P + P.linear_transformIt(svd.matrixU()); + + return svd.matrixU().inverse(); +} + +#endif diff --git a/src/volesti/include/volume/sampling_policies.hpp b/src/volesti/include/volume/sampling_policies.hpp new file mode 100644 index 00000000..6645fcb6 --- /dev/null +++ b/src/volesti/include/volume/sampling_policies.hpp @@ -0,0 +1,50 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef SAMPLING_POLICIES_HPP +#define SAMPLING_POLICIES_HPP + +struct PushBackWalkPolicy +{ + template + void apply(PointList &randPoints, + Point &p) const + { + randPoints.push_back(p); + } +}; + +template +struct CountingWalkPolicy +{ + CountingWalkPolicy(unsigned int const& nump_PBSmall, BallPoly const& PBSmall) + : _nump_PBSmall(nump_PBSmall) + , _PBSmall(PBSmall) + {} + + template + void apply(PointList &randPoints, + Point &p) + { + if (_PBSmall.second().is_in(p) == -1)//is in + { + randPoints.push_back(p); + ++_nump_PBSmall; + } + } + + unsigned int get_nump_PBSmall() const + { + return _nump_PBSmall; + } + +private : + unsigned int _nump_PBSmall; + BallPoly _PBSmall; +}; + +#endif // SAMPLING_POLICIES_HPP diff --git a/src/volesti/include/volume/volume_cooling_balls.hpp b/src/volesti/include/volume/volume_cooling_balls.hpp new file mode 100644 index 00000000..d90604fa --- /dev/null +++ b/src/volesti/include/volume/volume_cooling_balls.hpp @@ -0,0 +1,853 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Repouskos Panagiotis, as part of Google Summer of Code 2019 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLUME_COOLING_BALLS_HPP +#define VOLUME_COOLING_BALLS_HPP + +#include +#include + +#include "cartesian_geom/cartesian_kernel.h" +#include "convex_bodies/hpolytope.h" +#ifndef DISABLE_LPSOLVE + #include "convex_bodies/vpolytope.h" + #include "convex_bodies/zpolytope.h" + #include "convex_bodies/vpolyintersectvpoly.h" +#endif +#include "random_walks/uniform_cdhr_walk.hpp" +#include "convex_bodies/ball.h" +#include "convex_bodies/ballintersectconvex.h" +#include "sampling/random_point_generators.hpp" +#include "volume/math_helpers.hpp" + + +//////////////////////////////////// +// ball annealing + +template +struct cooling_ball_parameters +{ + cooling_ball_parameters(unsigned int win_len) + : lb(0.1) + , ub(0.15) + , p(0.75) + , rmax(0) + , alpha(0.2) + , win_len(win_len) + , N(125) + , nu(10) + , window2(false) + {} + + NT lb; + NT ub; + NT p; + NT rmax; + NT alpha; + int win_len; + int N; + int nu; + bool window2; +}; + +/// Helpers + +template +bool check_convergence(ConvexBody &P, + PointList const& randPoints, + bool& too_few, + NT& ratio, + int const& nu, + bool const& precheck, + bool const& lastball, + cooling_ball_parameters const& parameters) +{ + NT alpha = parameters.alpha; + std::vector ratios; + std::pair mv; + int m = randPoints.size()/nu; + int i = 1; + NT T; + NT rs; + NT alpha_check = 0.01; + size_t countsIn = 0; + + for (auto pit=randPoints.begin(); pit!=randPoints.end(); ++pit, i++) + { + if (P.is_in(*pit)==-1) countsIn++; + if (i % m == 0) + { + ratios.push_back(NT(countsIn)/m); + countsIn = 0; + if (ratios.size()>1 && precheck) + { + boost::math::students_t dist(ratios.size() - 1); + mv = get_mean_variance(ratios); + ratio = mv.first; + rs = std::sqrt(mv.second); + T = rs * (boost::math::quantile + (boost::math::complement(dist, alpha_check / 2.0)) + / std::sqrt(NT(ratios.size()))); + if (ratio + T < parameters.lb) + { + too_few = true; + return false; + } else if (ratio - T > parameters.ub) return false; + } + } + } + + if (precheck) alpha *= 0.5; + mv = get_mean_variance(ratios); + ratio = mv.first; + rs = std::sqrt(mv.second); + boost::math::students_t dist(nu - 1); + T = rs * (boost::math::quantile(boost::math::complement(dist, alpha)) + / std::sqrt(NT(nu))); + if (ratio > parameters.lb + T) + { + if (lastball) return true; + if ((precheck && ratio < parameters.ub - T) + || (!precheck && ratio < parameters.ub + T)) return true; + return false; + } + too_few = true; + return false; +} + + +template +bool get_first_ball(Polytope &P, + Ball& B0, + NT& ratio, + NT const& radius_input, + cooling_ball_parameters const& parameters, + RNG& rng) { + const unsigned max_iterarions = 20; + NT tolerance = 0.00000000001; + typedef typename Polytope::PointType Point; + int n = P.dimension(); + int iter = 1; + bool bisection_int = false; + bool pass = false; + bool too_few = false; + std::list randPoints; + NT rmax = parameters.rmax; + NT sqrt_n = std::sqrt(NT(n)); + NT radius1 = radius_input; + + if (rmax > 0.0) { + for (int i = 0; i < 1200; ++i) { + randPoints.push_back(GetPointInDsphere::apply(n, rmax, rng)); + } + pass = check_convergence(P, randPoints, too_few, ratio, + 10, true, false, parameters); + if (pass || !too_few) { + B0 = Ball(Point(n), rmax * rmax); + return true; + } + bisection_int = true; + } else { + rmax = 2 * sqrt_n * radius1; + } + NT radius = radius1; + + while (!bisection_int) { + randPoints.clear(); + too_few = false; + + for (int i = 0; i < 1200; ++i) { + randPoints.push_back(GetPointInDsphere::apply(n, rmax, rng)); + } + + if (check_convergence(P, randPoints, too_few, ratio, 10, + true, false, parameters)) { + B0 = Ball(Point(n), rmax * rmax); + return true; + } + + if (too_few) break; + radius1 = rmax; + rmax = rmax + 2 * sqrt_n * radius; + } + + NT rad_med; + NT rad0 = radius1; + NT rad_m = rmax; + + while (iter <= max_iterarions) { + rad_med = 0.5 * (radius1 + rmax); + randPoints.clear(); + too_few = false; + + for (int i = 0; i < 1200; ++i) { + randPoints.push_back(GetPointInDsphere::apply(n, rad_med, rng)); + } + + if (check_convergence(P, randPoints, too_few, ratio, 10, + true, false, parameters)) { + B0 = Ball(Point(n), rad_med * rad_med); + return true; + } + + if (too_few) { + rmax = rad_med; + } else { + radius1 = rad_med; + } + + if (rmax - radius1 < tolerance) { + radius1 = rad0; + rmax = rad_m; + iter++; + } + } + return false; +} + + +template +bool get_next_zonotopeball(std::vector& BallSet, + PointList const& randPoints, + NT const& rad_min, + std::vector& ratios, + cooling_ball_parameters const& parameters) +{ + const unsigned max_iterarions = 20; + NT tolerance = 0.00000000001; + int n = (*randPoints.begin()).dimension(); + int iter = 1; + bool too_few; + NT radmax = NT(0); + NT radmin = rad_min; + + for (auto rpit = randPoints.begin(); rpit!=randPoints.end(); ++rpit) + { + NT pnorm = (*rpit).squared_length(); + if (pnorm > radmax) radmax = pnorm; + } + ball Biter; + radmax = std::sqrt(radmax); + NT radmin_init = radmin; + NT radmax_init = radmax; + + while (iter <= max_iterarions) + { + NT rad = 0.5 * (radmin + radmax); + Biter = ball(Point(n), rad * rad); + too_few = false; + + NT ratio; + if (check_convergence(Biter, randPoints, too_few, ratio, + parameters.nu, false, false, parameters)) + { + BallSet.push_back(Biter); + ratios.push_back(ratio); + return true; + } + + if (too_few) + { + radmin = rad; + } else + { + radmax = rad; + } + + if (radmax-radmin < tolerance) + { + radmin = radmin_init; + radmax = radmax_init; + iter++; + } + } + return false; +} + + +template +< + typename RandomPointGenerator, + typename PolyBall, + typename ball, + typename Polytope, + typename NT, + typename RNG +> +bool get_sequence_of_polytopeballs(Polytope &P, + std::vector& BallSet, + std::vector& ratios, + int const& Ntot, + NT const& radius, + unsigned int const& walk_length, + cooling_ball_parameters const& parameters, + RNG& rng) +{ + typedef typename Polytope::PointType Point; + bool fail; + int n = P.dimension(); + NT ratio; + NT ratio0; + std::list randPoints; + ball B0; + Point q(n); + + if ( !get_first_ball(P, B0, ratio, radius, parameters, rng) ) + { + return false; + } + + ratio0 = ratio; + + PushBackWalkPolicy push_back_policy; + RandomPointGenerator::apply(P, q, Ntot, walk_length, + randPoints, push_back_policy, rng); + + if (check_convergence(B0, randPoints, + fail, ratio, parameters.nu, + false, true, parameters)) + { + ratios.push_back(ratio); + BallSet.push_back(B0); + ratios.push_back(ratio0); + return true; + } + if ( !get_next_zonotopeball(BallSet, randPoints, B0.radius(), ratios, + parameters) ) + { + return false; + } + + while (true) + { + PolyBall zb_it(P, BallSet[BallSet.size()-1]); + q.set_to_origin(); + randPoints.clear(); + + RandomPointGenerator::apply(zb_it, q, Ntot, walk_length, + randPoints, push_back_policy, rng); + if (check_convergence(B0, randPoints, fail, ratio, parameters.nu, + false, true, parameters)) + { + ratios.push_back(ratio); + BallSet.push_back(B0); + ratios.push_back(ratio0); + return true; + } + if ( !get_next_zonotopeball(BallSet, randPoints, B0.radius(), + ratios, parameters) ) + { + return false; + } + } +} + + +//////////////////////////////////// +/// +/// ratio estimation + +template +bool is_max_error(NT const& a, NT const& b, NT const& error) +{ + return ((b-a)/a +struct estimate_ratio_parameters +{ +public: + + estimate_ratio_parameters(unsigned int W_len, unsigned int N, NT ratio) + : min_val(std::numeric_limits::lowest()) + , max_val(std::numeric_limits::max()) + , max_iterations_estimation(10000000) + , min_index(W_len-1) + , max_index(W_len-1) + , W(W_len) + , index(0) + , tot_count(N) + , count_in(N * ratio) + , iter(0) + , last_W(std::vector(W_len)) + , minmaxIt(last_W.begin()) + {} + + NT min_val; + NT max_val; + const unsigned int max_iterations_estimation; + unsigned int min_index; + unsigned int max_index; + unsigned int W; + unsigned int index; + size_t tot_count; + size_t count_in; + unsigned int iter; + std::vector last_W; + typename std::vector::iterator minmaxIt; +}; + +template +bool estimate_ratio_generic(Pollyball &Pb2, Point const& p, NT const& error, + estimate_ratio_parameters &ratio_parameters) +{ + if (ratio_parameters.iter++ <= ratio_parameters.max_iterations_estimation) + { + if (Pb2.is_in(p) == -1) ratio_parameters.count_in = ratio_parameters.count_in + 1.0; + + ratio_parameters.tot_count = ratio_parameters.tot_count + 1.0; + NT val = NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); + ratio_parameters.last_W[ratio_parameters.index] = val; + + if (val <= ratio_parameters.min_val) + { + ratio_parameters.min_val = val; + ratio_parameters.min_index = ratio_parameters.index; + } else if (ratio_parameters.min_index == ratio_parameters.index) + { + ratio_parameters.minmaxIt = std::min_element(ratio_parameters.last_W.begin(), ratio_parameters.last_W.end()); + ratio_parameters.min_val = (*ratio_parameters.minmaxIt); + ratio_parameters.min_index = std::distance(ratio_parameters.last_W.begin(), ratio_parameters.minmaxIt); + } + + if (val >= ratio_parameters.max_val) + { + ratio_parameters.max_val = val; + ratio_parameters.max_index = ratio_parameters.index; + } else if (ratio_parameters.max_index == ratio_parameters.index) + { + ratio_parameters.minmaxIt = std::max_element(ratio_parameters.last_W.begin(), ratio_parameters.last_W.end()); + ratio_parameters.max_val = (*ratio_parameters.minmaxIt); + ratio_parameters.max_index = std::distance(ratio_parameters.last_W.begin(), ratio_parameters.minmaxIt); + } + + if ( (ratio_parameters.max_val - ratio_parameters.min_val) / ratio_parameters.max_val <= error/2.0 ) + { + return true; + } + + ratio_parameters.index = ratio_parameters.index % ratio_parameters.W + 1; + if (ratio_parameters.index == ratio_parameters.W) ratio_parameters.index = 0; + + return false; + } + return true; +} + + +template +< + typename WalkType, + typename Point, + typename PolyBall1, + typename PolyBall2, + typename NT, + typename RNG +> +NT estimate_ratio(PolyBall1 &Pb1, + PolyBall2 &Pb2, + NT const& ratio, + NT const& error, + unsigned int const& W, + unsigned int const& Ntot, + unsigned int const& walk_length, + RNG& rng) +{ + estimate_ratio_parameters ratio_parameters(W, Ntot, ratio); + unsigned int n = Pb1.dimension(); + Point p(n); + WalkType walk(Pb1, p, rng); + + do + { + walk.template apply(Pb1, p, walk_length, rng); + } while(!estimate_ratio_generic(Pb2, p, error, ratio_parameters)); + + return NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); +} + +template +< typename Point, + typename ball, + typename PolyBall, + typename NT, + typename RNG +> +NT estimate_ratio(ball const& B, + PolyBall &Pb2, + NT const& ratio, + NT const& error, + int const& W, + int const& Ntot, + RNG& rng) +{ + estimate_ratio_parameters ratio_parameters(W, Ntot, ratio); + unsigned int n = B.dimension(); + Point p(n); + NT radius = B.radius(); + + do + { + p = GetPointInDsphere::apply(n, radius, rng); + } while(!estimate_ratio_generic(Pb2, p, error, ratio_parameters)); + + return NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); +} + +//-------------------------------------------------------------------------- + +template +struct estimate_ratio_interval_parameters +{ +public: + estimate_ratio_interval_parameters(unsigned int W_len, + unsigned int N, + NT ratio) + : mean(0) + , sum_sq(0) + , sum(0) + , s(0) + , max_iterations_estimation(10000000) + , W(W_len) + , index(0) + , tot_count(N) + , count_in(N * ratio) + , iter(0) + , last_W(std::vector(W_len)) + {} + + NT mean; + NT sum_sq; + NT sum; + NT s; + const unsigned int max_iterations_estimation; + unsigned int W; + unsigned int index; + size_t tot_count; + size_t count_in; + unsigned int iter; + std::vector last_W; +}; + +template +void full_sliding_window(Pollyball &Pb2, + Point const& p, + estimate_ratio_interval_parameters& ratio_parameters) +{ + if (Pb2.is_in(p) == -1) ratio_parameters.count_in = ratio_parameters.count_in + 1.0; + + ratio_parameters.tot_count = ratio_parameters.tot_count + 1.0; + NT val = NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); + ratio_parameters.sum += val; + ratio_parameters.sum_sq += val * val; + ratio_parameters.last_W[ratio_parameters.index] = val; + ratio_parameters.index = ratio_parameters.index % ratio_parameters.W + 1; + if (ratio_parameters.index == ratio_parameters.W) ratio_parameters.index = 0; +} + +template +bool estimate_ratio_interval_generic(Pollyball &Pb2, + Point const& p, + NT const& error, + NT const& zp, + estimate_ratio_interval_parameters + & ratio_parameters) +{ + if (ratio_parameters.iter++ <= ratio_parameters.max_iterations_estimation) + { + if (Pb2.is_in(p) == -1) ratio_parameters.count_in = ratio_parameters.count_in + 1.0; + + ratio_parameters.tot_count = ratio_parameters.tot_count + 1.0; + NT val = NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); + + ratio_parameters.mean = (ratio_parameters.mean + - ratio_parameters.last_W[ratio_parameters.index] / + NT(ratio_parameters.W)) + val / NT(ratio_parameters.W); + + ratio_parameters.sum_sq = (ratio_parameters.sum_sq - + ratio_parameters.last_W[ratio_parameters.index] + * ratio_parameters.last_W[ratio_parameters.index]) + + val * val; + + ratio_parameters.sum = (ratio_parameters.sum + - ratio_parameters.last_W[ratio_parameters.index]) + + val; + + ratio_parameters.s = std::sqrt((ratio_parameters.sum_sq + NT(ratio_parameters.W) * + ratio_parameters.mean * ratio_parameters.mean - NT(2) + * ratio_parameters.mean + * ratio_parameters.sum) / + NT(ratio_parameters.W)); + + ratio_parameters.last_W[ratio_parameters.index] = val; + + ratio_parameters.index = ratio_parameters.index % ratio_parameters.W + 1; + if (ratio_parameters.index == ratio_parameters.W) + { + ratio_parameters.index = 0; + } + + if (is_max_error(val - zp * ratio_parameters.s, + val + zp * ratio_parameters.s, + error)) + { + return true; + } + return false; + } + return true; +} + +template +< + typename Point, + typename ball, + typename PolyBall2, + typename NT, + typename RNG +> +NT estimate_ratio_interval(ball const& B, + PolyBall2 &Pb2, + NT const& ratio, + NT const& error, + int const& W, + int const& Ntot, + NT const& prob, + RNG& rng) +{ + estimate_ratio_interval_parameters ratio_parameters(W, Ntot, ratio); + boost::math::normal dist(0.0, 1.0); + NT zp = boost::math::quantile(boost::math::complement(dist, (1.0 - prob)/2.0)); + NT radius = B.radius(); + + unsigned int n = Pb2.dimension(); + Point p(n); + + for (int i = 0; i < ratio_parameters.W; ++i) + { + p = GetPointInDsphere::apply(n, radius, rng); + full_sliding_window(Pb2, p, ratio_parameters); + } + + ratio_parameters.mean = ratio_parameters.sum / NT(ratio_parameters.W); + + do { + p = GetPointInDsphere::apply(n, radius, rng); + } while (!estimate_ratio_interval_generic(Pb2, p, error, zp, ratio_parameters)); + + return NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); +} + + +template +< + typename WalkType, + typename Point, + typename PolyBall1, + typename PolyBall2, + typename NT, + typename RNG +> +NT estimate_ratio_interval(PolyBall1 &Pb1, + PolyBall2 &Pb2, + NT const& ratio, + NT const& error, + int const& W, + int const& Ntot, + NT const& prob, + unsigned int const& walk_length, + RNG& rng) +{ + estimate_ratio_interval_parameters ratio_parameters(W, Ntot, ratio); + boost::math::normal dist(0.0, 1.0); + NT zp = boost::math::quantile(boost::math::complement(dist, (1.0 - prob)/2.0)); + + unsigned int n = Pb1.dimension(); + Point p(n); + WalkType walk(Pb1, p, rng); + + for (int i = 0; i < ratio_parameters.W; ++i) + { + walk.template apply(Pb1, p, walk_length, rng); + full_sliding_window(Pb2, p, ratio_parameters); + } + + ratio_parameters.mean = ratio_parameters.sum / NT(ratio_parameters.W); + + do { + walk.template apply(Pb1, p, walk_length, rng); + } while (!estimate_ratio_interval_generic(Pb2, p, error, zp, ratio_parameters)); + + return NT(ratio_parameters.count_in) / NT(ratio_parameters.tot_count); +} + +template +< + typename WalkTypePolicy, + typename Polytope, + typename RandomNumberGenerator = BoostRandomNumberGenerator +> +std::pair volume_cooling_balls(Polytope& Pin, + RandomNumberGenerator &rng, + double const& error = 0.1, + unsigned int const& walk_length = 1, + unsigned int const& win_len = 300) +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef Ball BallType; + typedef BallIntersectPolytope PolyBall; + typedef typename Polytope::VT VT; + typedef std::list PointList; + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > WalkType; + typedef RandomPointGenerator RandomPointGenerator; + + auto P(Pin); + cooling_ball_parameters parameters(win_len); + + int n = P.dimension(); + NT prob = parameters.p; + int N_times_nu = parameters.N * parameters.nu; + + auto InnerBall = P.ComputeInnerBall(); + if (InnerBall.second < 0.0) return std::pair (-1.0, 0.0); + + NT radius = InnerBall.second; + Point c = InnerBall.first; + + std::vector BallSet; + std::vector ratios; + + // move the chebychev center to the origin + // and apply the same shifting to the polytope + P.shift(c.getCoefficients()); + + if ( !get_sequence_of_polytopeballs + < + RandomPointGenerator, + PolyBall + >(P, BallSet, ratios, + N_times_nu, radius, walk_length, + parameters, rng) ) + { + return std::pair (-1.0, 0.0); + } + + NT vol = (NT(n)/NT(2) * std::log(M_PI)) + NT(n)*std::log((*(BallSet.end() - 1)).radius()) - log_gamma_function(NT(n) / NT(2) + 1); + + int mm = BallSet.size() + 1; + prob = std::pow(prob, 1.0 / NT(mm)); + NT er0 = error / (2.0 * std::sqrt(NT(mm))); + NT er1 = (error * std::sqrt(4.0 * NT(mm) - 1)) / (2.0 * std::sqrt(NT(mm))); + + vol += (parameters.window2) ? + std::log(estimate_ratio(*(BallSet.end() - 1), + P, *(ratios.end() - 1), + er0, parameters.win_len, 1200, rng)) + : std::log(estimate_ratio_interval(*(BallSet.end() - 1), + P, *(ratios.end() - 1), + er0, parameters.win_len, 1200, + prob, rng)); + auto balliter = BallSet.begin(); + auto ratioiter = ratios.begin(); + + er1 = er1 / std::sqrt(NT(mm) - 1.0); + + if (*ratioiter != 1) + { + vol += (!parameters.window2) ? + std::log(NT(1) / estimate_ratio_interval + (P, + *balliter, + *ratioiter, + er1, + parameters.win_len, + N_times_nu, + prob, + walk_length, + rng)) + : std::log(NT(1) / estimate_ratio + (P, + *balliter, + *ratioiter, + er1, + parameters.win_len, + N_times_nu, + walk_length, + rng)); + } + + for ( ; balliter < BallSet.end() - 1; ++balliter, ++ratioiter) + { + PolyBall Pb(P, *balliter); + vol += (!parameters.window2) ? + std::log(NT(1) / estimate_ratio_interval + (Pb, + *(balliter + 1), + *(ratioiter + 1), + er1, parameters.win_len, + N_times_nu, + prob, walk_length, + rng)) + : std::log(NT(1) / estimate_ratio + (Pb, + *balliter, + *ratioiter, + er1, + parameters.win_len, + N_times_nu, + walk_length, + rng)); + } + + return std::pair (vol, std::exp(vol)); +} + + + +template +< + typename WalkTypePolicy = CDHRWalk, + typename RandomNumberGenerator = BoostRandomNumberGenerator, + typename Polytope +> +std::pair volume_cooling_balls(Polytope &Pin, + double const& error = 0.1, + unsigned int const& walk_length = 1) +{ + RandomNumberGenerator rng(Pin.dimension()); + return volume_cooling_balls(Pin, rng, error, walk_length); +} + + +template +< + typename WalkTypePolicy = CDHRWalk, + typename RandomNumberGenerator = BoostRandomNumberGenerator, + typename Polytope +> +std::pair volume_cooling_balls(Polytope &Pin, + Cartesian::Point const& interior_point, + unsigned int const& walk_length = 1, + double const& error = 0.1) +{ + RandomNumberGenerator rng(Pin.dimension()); + Pin.set_interior_point(interior_point); + return volume_cooling_balls(Pin, rng, error, walk_length); +} + +#endif // VOLUME_COOLING_BALLS_HPP diff --git a/src/volesti/include/volume/volume_cooling_gaussians.hpp b/src/volesti/include/volume/volume_cooling_gaussians.hpp new file mode 100644 index 00000000..ca06ce31 --- /dev/null +++ b/src/volesti/include/volume/volume_cooling_gaussians.hpp @@ -0,0 +1,490 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLUME_COOLING_GAUSSIANS_HPP +#define VOLUME_COOLING_GAUSSIANS_HPP + +//#define VOLESTI_DEBUG + +#include +#include +#include +#include +#include + +#include "cartesian_geom/cartesian_kernel.h" +#include "random_walks/gaussian_helpers.hpp" +#include "random_walks/gaussian_ball_walk.hpp" +#include "random_walks/gaussian_cdhr_walk.hpp" +#include "sampling/random_point_generators.hpp" +#include "volume/math_helpers.hpp" + + +/////////////////// Helpers for random walks + +template +struct update_delta +{ + template + static void apply(WalkType, NT) {} +}; + +template +struct update_delta> +{ + template + static void apply(GaussianBallWalk::Walk walk, + NT delta) + { + walk.update_delta(delta); + } +}; + + +////////////////////////////// Algorithms + +// Gaussian Anealling + +// Implementation is based on algorithm from paper "A practical volume algorithm", +// Springer-Verlag Berlin Heidelberg and The Mathematical Programming Society 2015 +// Ben Cousins, Santosh Vempala + +// Compute the first variance a_0 for the starting gaussian +template +void get_first_gaussian(Polytope& P, + NT const& frac, + NT const& chebychev_radius, + NT const& error, + std::vector & a_vals) +{ + // if tol is smaller than 1e-6 no convergence can be obtained when float is used + NT tol = std::is_same::value ? 0.001 : 0.0000001; + + std::vector dists = P.get_dists(chebychev_radius); + NT lower = 0.0; + NT upper = 1.0; + + // Compute an upper bound for a_0 + unsigned int i; + const unsigned int maxiter = 10000; + for (i= 1; i <= maxiter; ++i) { + NT sum = 0.0; + for (auto it = dists.begin(); it != dists.end(); ++it) + { + sum += std::exp(-upper * std::pow(*it, 2.0)) + / (2.0 * (*it) * std::sqrt(M_PI * upper)); + } + + if (sum > frac * error) + { + upper = upper * 10; + } else { + break; + } + } + + if (i == maxiter) { +#ifdef VOLESTI_DEBUG + std::cout << "Cannot obtain sharp enough starting Gaussian" << std::endl; +#endif + return; + } + + //get a_0 with binary search + while (upper - lower > tol) + { + NT mid = (upper + lower) / 2.0; + NT sum = 0.0; + for (auto it = dists.begin(); it != dists.end(); ++it) { + sum += std::exp(-mid * std::pow(*it, 2.0)) + / (2.0 * (*it) * std::sqrt(M_PI * mid)); + } + + if (sum < frac * error) { + upper = mid; + } else { + lower = mid; + } + } + a_vals.push_back((upper + lower) / NT(2.0)); +} + + +// Compute a_{i+1} when a_i is given +template +< + typename RandomPointGenerator, + typename Polytope, + typename Point, + typename NT, + typename RandomNumberGenerator +> +NT get_next_gaussian(Polytope& P, + Point &p, + NT const& a, + const unsigned int &N, + const NT &ratio, + const NT &C, + const unsigned int& walk_length, + RandomNumberGenerator& rng) +{ + + NT last_a = a; + NT last_ratio = 0.1; + //k is needed for the computation of the next variance a_{i+1} = a_i * (1-1/d)^k + NT k = 1.0; + const NT tol = 0.00001; + bool done=false; + std::vector fn(N,NT(0.0)); + std::list randPoints; + typedef typename std::vector::iterator viterator; + + //sample N points + PushBackWalkPolicy push_back_policy; + RandomPointGenerator::apply(P, p, last_a, N, walk_length, randPoints, + push_back_policy, rng); + + while (!done) + { + NT new_a = last_a * std::pow(ratio,k); + + auto fnit = fn.begin(); + for (auto pit=randPoints.begin(); pit!=randPoints.end(); ++pit, fnit++) + { + *fnit = eval_exp(*pit, new_a)/eval_exp(*pit, last_a); + } + std::pair mv = get_mean_variance(fn); + + // Compute a_{i+1} + if (mv.second/(mv.first * mv.first)>=C || mv.first/last_ratio<1.0+tol) + { + if (k != 1.0) + { + k = k / 2; + } + done = true; + } else { + k = 2 * k; + } + last_ratio = mv.first; + } + return last_a * std::pow(ratio, k); +} + +// Compute the sequence of spherical gaussians +template +< + typename WalkType, + typename RandomPointGenerator, + typename Polytope, + typename NT, + typename RandomNumberGenerator +> +void compute_annealing_schedule(Polytope& P, + NT const& ratio, + NT const& C, + NT const& frac, + unsigned int const& N, + unsigned int const& walk_length, + NT const& chebychev_radius, + NT const& error, + std::vector& a_vals, + RandomNumberGenerator& rng) +{ + typedef typename Polytope::PointType Point; + typedef typename Polytope::VT VT; + + // Compute the first gaussian + get_first_gaussian(P, frac, chebychev_radius, error, a_vals); + +#ifdef VOLESTI_DEBUG + std::cout<<"first gaussian computed\n"< + (P, p, a_vals[it], N, ratio, C, walk_length, rng); + + NT curr_fn = 0; + NT curr_its = 0; + auto steps = totalSteps; + + WalkType walk(P, p, a_vals[it], rng); + //TODO: test update delta here? + + update_delta + ::apply(walk, 4.0 * chebychev_radius + / std::sqrt(std::max(NT(1.0), a_vals[it]) * NT(n))); + + // Compute some ratios to decide if this is the last gaussian + for (unsigned int j = 0; j < steps; j++) + { + walk.template apply(P, p, a_vals[it], walk_length, rng); + curr_its += 1.0; + curr_fn += eval_exp(p, next_a) / eval_exp(p, a_vals[it]); + } + + // Remove the last gaussian. + // Set the last a_i equal to zero + if (next_a>0 && curr_fn/curr_its>(1.0+tol)) + { + a_vals.push_back(next_a); + it++; + } else if (next_a <= 0) + { + a_vals.push_back(a_stop); + it++; + break; + } else { + a_vals[it] = a_stop; + break; + } + } +} + +template +struct gaussian_annealing_parameters +{ + gaussian_annealing_parameters(unsigned int d) + : frac(0.1) + , ratio(NT(1)-NT(1)/(NT(d))) + , C(NT(2)) + , N(500 * ((int) C) + ((int) (d * d / 2))) + , W(6*d*d+800) + {} + + NT frac; + NT ratio; + NT C; + unsigned int N; + unsigned int W; +}; + +template +< + typename WalkTypePolicy, + typename Polytope, + typename RandomNumberGenerator + +> +double volume_cooling_gaussians(Polytope& Pin, + RandomNumberGenerator& rng, + double const& error = 0.1, + unsigned int const& walk_length = 1) +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef typename Polytope::VT VT; + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > WalkType; + typedef GaussianRandomPointGenerator RandomPointGenerator; + + //const NT maxNT = std::numeric_limits::max();//1.79769e+308; + //const NT minNT = std::numeric_limits::min();//-1.79769e+308; + + auto P(Pin); //copy and work with P because we are going to shift + unsigned int n = P.dimension(); + unsigned int m = P.num_of_hyperplanes(); + gaussian_annealing_parameters parameters(P.dimension()); + //RandomNumberGenerator rng(n); + + // Consider Chebychev center as an internal point + auto InnerBall = P.ComputeInnerBall(); + if (InnerBall.second < 0.0) return -1.0; + + Point c = InnerBall.first; + NT radius = InnerBall.second; + + // Move the chebychev center to the origin and apply the same shifting to the polytope + P.shift(c.getCoefficients()); + + // Computing the sequence of gaussians +#ifdef VOLESTI_DEBUG + std::cout<<"\n\nComputing annealing...\n"< a_vals; + NT ratio = parameters.ratio; + NT C = parameters.C; + unsigned int N = parameters.N; + + compute_annealing_schedule + < + WalkType, + RandomPointGenerator + >(P, ratio, C, parameters.frac, N, walk_length, radius, error, a_vals, rng); + +#ifdef VOLESTI_DEBUG + std::cout<<"All the variances of schedule_annealing computed in = " + << (double)clock()/(double)CLOCKS_PER_SEC-tstart2<<" sec"< last_W2(W,0); + std::vector fn(mm,0); + std::vector its(mm,0); + VT lamdas; + lamdas.setZero(m); + NT vol = std::pow(M_PI/a_vals[0], (NT(n))/2.0); + Point p(n); // The origin is the Chebychev center of the Polytope + unsigned int i=0; + + typedef typename std::vector::iterator viterator; + viterator itsIt = its.begin(); + viterator avalsIt = a_vals.begin(); + viterator minmaxIt; + +#ifdef VOLESTI_DEBUG + std::cout<<"volume of the first gaussian = "<::min(); + NT max_val = std::numeric_limits::max(); + unsigned int min_index = W-1; + unsigned int max_index = W-1; + unsigned int index = 0; + unsigned int min_steps = 0; + std::vector last_W = last_W2; + + // Set the radius for the ball walk + WalkType walk(P, p, *avalsIt, rng); + + update_delta + ::apply(walk, 4.0 * radius + / std::sqrt(std::max(NT(1.0), *avalsIt) * NT(n))); + + while (!done || (*itsIt)= max_val) + { + max_val = val; + max_index = index; + } else if (max_index == index) + { + minmaxIt = std::max_element(last_W.begin(), last_W.end()); + max_val = *minmaxIt; + max_index = std::distance(last_W.begin(), minmaxIt); + } + + if ( (max_val-min_val)/max_val <= curr_eps/2.0 ) + { + done=true; + } + + index = index%W + 1; + if (index == W) index = 0; + } +#ifdef VOLESTI_DEBUG + std::cout << "ratio " << i << " = " << (*fnIt) / (*itsIt) + << " N_" << i << " = " << *itsIt << std::endl; +#endif + vol *= ((*fnIt) / (*itsIt)); + } + +#ifdef VOLESTI_DEBUG + NT sum_of_steps = 0.0; + for(viterator it = its.begin(); it != its.end(); ++it) { + sum_of_steps += *it; + } + auto steps= int(sum_of_steps); + std::cout<<"\nTotal number of steps = "<, + typename Polytope +> +double volume_cooling_gaussians(Polytope &Pin, + double const& error = 0.1, + unsigned int const& walk_length = 1) +{ + RandomNumberGenerator rng(Pin.dimension()); + return volume_cooling_gaussians(Pin, rng, error, walk_length); +} + + +template +< + typename WalkTypePolicy = GaussianCDHRWalk, + typename RandomNumberGenerator = BoostRandomNumberGenerator, + typename Polytope +> +double volume_cooling_gaussians(Polytope &Pin, + Cartesian::Point const& interior_point, + unsigned int const& walk_length = 1, + double const& error = 0.1) +{ + RandomNumberGenerator rng(Pin.dimension()); + Pin.set_interior_point(interior_point); + + return volume_cooling_gaussians(Pin, rng, error, walk_length); +} + +#endif // VOLUME_COOLING_GAUSSIANS_HPP diff --git a/src/volesti/include/volume/volume_cooling_hpoly.hpp b/src/volesti/include/volume/volume_cooling_hpoly.hpp new file mode 100644 index 00000000..c8183384 --- /dev/null +++ b/src/volesti/include/volume/volume_cooling_hpoly.hpp @@ -0,0 +1,385 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLUME_COOLING_HPOLY_HPP +#define VOLUME_COOLING_HPOLY_HPP + +#include "volume/volume_cooling_gaussians.hpp" +#include "sampling/random_point_generators.hpp" +#include "preprocess/min_sampling_covering_ellipsoid_rounding.hpp" + + +template +< + typename RandomPointGenerator, + typename Zonotope, + typename HPolytope, + typename NT, + typename RNG, + typename VT +> +bool get_first_poly(Zonotope &P, + HPolytope &HP, + NT &ratio, + cooling_ball_parameters const& parameters, + RNG& rng, VT &b_max) +{ + + typedef typename Zonotope::PointType Point; + typedef typename Zonotope::MT MT; + + PushBackWalkPolicy push_back_policy; + + const unsigned max_iterarions = 20; + const NT tolerance = 0.00000000001; + + MT G = P.get_mat().transpose(), A = HP.get_mat(); + b_max = (A*G).cwiseAbs().rowwise().sum(); + VT b_min = HP.get_vec(); + HPolytope HPiter(HP); + + int n = P.dimension(), m = b_max.size(), N = 1200, iter = 1, count = 0; + Point q(n); + bool too_few, print = false; + std::list randPoints; + + NT l=0.0, u=1.0, med; + VT b_med(m); + + while(iter <= max_iterarions) { + + q=Point(n); + med = (u + l) * 0.5; + b_med = b_min + (b_max-b_min)*med; + HPiter.set_vec(b_med); + + randPoints.clear(); + RandomPointGenerator::apply(HPiter, q, 1200, 10+10*n, + randPoints, push_back_policy, rng); + too_few = false; + + if(check_convergence(P, randPoints, + too_few, ratio, parameters.nu, + true, false, parameters)){ + HP.set_vec(b_med); + return true; + } + + if (too_few) { + u = med; + } else { + l = med; + } + if(med>0.9) { + HP.set_vec(b_med); + return true; + } + if(u-l < tolerance) { + u=1.0; + l=0.0; + iter++; + } + } + return false; +} + + +template +bool get_next_zonoball(std::vector &HPolySet, + HPolytope &HP2, + const VT &b_max, + const VT &b_min, + PointList &randPoints, + std::vector &ratios, + cooling_ball_parameters const& parameters) +{ + + typedef typename Zonotope::PointType Point; + + const unsigned max_iterarions = 20; + const NT tolerance = 0.00000000001; + + int n = HP2.dimension(), iter = 1; + bool too_few; + VT b_med(b_max.size()); + NT ratio, med, u = 1.0, l = 0.0; + + while (iter <= max_iterarions) { + med = (u + l) * 0.5; + b_med = b_min + (b_max-b_min) * med; + HP2.set_vec(b_med); + too_few = false; + + if(check_convergence(HP2, randPoints, + too_few, ratio, parameters.nu, + false, false, parameters)){ + HPolySet.push_back(HP2); + ratios.push_back(ratio); + return true; + } + if(too_few) { + l = med; + } else { + u = med; + } + if(u-l < tolerance) { + u=1.0; + l=0.0; + iter++; + } + } + return false; +} + +template +< + typename RandomPointGenerator, + typename ZonoHP, + typename Zonotope, + typename HPolytope, + typename VT, + typename NT, + typename RNG +> +bool get_sequence_of_zonopolys(Zonotope &Z, + const HPolytope &HP, + std::vector &HPolySet, + std::vector &ratios, + const VT &b_max, + unsigned int const& N_times_nu, + unsigned int const& walk_length, + cooling_ball_parameters const& parameters, + RNG& rng) +{ + + bool too_few=false; + typedef typename Zonotope::PointType Point; + typedef typename Zonotope::MT MT; + + PushBackWalkPolicy push_back_policy; + + int n = Z.dimension(); + MT G = Z.get_mat().transpose(); + MT AG = HP.get_mat()*G; + NT ratio; + std::list randPoints; + Point q(n); + + RandomPointGenerator::apply(Z, q, N_times_nu, walk_length, + randPoints, push_back_policy, rng); + HPolytope HP2(HP); + if (check_convergence(HP, randPoints, + too_few, ratio, parameters.nu, + false, true, parameters)) { + ratios.push_back(ratio); + return true; + } + if ( !get_next_zonoball(HPolySet, HP2, b_max, HP.get_vec(), + randPoints, ratios, parameters)) + { + return false; + } + + VT Zs_min = HP.get_vec(); + + while (true) { + + ZonoHP ZHP2(Z,HP2); + q=Point(n); + randPoints.clear(); + RandomPointGenerator::apply(ZHP2, q, N_times_nu, walk_length, + randPoints, push_back_policy, rng); + + if (check_convergence(HP, randPoints, + too_few, ratio, parameters.nu, + false, true, parameters)) + { + ratios.push_back(ratio); + return true; + } + if ( !get_next_zonoball(HPolySet, HP2, HP2.get_vec(), + Zs_min, randPoints, ratios, parameters) ) + { + return false; + } + } +} + +// TODO: Rewrite to avoid some matrix operations and improve the signature +template + < + typename Zonotope, + typename HPolytope + > +HPolytope compute_hpoly_for_mmc(Zonotope const& P) { + typedef typename Zonotope::PointType Point; + typedef typename Zonotope::NT NT; + typedef typename Zonotope::VT VT; + typedef typename Zonotope::MT MT; + + MT V = P.get_mat(); + MT G = V.transpose(); + int m = G.cols(); + std::list randPoints; + + MT XX(m, 2*m); + XX << MT::Identity(m,m), -MT::Identity(m,m); + MT AA = XX.transpose(); VT b = VT::Ones(2*m); + MT T = P.get_T(); + MT Tt = T.transpose(); + MT A2 = AA * Tt, B = G * Tt; + MT A3 = A2 * B.inverse(); + + NT row_norm; + for (int i = 0; i < A3.rows(); ++i) { + row_norm = A3.row(i).norm(); + A3.row(i) = A3.row(i) / row_norm; + b(i) = b(i) / row_norm; + } + + return HPolytope(P.dimension(), A3, b); +} + + +template +< + typename WalkTypePolicy, + typename HPolytope, + typename Zonotope, + typename RandomNumberGenerator +> +double volume_cooling_hpoly (Zonotope const& Pin, + RandomNumberGenerator &rng, + double const& error = 1.0, + unsigned int const& walk_length = 1, + unsigned int const& win_len = 200) +{ + + typedef typename Zonotope::PointType Point; + typedef typename Point::FT NT; + typedef ZonoIntersectHPoly ZonoHP; + typedef typename Zonotope::VT VT; + typedef typename Zonotope::MT MT; + typedef std::list PointList; + + typedef typename WalkTypePolicy::template Walk WalkType; + typedef RandomPointGenerator ZonoRandomPointGenerator; + + typedef typename CDHRWalk::template Walk CdhrWalk; + typedef RandomPointGenerator CdhrRandomPointGenerator; + + auto P(Pin); + cooling_ball_parameters parameters(win_len); + + int n = P.dimension(); + NT prob = parameters.p, ratio; + int N_times_nu = parameters.N * parameters.nu; + + HPolytope HP(compute_hpoly_for_mmc(P)); + VT b_max(2 * P.num_of_generators()); + if ( !get_first_poly(P, HP, ratio, parameters, rng, b_max) ) + { + return -1.0; + } + + std::vector HPolySet; + std::vector ratios; + std::vector diams_inter; + + if ( !get_sequence_of_zonopolys + (P, HP, HPolySet, ratios, + b_max, N_times_nu, walk_length, parameters, rng) ) + { + return -1.0; + } + + int mm = HPolySet.size() + 2; + int mm2 = mm + 1; + prob = std::pow(prob, 1.0/NT(mm2)); + NT er0 = error/(2.0*std::sqrt(NT(mm2))); + NT er1 = (error*std::sqrt(2.0*NT(mm2)-1))/(std::sqrt(2.0*NT(mm2))); + NT Her = error/(2.0*std::sqrt(NT(mm2))); + + HPolytope HP2(HP); + std::pair InnerBall = HP2.ComputeInnerBall(); + std::tuple res = min_sampling_covering_ellipsoid_rounding(HP2, InnerBall, + 10 + 10 * n, rng); + NT vol = std::get<2>(res) * volume_cooling_gaussians(HP2, rng, Her/2.0, 1); + + if (!parameters.window2) { + vol *= estimate_ratio_interval(HP, P, ratio, er0, parameters.win_len, 1200, + prob, 10 + 10 * n, rng); + } else { + vol *= estimate_ratio(HP, P, ratio, er0, parameters.win_len, 1200, 10 + 10 * n, rng); + } + + HPolytope b1, b2; + if (HPolySet.size()==0) { + if (ratios[0]!=1) { + if(!parameters.window2) { + vol = vol / estimate_ratio_interval(P, HP, ratios[0], er1, parameters.win_len, N_times_nu, + prob, walk_length, rng); + } else { + vol = vol / estimate_ratio(P, HP, ratios[0], er1, parameters.win_len, N_times_nu, + walk_length, rng); + } + } + } else { + er1 = er1 / std::sqrt(NT(mm)-1.0); + b1 = HPolySet[0]; + if(!parameters.window2) { + vol = vol / estimate_ratio_interval(P, b1, ratios[0], er1, parameters.win_len, N_times_nu, + prob, walk_length, rng); + } else { + vol = vol / estimate_ratio(P, b1, ratios[0], er1, parameters.win_len, N_times_nu, + walk_length, rng); + } + + for (int i = 0; i < HPolySet.size()-1; ++i) { + ZonoHP zb1(P,HPolySet[i]); + b2 = HPolySet[i+1]; + if(!parameters.window2) { + vol = vol / estimate_ratio_interval(zb1, b2, ratios[i], er1, parameters.win_len, + N_times_nu, prob, walk_length, rng); + } else { + vol = vol / estimate_ratio(zb1, b2, ratios[i], er1, parameters.win_len, N_times_nu, + walk_length, rng); + } + } + + ZonoHP zb1(P, HPolySet[HPolySet.size() - 1]); + if (!parameters.window2) { + vol = vol / estimate_ratio_interval(zb1, HP, ratios[ratios.size() - 1], er1, + parameters.win_len, N_times_nu, prob, walk_length, + rng); + } else { + vol = vol / estimate_ratio(zb1, HP, ratios[ratios.size() - 1], er1, parameters.win_len, + N_times_nu, walk_length, rng); + } + } + + return vol; + +} + + +template +< + typename WalkTypePolicy, + typename RandomNumberGenerator, + typename HPolytope, + typename Polytope +> +double volume_cooling_hpoly(Polytope& Pin, + double const& error = 0.1, + unsigned int const& walk_length = 1) +{ + RandomNumberGenerator rng(Pin.dimension()); + return volume_cooling_hpoly(Pin, rng, error, walk_length); +} + +#endif // VOLUME_COOLING_HPOLY_HPP diff --git a/src/volesti/include/volume/volume_sequence_of_balls.hpp b/src/volesti/include/volume/volume_sequence_of_balls.hpp new file mode 100644 index 00000000..b79358a6 --- /dev/null +++ b/src/volesti/include/volume/volume_sequence_of_balls.hpp @@ -0,0 +1,266 @@ +// VolEsti (volume computation and sampling library) + +// Copyright (c) 2012-2020 Vissarion Fisikopoulos +// Copyright (c) 2018-2020 Apostolos Chalkis + +//Contributed and/or modified by Apostolos Chalkis, as part of Google Summer of Code 2018 program. + +// Licensed under GNU LGPL.3, see LICENCE file + +#ifndef VOLUME_SEQUENCE_OF_BALLS_HPP +#define VOLUME_SEQUENCE_OF_BALLS_HPP + +#include +#include +#include +#include +#include + +#include "cartesian_geom/cartesian_kernel.h" +#include "generators/boost_random_number_generator.hpp" +#include "convex_bodies/hpolytope.h" +#ifndef DISABLE_LPSOLVE + #include "convex_bodies/vpolytope.h" + #include "convex_bodies/zpolytope.h" + #include "convex_bodies/zonoIntersecthpoly.h" + #include "convex_bodies/vpolyintersectvpoly.h" +#endif +#include "convex_bodies/ball.h" +#include "convex_bodies/ballintersectconvex.h" +#include "random_walks/uniform_cdhr_walk.hpp" +#include "sampling/random_point_generators.hpp" +#include "volume/sampling_policies.hpp" + + +////////////////////////////// Algorithms + + +// ----- VOLUME ------ // + +template +< + typename WalkTypePolicy, + typename Polytope, + typename RandomNumberGenerator + +> +double volume_sequence_of_balls(Polytope& Pin, + RandomNumberGenerator &rng, + double const& error = 1.0, + unsigned int const& walk_length = 1, + unsigned int const& n_threads = 1) +{ + typedef typename Polytope::PointType Point; + typedef typename Polytope::VT VT; + typedef typename Polytope::MT MT; + typedef typename Point::FT NT; + typedef Ball Ball; + typedef BallIntersectPolytope BallPoly; + + typedef typename WalkTypePolicy::template Walk + < + Polytope, + RandomNumberGenerator + > walk; + + typedef RandomPointGenerator RandomPointGenerator; + + auto P(Pin); //copy and work with P because we are going to shift + unsigned int n = P.dimension(); + unsigned int rnum = std::pow(error, -2) * 400 * n * std::log(n); + //RandomNumberGenerator rng(P.dimension()); + + //Compute the Chebychev ball (largest inscribed ball) with center and radius + auto InnerBall = P.ComputeInnerBall(); + if (InnerBall.second < 0.0) return -1.0; + + Point c = InnerBall.first; + NT radius = InnerBall.second; + + // Move the chebychev center to the origin and apply the same shifting to the polytope + P.shift(c.getCoefficients()); + c = Point(n); + + // Scale by number of threads and prevent edge case rnum=0 from producing overflow later + rnum = rnum >= 2*n_threads ? rnum/n_threads : 2u; + NT vol = NT(0); + + // Perform the procedure for a number of threads and then take the average + for (auto t=0u; t::apply(P.dimension(), radius, rng); + std::list randPoints; //ds for storing rand points + + PushBackWalkPolicy push_back_policy; + RandomPointGenerator::apply(P, p, 1, 50*n, randPoints, push_back_policy, rng); + +#ifdef VOLESTI_DEBUG + double tstart2 = (double)clock()/(double)CLOCKS_PER_SEC; + std::cout<<"\nCompute "< balls; + + for (auto i=nb1; i<=nb2; ++i) + { + if (i == nb1) + { + balls.push_back(Ball(c,radius*radius)); + vol = (std::pow(M_PI,n/2.0)*(std::pow(balls[0].radius(), n) ) ) + / (tgamma(n/2.0+1)); + } else { + balls.push_back(Ball(c,std::pow(std::pow(2.0,NT(i)/NT(n)),2))); + } + } + assert(!balls.empty()); + + // Estimate Vol(P) + typename std::vector::iterator bit2=balls.end(); + bit2--; + + while (bit2!=balls.begin()) + { + //each step starts with some random points in PBLarge stored + //in list "randPoints", these points have been generated in a + //previous step + + BallPoly PBLarge(P,*bit2); + --bit2; + BallPoly PBSmall(P,*bit2); + +#ifdef VOLESTI_DEBUG + std::cout<<"("< counting_policy(nump_PBSmall, PBSmall); + RandomPointGenerator::apply(PBLarge, p_gen, rnum-nump_PBLarge, + walk_length, randPoints, + counting_policy, rng); + + nump_PBSmall = counting_policy.get_nump_PBSmall(); + + vol *= NT(rnum)/NT(nump_PBSmall); + +#ifdef VOLESTI_DEBUG + std::cout<, + typename Polytope +> +double volume_sequence_of_balls(Polytope &Pin, + double const& error = 1.0, + unsigned int const& walk_length = 1, + unsigned int const& n_threads = 1) +{ + RandomNumberGenerator rng(Pin.dimension()); + return volume_sequence_of_balls(Pin, rng, error, + walk_length, n_threads); +} + + +template +< + typename WalkTypePolicy = CDHRWalk, + typename RandomNumberGenerator = BoostRandomNumberGenerator, + typename Polytope +> +double volume_sequence_of_balls(Polytope &Pin, + Cartesian::Point const& interior_point, + unsigned int const& walk_length = 1, + double const& error = 1.0, + unsigned int const& n_threads = 1) +{ + RandomNumberGenerator rng(Pin.dimension()); + Pin.set_interior_point(interior_point); + + return volume_sequence_of_balls(Pin, rng, error, + walk_length, n_threads); +} + +#endif // VOLUME_SEQUENCE_OF_BALLS_HPP