diff --git a/include/preprocess/barrier_center_ellipsoid.hpp b/include/preprocess/barrier_center_ellipsoid.hpp index 0ab8c0457..9f440f222 100644 --- a/include/preprocess/barrier_center_ellipsoid.hpp +++ b/include/preprocess/barrier_center_ellipsoid.hpp @@ -27,7 +27,12 @@ The volumetric center is the minimizer of the volumetric barrier function, i.e., the optimal solution of the following optimization problem, - \min logdet \nabla^2 f(x), where f(x) the log barrier function + \min logdet \nabla^2 f(x), where f(x) the log barrier function + + The Vaidya center is the minimizer of the Vaidya barrier function, i.e., the optimal + solution of the following optimization problem, + + \min logdet \nabla^2 f(x) + d/m f(x), where f(x) the log barrier function. The function solves the problems by using the Newton method. diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index b524403ce..615791148 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -302,6 +302,8 @@ add_test(NAME round_max_ellipsoid_sparse COMMAND rounding_test -tc=round_max_ellipsoid_sparse) add_test(NAME round_volumetric_barrier_test COMMAND rounding_test -tc=round_volumetric_barrier_test) +add_test(NAME round_vaidya_barrier_test + COMMAND rounding_test -tc=round_vaidya_barrier_test) @@ -367,6 +369,8 @@ add_test(NAME test_max_ball_sparse COMMAND test_internal_points -tc=test_max_ball_sparse) add_test(NAME test_volumetric_center COMMAND test_internal_points -tc=test_volumetric_center) +add_test(NAME test_vaidya_center + COMMAND test_internal_points -tc=test_vaidya_center) diff --git a/test/rounding_test.cpp b/test/rounding_test.cpp index 3817cebb3..5beae0415 100644 --- a/test/rounding_test.cpp +++ b/test/rounding_test.cpp @@ -228,6 +228,37 @@ void rounding_volumetric_barrier_test(Polytope &HP, test_values(volume, expectedBilliard, exact); } +template +void rounding_vaidya_barrier_test(Polytope &HP, + double const& expectedBall, + double const& expectedCDHR, + double const& expectedRDHR, + double const& expectedBilliard, + double const& exact) +{ + typedef typename Polytope::PointType Point; + typedef typename Point::FT NT; + typedef typename Polytope::MT MT; + typedef typename Polytope::VT VT; + + int d = HP.dimension(); + + typedef BoostRandomNumberGenerator RNGType; + RNGType rng(d); + + VT x0(d); + x0 << 0.7566, 0.6374, 0.3981, 0.9248, 0.9828; + std::tuple res = inscribed_ellipsoid_rounding(HP, Point(x0)); + // Setup the parameters + int walk_len = 1; + NT e = 0.1; + + // Estimate the volume + std::cout << "Number type: " << typeid(NT).name() << std::endl; + + NT volume = std::get<2>(res) * volume_cooling_balls(HP, e, walk_len).second; + test_values(volume, expectedBilliard, exact); +} template void rounding_svd_test(Polytope &HP, @@ -326,6 +357,17 @@ void call_test_volumetric_barrier() { rounding_volumetric_barrier_test(P, 0, 3070.64, 3188.25, 3262.77, 3200.0); } +template +void call_test_vaidya_barrier() { + typedef Cartesian Kernel; + typedef typename Kernel::Point Point; + typedef HPolytope Hpolytope; + Hpolytope P; + + std::cout << "\n--- Testing vaidya barrier rounding of H-skinny_cube5" << std::endl; + P = generate_skinny_cube(5); + rounding_vaidya_barrier_test(P, 0, 3070.64, 3188.25, 3262.77, 3200.0); +} template void call_test_svd() { @@ -360,6 +402,10 @@ TEST_CASE("round_volumetric_barrier_test") { call_test_volumetric_barrier(); } +TEST_CASE("round_vaidya_barrier_test") { + call_test_vaidya_barrier(); +} + TEST_CASE("round_svd") { call_test_svd(); } diff --git a/test/test_internal_points.cpp b/test/test_internal_points.cpp index 50965128c..d0ee44008 100644 --- a/test/test_internal_points.cpp +++ b/test/test_internal_points.cpp @@ -184,6 +184,42 @@ void call_test_volumetric_center() { CHECK((Hessian - Hessian_sp).norm() < 1e-12); } +template +void call_test_vaidya_center() { + typedef Cartesian Kernel; + typedef typename Kernel::Point Point; + typedef HPolytope Hpolytope; + typedef typename Hpolytope::MT MT; + typedef typename Hpolytope::VT VT; + typedef boost::mt19937 PolyRNGType; + typedef Eigen::SparseMatrix SpMT; + Hpolytope P; + + std::cout << "\n--- Testing vaidya center for skinny H-polytope" << std::endl; + bool pre_rounding = true; // round random polytope before applying the skinny transformation + NT max_min_eig_ratio = NT(100); + P = skinny_random_hpoly(3, 15, pre_rounding, max_min_eig_ratio, 127); + P.normalize(); + + auto [Hessian, vaidya_center, converged] = barrier_center_ellipsoid_linear_ineq(P.get_mat(), P.get_vec()); + SpMT Asp = P.get_mat().sparseView(); + auto [Hessian_sp, vaidya_center2, converged2] = barrier_center_ellipsoid_linear_ineq(Asp, P.get_vec()); + + CHECK(P.is_in(Point(vaidya_center)) == -1); + CHECK(converged); + CHECK(std::abs(vaidya_center(0) + 2.4076) < 1e-04); + CHECK(std::abs(vaidya_center(1) + 2.34072) < 1e-04); + CHECK(std::abs(vaidya_center(2) - 3.97138) < 1e-04); + + CHECK(P.is_in(Point(vaidya_center2)) == -1); + CHECK(converged2); + CHECK(std::abs(vaidya_center(0) - vaidya_center2(0)) < 1e-12); + CHECK(std::abs(vaidya_center(1) - vaidya_center2(1)) < 1e-12); + CHECK(std::abs(vaidya_center(2) - vaidya_center2(2)) < 1e-12); + + CHECK((Hessian - Hessian_sp).norm() < 1e-12); +} + TEST_CASE("test_max_ball") { call_test_max_ball(); } @@ -200,6 +236,10 @@ TEST_CASE("test_volumetric_center") { call_test_volumetric_center(); } +TEST_CASE("test_vaidya_center") { + call_test_vaidya_center(); +} + TEST_CASE("test_max_ball_sparse") { call_test_max_ball_sparse(); }