ViennaLS
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
lsCheck.hpp
Go to the documentation of this file.
1#pragma once
2
3#include <ostream>
4#include <string>
5
6#include <hrleSparseStarIterator.hpp>
7
8#include <lsDomain.hpp>
9
10namespace viennals {
11
12using namespace viennacore;
13
14enum struct CheckStatusEnum : unsigned {
16 FAILED = 1,
18};
19
22template <class T, int D> class Check {
23 SmartPointer<Domain<T, D>> levelSet = nullptr;
25 std::string errors = "Level Set has not been checked yet!";
26 bool printMessage = false;
27
28 static std::string getDirectionString(int i) {
29 std::string result;
30 if (i < D) {
31 result += "-";
32 } else {
33 i -= D;
34 result += "+";
35 }
36 if (i == 0) {
37 result += "x";
38 } else if (i == 1) {
39 result += "y";
40 } else if (i == 2) {
41 result += "z";
42 } else {
43 Logger::getInstance().addError("Check: " + std::to_string(i) +
44 " is not a valid direction index!");
45 }
46 return result;
47 }
48
49 bool isNegative(T val) { return val <= -std::numeric_limits<T>::epsilon(); }
50
51 static int GetStatusFromDistance(T value) {
52 int x = static_cast<int>(value);
53 if (value >= 0.) {
54 return (value <= static_cast<T>(x) + static_cast<T>(0.5)) ? x : x + 1;
55 } else {
56 return (value >= static_cast<T>(x) - static_cast<T>(0.5)) ? x : x - 1;
57 }
58 }
59
60public:
61 Check() = default;
62
63 Check(SmartPointer<Domain<T, D>> passedLevelSet, bool print = false)
64 : levelSet(passedLevelSet), printMessage(print) {}
65
66 void setLevelSet(SmartPointer<Domain<T, D>> passedLevelSet) {
67 levelSet = passedLevelSet;
68 }
69
70 void setPrintMessage(bool print) { printMessage = print; }
71
72 CheckStatusEnum getStatus() const { return status; }
73
74 bool isValid() const { return status == CheckStatusEnum::SUCCESS; }
75
76 std::string what() const { return errors; }
77
78 void apply() {
79 if (levelSet == nullptr) {
80 Logger::getInstance()
81 .addWarning("No level set was passed to Check.")
82 .print();
83 return;
84 }
85
86 std::ostringstream oss;
87
88 for (viennahrle::ConstSparseStarIterator<typename Domain<T, D>::DomainType,
89 1>
90 it(levelSet->getDomain());
91 !it.isFinished(); it.next()) {
92
93 if (it.getCenter().isDefined()) {
94 for (int i = 0; i < 2 * D; ++i) {
95 if (it.getNeighbor(i).isDefined()) {
96 if (std::abs(GetStatusFromDistance(it.getCenter().getValue()) -
97 GetStatusFromDistance(it.getNeighbor(i).getValue())) >
98 1) {
99 oss << "The defined point " << it.getCenter().getStartIndices()
100 << " has an inconsistent defined neighbor in direction "
101 << getDirectionString(i) << "!" << std::endl;
102 oss.precision(24);
103 oss << "Value center point: " << it.getCenter().getValue()
104 << " Value neighbor point: " << it.getNeighbor(i).getValue()
105 << std::endl;
106 }
107 } else {
108 if (it.getNeighbor(i).getValue() >= 0.) { // TODO POS_SIGN
109 if (it.getCenter().getValue() < T(-0.5)) {
110 oss << "The defined point " << it.getCenter().getStartIndices()
111 << " has a level set value less than -0.5 but has an "
112 "undefined positive neighbor in direction "
113 << getDirectionString(i) << "!" << std::endl;
114 }
115 } else {
116 if (it.getCenter().getValue() > T(0.5)) {
117 oss << "The defined point " << it.getCenter().getStartIndices()
118 << " has a level set value greater than 0.5 but has an "
119 "undefined negative neighbor in direction "
120 << getDirectionString(i) << "!" << std::endl;
121 }
122 }
123 }
124 }
125 } else {
126 for (int i = 0; i < 2 * D; ++i) {
127 if (!it.getNeighbor(i).isDefined()) {
128 if (isNegative(it.getCenter().getValue()) !=
129 isNegative(it.getNeighbor(i).getValue())) {
130 oss << "The undefined run from "
131 << it.getCenter().getStartIndices() << " to "
132 << it.getCenter().getEndIndices()
133 << " has undefined neighbor grid points of opposite sign in "
134 "direction "
135 << getDirectionString(i) << "!" << std::endl;
136 }
137 }
138 }
139 }
140 }
141
142 // output any faults as error
143 if (std::string s = oss.str(); !s.empty()) {
145 errors = s;
146 if (printMessage) {
147 std::string message = "Report from Check:\n" + s;
148 Logger::getInstance().addError(s);
149 }
150 } else {
152 errors = "";
153 }
154 }
155};
156
157} // namespace viennals
CheckStatusEnum getStatus() const
Definition lsCheck.hpp:72
Check()=default
bool isValid() const
Definition lsCheck.hpp:74
void setPrintMessage(bool print)
Definition lsCheck.hpp:70
void apply()
Definition lsCheck.hpp:78
Check(SmartPointer< Domain< T, D > > passedLevelSet, bool print=false)
Definition lsCheck.hpp:63
void setLevelSet(SmartPointer< Domain< T, D > > passedLevelSet)
Definition lsCheck.hpp:66
std::string what() const
Definition lsCheck.hpp:76
Class containing all information about the level set, including the dimensions of the domain,...
Definition lsDomain.hpp:27
viennahrle::Domain< T, D > DomainType
Definition lsDomain.hpp:32
Definition lsAdvect.hpp:36
CheckStatusEnum
Definition lsCheck.hpp:14
@ FAILED
Definition lsCheck.hpp:16
@ UNCHECKED
Definition lsCheck.hpp:17
@ SUCCESS
Definition lsCheck.hpp:15
constexpr int D
Definition pyWrap.cpp:70
double T
Definition pyWrap.cpp:68