summaryrefslogtreecommitdiff
path: root/src/Toplex_map/benchmark/benchmark_tm.cpp
blob: f132b783022d0b690df2ca9668105bfd3a72dd92 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
/*    This file is part of the Gudhi Library. The Gudhi library
 *    (Geometric Understanding in Higher Dimensions) is a generic C++
 *    library for computational topology.
 *
 *    Author:       François Godi, Vincent Rouvreau
 *
 *    Copyright (C) 2018  INRIA
 *
 *    Modification(s):
 *      - YYYY/MM Author: Description of the modification
 */

#include <iostream>
#include <random>
#include <chrono>

#include <gudhi/Simplex_tree.h>
#include <gudhi/Toplex_map.h>
#include <gudhi/Lazy_toplex_map.h>

using namespace Gudhi;

typedef Toplex_map::Simplex Simplex;
typedef Toplex_map::Vertex Vertex;
typedef std::pair<Simplex_tree<>::Simplex_handle, bool> typePairSimplexBool;

class ST_wrapper {
 public:
  void insert_simplex(const Simplex& tau) {
    /*std::cout << "insert_simplex - " << simplexTree.num_simplices() << " - ";
    for (auto v : tau)
      std::cout << v << ", ";
    std::cout << std::endl;
    */
    simplexTree.insert_simplex_and_subfaces(tau);
  }

  bool membership(const Simplex& tau) { return simplexTree.find(tau) != simplexTree.null_simplex(); }

  Vertex contraction(const Vertex x, const Vertex y) {
    // TODO (VR): edge contraction is not yet available for Simplex_tree
    return y;
  }

  std::size_t num_maximal_simplices() { return simplexTree.num_simplices(); }

 private:
  Simplex_tree<> simplexTree;
  void erase_max(const Simplex& sigma) {
    if (membership(sigma)) simplexTree.remove_maximal_simplex(simplexTree.find(sigma));
  }
};

int n = 300;

int nb_insert_simplex1 = 3000;
int nb_membership1 = 4000;
int nb_contraction = 300;
int nb_insert_simplex2 = 3000;
int nb_membership2 = 400000;

Simplex random_simplex(int n, std::size_t d) {
  std::random_device rd;
  std::mt19937 gen(rd());
  std::uniform_int_distribution<std::size_t> dis(1, n);
  Simplex s;
  while (s.size() < d) s.insert(dis(gen));
  return s;
}

std::vector<Simplex> r_vector_simplices(int n, int max_d, int m) {
  std::random_device rd;
  std::mt19937 gen(rd());
  std::uniform_int_distribution<std::size_t> dis(1, max_d);
  std::vector<Simplex> v;
  for (int i = 0; i < m; i++) v.push_back(random_simplex(n, dis(gen)));
  return v;
}

template <typename complex_type>
void chrono(int n, int d) {
  complex_type K;
  std::vector<Simplex> simplices_insert_simplex1 = r_vector_simplices(n, d, nb_insert_simplex1);
  std::vector<Simplex> simplices_membership1 = r_vector_simplices(n, d, nb_membership1);
  std::vector<Simplex> simplices_insert_simplex2 = r_vector_simplices(n - 2 * nb_contraction, d, nb_insert_simplex2);
  std::vector<Simplex> simplices_membership2 = r_vector_simplices(n - 2 * nb_contraction, d, nb_membership2);
  std::chrono::time_point<std::chrono::system_clock> start, end;

  for (const Simplex& s : simplices_insert_simplex1) K.insert_simplex(s);

  for (const Simplex& s : simplices_membership1) K.membership(s);

  start = std::chrono::system_clock::now();
  for (int i = 1; i <= nb_contraction; i++) K.contraction(n - 2 * i, n - 2 * i - 1);
  end = std::chrono::system_clock::now();
  auto c3 = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();

  start = std::chrono::system_clock::now();
  for (const Simplex& s : simplices_insert_simplex2) K.insert_simplex(s);
  end = std::chrono::system_clock::now();
  auto c1 = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();

  start = std::chrono::system_clock::now();
  for (const Simplex& s : simplices_membership2) K.membership(s);
  end = std::chrono::system_clock::now();
  auto c2 = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();

  if (c3 > 0)
    std::cout << c1 << "\t \t" << c2 << "\t \t" << c3 << "\t \t" << K.num_maximal_simplices() << std::endl;
  else
    std::cout << c1 << "\t \t" << c2 << "\t \tN/A\t \t" << K.num_maximal_simplices() << std::endl;
}

int main() {
  for (int d = 5; d <= 40; d += 5) {
    std::cout << "d=" << d << " \t  Insertions \t   Membership \t Contractions \t        Size" << std::endl;
    std::cout << "T Map \t \t";
    chrono<Toplex_map>(n, d);
    std::cout << "Lazy \t \t";
    chrono<Lazy_toplex_map>(n, d);
    if (d <= 15) {
      std::cout << "ST \t \t";
      chrono<ST_wrapper>(n, d);
    }
    std::cout << std::endl;
  }
}