Skip to content

Commit a81983f

Browse files
committed
feat: unify implementation of polyhedra
1 parent bfc27d6 commit a81983f

13 files changed

+1218
-833
lines changed

include/overlap/overlap.hpp

+722-619
Large diffs are not rendered by default.

python/overlap/overlap.cpp

+30-33
Original file line numberDiff line numberDiff line change
@@ -18,18 +18,17 @@
1818
* along with this program. If not, see <http://www.gnu.org/licenses/>.
1919
*/
2020

21+
#include "overlap/overlap.hpp"
22+
23+
#include <pybind11/eigen.h>
24+
#include <pybind11/numpy.h>
25+
#include <pybind11/pybind11.h>
26+
#include <pybind11/stl.h>
27+
2128
#include <cctype>
2229
#include <locale>
2330
#include <map>
2431
#include <typeindex>
25-
#include <typeinfo>
26-
27-
#include "pybind11/eigen.h"
28-
#include "pybind11/numpy.h"
29-
#include "pybind11/pybind11.h"
30-
#include "pybind11/stl.h"
31-
32-
#include "overlap/overlap.hpp"
3332

3433
namespace py = pybind11;
3534

@@ -42,64 +41,62 @@ using namespace overlap;
4241

4342
template<typename Element,
4443
typename = std::enable_if_t<detail::is_element_v<Element>>>
45-
void createBindings(py::module& m) {
44+
void create_bindings(py::module& m) {
4645
static const auto element_names = std::map<std::type_index, std::string>{
4746
{std::type_index(typeid(Tetrahedron)), "Tetrahedron"},
4847
{std::type_index(typeid(Wedge)), "Wedge"},
4948
{std::type_index(typeid(Hexahedron)), "Hexahedron"}};
5049

5150
const auto& name = element_names.at(std::type_index(typeid(Element)));
52-
const auto nameLower =
51+
const auto name_lower =
5352
std::tolower(name.front(), std::locale{}) + name.substr(1);
5453

55-
static constexpr auto nrVertices = detail::num_vertices<Element>();
54+
static constexpr auto num_vertices = Element::num_vertices();
5655

5756
py::class_<Element>(m, name.c_str())
58-
.def(py::init<std::array<Vector, nrVertices>>(), py::arg("vertices"))
57+
.def(py::init<std::array<Vector, num_vertices>>(), py::arg("vertices"))
5958
.def(py::init([](py::array_t<double> vertices) {
6059
auto proxy = vertices.unchecked<2>();
61-
if (proxy.shape(0) != nrVertices || proxy.shape(1) != 3) {
60+
if (proxy.shape(0) != num_vertices || proxy.shape(1) != 3) {
6261
throw std::invalid_argument{
6362
"invalid shape for vertex list, must be (" +
64-
std::to_string(nrVertices) + ", 3)"};
63+
std::to_string(num_vertices) + ", 3)"};
6564
}
6665

67-
std::array<Vector, nrVertices> tmp{};
66+
std::array<Vector, num_vertices> tmp{};
6867
for (py::ssize_t v = 0; v < proxy.shape(0); ++v) {
6968
tmp[v] = Vector{proxy(v, 0), proxy(v, 1), proxy(v, 2)};
7069
}
7170

7271
return Element{std::move(tmp)};
7372
}),
7473
py::arg("vertices"))
75-
.def_readonly("vertices", &Element::vertices,
76-
"Return the vertices of the element.")
77-
.def_readonly("center", &Element::center,
78-
"Return the center point of the element.")
79-
.def_readonly("volume", &Element::volume,
80-
"Return the volume of the element.")
81-
.def_property_readonly(
82-
"surface_area",
83-
[](const Element& elem) { return elem.surface_area(); },
84-
"Return the surface area of the element.");
74+
.def_property_readonly("vertices", &Element::vertices,
75+
"Return the vertices of the element.")
76+
.def_property_readonly("center", &Element::center,
77+
"Return the center point of the element.")
78+
.def_property_readonly("volume", &Element::volume,
79+
"Return the volume of the element.")
80+
.def_property_readonly("surface_area", &Element::surface_area,
81+
"Return the surface area of the element.");
8582

8683
m.def(
8784
"overlap_volume",
8885
overload_cast_<const Sphere&, const Element&>()(&overlap_volume<Element>),
89-
"sphere"_a, py::arg{nameLower.c_str()},
90-
("Calculate the overlap volume of a sphere and a " + nameLower + ".")
86+
"sphere"_a, py::arg{name_lower.c_str()},
87+
("Calculate the overlap volume of a sphere and a " + name_lower + ".")
9188
.c_str());
9289

9390
m.def("overlap_area",
9491
overload_cast_<const Sphere&, const Element&>()(&overlap_area<Element>),
95-
"sphere"_a, py::arg{nameLower.c_str()},
96-
("Calculate the overlap area of a sphere and a " + nameLower + ".")
92+
"sphere"_a, py::arg{name_lower.c_str()},
93+
("Calculate the overlap area of a sphere and a " + name_lower + ".")
9794
.c_str());
9895
}
9996

10097
PYBIND11_MODULE(overlap, m) {
10198
m.doc() = R"pbdoc(
102-
Pybind11 example plugin
99+
Overlap Python bindings
103100
-----------------------
104101
105102
This originates fom CPP
@@ -128,7 +125,7 @@ PYBIND11_MODULE(overlap, m) {
128125
"surface_area", [](const Sphere& s) { return s.surface_area(); },
129126
"Return the surface area of the sphere.");
130127

131-
createBindings<Tetrahedron>(m);
132-
createBindings<Wedge>(m);
133-
createBindings<Hexahedron>(m);
128+
create_bindings<Tetrahedron>(m);
129+
create_bindings<Wedge>(m);
130+
create_bindings<Hexahedron>(m);
134131
}

tests/python/test_overlap.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,7 @@ def test_wedge(self):
113113

114114
np.testing.assert_almost_equal(overlap.overlap_area(sphere, wedge), [
115115
sphere.surface_area / 2,
116-
0., 0., 0., sphere.radius**2 * np.pi, 0.,
116+
0., 0., sphere.radius**2 * np.pi, 0., 0.,
117117
sphere.radius**2 * np.pi
118118
])
119119

tests/src/CMakeLists.txt

+3-3
Original file line numberDiff line numberDiff line change
@@ -74,9 +74,9 @@ set(unit_tests
7474
clamp
7575
contains
7676
decompose_elements
77-
elements
7877
normal_newell
7978
polygon
79+
polyhedron
8080
regularized_wedge
8181
regularized_wedge_area
8282
sphere
@@ -107,11 +107,11 @@ if(_have_fenv_h)
107107
endif()
108108
cmake_reset_check_state()
109109

110-
# activate additional compiler warnings
110+
# activate additional compiler warnings and keep frame pointers
111111
if(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Clang)" AND CMAKE_BUILD_TYPE MATCHES
112112
"(Debug|RelWithDebInfo)"
113113
)
114-
string(APPEND CMAKE_CXX_FLAGS " -Wall -Werror -Wextra")
114+
string(APPEND CMAKE_CXX_FLAGS "-Wall -Werror -Wextra -fno-omit-frame-pointer")
115115
endif()
116116

117117
# register the individual unit tests

tests/src/common.hpp

+5-23
Original file line numberDiff line numberDiff line change
@@ -21,21 +21,19 @@
2121
#ifndef OVERLAP_TEST_COMMON_HPP
2222
#define OVERLAP_TEST_COMMON_HPP
2323

24-
#include <iostream>
2524
#include <string_view>
2625

2726
#include <doctest/doctest.h>
2827

2928
class AssertionError : public std::runtime_error {
3029
public:
31-
static inline constexpr auto assertion_failed_msg =
32-
"[overlap] assertion failed: ";
30+
static constexpr auto assertion_failed_msg = "[overlap] assertion failed: ";
3331

3432
explicit AssertionError(std::string_view msg) :
3533
std::runtime_error{assertion_failed_msg + std::string{msg}} {}
3634
};
3735

38-
[[noreturn]] inline void throw_assertion_error(std::string_view msg) {
36+
[[noreturn]] void throw_assertion_error(std::string_view msg) {
3937
throw AssertionError{msg};
4038
}
4139

@@ -102,12 +100,6 @@ inline void validate_overlap_volume(const Sphere& s, const Hexahedron& hex,
102100
overlapCalcWedges += overlap_volume(s, wedge);
103101
}
104102

105-
std::cout << "volume hex: " << overlapCalcHex << std::endl;
106-
std::cout << "volume wedges: " << overlapCalcWedges << std::endl;
107-
std::cout << "volume tets5: " << overlapCalcTets5 << std::endl;
108-
std::cout << "volume tets6: " << overlapCalcTets6 << std::endl;
109-
std::cout << "volume tets24: " << overlapCalcTets24 << std::endl;
110-
111103
CHECK(overlapCalcHex == Approx(overlapCalcWedges).epsilon(epsilon));
112104
CHECK(overlapCalcHex == Approx(overlapCalcTets5).epsilon(epsilon));
113105
CHECK(overlapCalcHex == Approx(overlapCalcTets6).epsilon(epsilon));
@@ -129,15 +121,13 @@ inline void validate_overlap_area(const Sphere& s, const Hexahedron& hex,
129121

130122
const auto areaCalcHex = overlap_area(s, hex)[0];
131123

132-
auto areaCalcWedges = Scalar{0};
133124
auto areaCalcTets5 = Scalar{0};
134-
auto areaCalcTets6 = Scalar{0};
135-
auto areaCalcTets24 = Scalar{0};
136-
137125
for (const auto& tet : tets5) {
138126
areaCalcTets5 += overlap_area(s, tet)[0];
139127
}
140128

129+
auto areaCalcTets6 = Scalar{0};
130+
auto areaCalcTets24 = Scalar{0};
141131
for (const auto& tet : tets6) {
142132
std::array<Tetrahedron, 4> subTets;
143133
decompose(tet, subTets);
@@ -149,19 +139,11 @@ inline void validate_overlap_area(const Sphere& s, const Hexahedron& hex,
149139
areaCalcTets6 += overlap_area(s, tet)[0];
150140
}
151141

142+
auto areaCalcWedges = Scalar{0};
152143
for (const auto& wedge : wedges) {
153144
areaCalcWedges += overlap_area(s, wedge)[0];
154145
}
155146

156-
std::cout << "sphere center: [" << s.center.transpose()
157-
<< "], radius: " << s.radius << std::endl;
158-
159-
std::cout << "sphere surface hex: " << areaCalcHex << std::endl;
160-
std::cout << "sphere surface wedges: " << areaCalcWedges << std::endl;
161-
std::cout << "sphere surface tets5: " << areaCalcTets5 << std::endl;
162-
std::cout << "sphere surface tets6: " << areaCalcTets6 << std::endl;
163-
std::cout << "sphere surface tets24: " << areaCalcTets24 << std::endl;
164-
165147
CHECK(areaCalcHex == Approx(areaCalcWedges).epsilon(epsilon));
166148
CHECK(areaCalcHex == Approx(areaCalcTets5).epsilon(epsilon));
167149
CHECK(areaCalcHex == Approx(areaCalcTets6).epsilon(epsilon));

tests/src/test_decompose_elements.cpp

+9-9
Original file line numberDiff line numberDiff line change
@@ -35,28 +35,28 @@ TEST_SUITE("DecomposeElements") {
3535

3636
auto tets5Volume = Scalar{0};
3737
for (const auto& tet : tets5) {
38-
tets5Volume += tet.volume;
38+
tets5Volume += tet.volume();
3939
}
4040

4141
auto tets6Volume = Scalar{0};
4242
auto tets24Volume = Scalar{0};
4343
for (const auto& tet : tets6) {
4444
std::array<Tetrahedron, 4> subTets;
4545
decompose(tet, subTets);
46-
tets6Volume += tet.volume;
46+
tets6Volume += tet.volume();
4747

4848
for (const auto& subTet : subTets) {
49-
tets24Volume += subTet.volume;
49+
tets24Volume += subTet.volume();
5050
}
5151
}
5252

5353
constexpr auto epsilon =
5454
Scalar{5e2} * std::numeric_limits<Scalar>::epsilon();
5555

56-
REQUIRE(hex.volume == Approx(tets5Volume).epsilon(epsilon));
57-
REQUIRE(hex.volume == Approx(tets6Volume).epsilon(epsilon));
58-
REQUIRE(hex.volume == Approx(tets24Volume).epsilon(epsilon));
59-
REQUIRE(hex.volume ==
60-
Approx(wedges[0].volume + wedges[1].volume).epsilon(epsilon));
56+
REQUIRE(hex.volume() == Approx(tets5Volume).epsilon(epsilon));
57+
REQUIRE(hex.volume() == Approx(tets6Volume).epsilon(epsilon));
58+
REQUIRE(hex.volume() == Approx(tets24Volume).epsilon(epsilon));
59+
REQUIRE(hex.volume() ==
60+
Approx(wedges[0].volume() + wedges[1].volume()).epsilon(epsilon));
6161
}
62-
}
62+
}

tests/src/test_elements.cpp

-123
This file was deleted.

0 commit comments

Comments
 (0)