Merge branch 'fix_warnings' into 'master'

Some memory tracer fixes/features, fix build warnings

See merge request ngsolve/netgen!688
This commit is contained in:
Schöberl, Joachim 2024-12-03 20:33:27 +01:00
commit 27197f146c
28 changed files with 205 additions and 125 deletions

View File

@ -687,6 +687,7 @@ namespace ngcore
size_t allocsize; size_t allocsize;
/// that's the data we have to delete, nullptr for not owning the memory /// that's the data we have to delete, nullptr for not owning the memory
T * mem_to_delete; T * mem_to_delete;
MemoryTracer mt;
using FlatArray<T,IndexType>::size; using FlatArray<T,IndexType>::size;
@ -708,6 +709,7 @@ namespace ngcore
{ {
allocsize = asize; allocsize = asize;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*asize);
} }
@ -717,7 +719,10 @@ namespace ngcore
{ {
allocsize = asize; allocsize = asize;
if(ownMemory) if(ownMemory)
{
mem_to_delete = adata; mem_to_delete = adata;
mt.Alloc(sizeof(T)*asize);
}
else else
mem_to_delete = nullptr; mem_to_delete = nullptr;
} }
@ -733,8 +738,7 @@ namespace ngcore
NETGEN_INLINE Array (Array && a2) NETGEN_INLINE Array (Array && a2)
{ {
mt.Swap(0., a2.mt, sizeof(T) * a2.allocsize); mt = std::move(a2.mt);
size = a2.size; size = a2.size;
data = a2.data; data = a2.data;
allocsize = a2.allocsize; allocsize = a2.allocsize;
@ -753,6 +757,7 @@ namespace ngcore
{ {
allocsize = size; allocsize = size;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*size);
for (size_t i = 0; i < size; i++) for (size_t i = 0; i < size; i++)
data[i] = a2.data[i]; data[i] = a2.data[i];
} }
@ -772,6 +777,7 @@ namespace ngcore
{ {
allocsize = size; allocsize = size;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*size);
/* /*
for (size_t i = 0; i < size; i++) for (size_t i = 0; i < size; i++)
data[i] = a2[i]; data[i] = a2[i];
@ -788,6 +794,7 @@ namespace ngcore
{ {
allocsize = size; allocsize = size;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*size);
size_t cnt = 0; size_t cnt = 0;
for (auto val : list) for (auto val : list)
data[cnt++] = val; data[cnt++] = val;
@ -800,6 +807,7 @@ namespace ngcore
{ {
allocsize = size; allocsize = size;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*size);
for(size_t i = 0; i < a2.Size(); i++) for(size_t i = 0; i < a2.Size(); i++)
data[i] = a2[i]; data[i] = a2[i];
for (size_t i = a2.Size(), j=0; i < size; i++,j++) for (size_t i = a2.Size(), j=0; i < size; i++,j++)
@ -834,6 +842,9 @@ namespace ngcore
NETGEN_INLINE void NothingToDelete () NETGEN_INLINE void NothingToDelete ()
{ {
mem_to_delete = nullptr; mem_to_delete = nullptr;
// this memory is not managed by the Array anymore, so set the memory usage to 0
mt.Free(sizeof(T)*allocsize);
} }
/// Change logical size. If necessary, do reallocation. Keeps contents. /// Change logical size. If necessary, do reallocation. Keeps contents.
@ -1011,8 +1022,7 @@ namespace ngcore
/// steal array /// steal array
NETGEN_INLINE Array & operator= (Array && a2) NETGEN_INLINE Array & operator= (Array && a2)
{ {
mt.Swap(sizeof(T)*allocsize, a2.mt, sizeof(T)*a2.allocsize); mt = std::move(a2.mt);
ngcore::Swap (size, a2.size); ngcore::Swap (size, a2.size);
ngcore::Swap (data, a2.data); ngcore::Swap (data, a2.data);
ngcore::Swap (allocsize, a2.allocsize); ngcore::Swap (allocsize, a2.allocsize);
@ -1086,8 +1096,7 @@ namespace ngcore
NETGEN_INLINE void Swap (Array & b) NETGEN_INLINE void Swap (Array & b)
{ {
mt.Swap(sizeof(T) * allocsize, b.mt, sizeof(T) * b.allocsize); mt = std::move(b.mt);
ngcore::Swap (size, b.size); ngcore::Swap (size, b.size);
ngcore::Swap (data, b.data); ngcore::Swap (data, b.data);
ngcore::Swap (allocsize, b.allocsize); ngcore::Swap (allocsize, b.allocsize);
@ -1096,7 +1105,8 @@ namespace ngcore
NETGEN_INLINE void StartMemoryTracing () const NETGEN_INLINE void StartMemoryTracing () const
{ {
mt.Alloc(sizeof(T) * allocsize); if(mem_to_delete)
mt.Alloc(sizeof(T) * allocsize);
} }
const MemoryTracer& GetMemoryTracer() const { return mt; } const MemoryTracer& GetMemoryTracer() const { return mt; }
@ -1105,7 +1115,6 @@ namespace ngcore
/// resize array, at least to size minsize. copy contents /// resize array, at least to size minsize. copy contents
NETGEN_INLINE void ReSize (size_t minsize); NETGEN_INLINE void ReSize (size_t minsize);
MemoryTracer mt;
}; };
@ -1158,6 +1167,7 @@ namespace ngcore
using Array<T>::allocsize; using Array<T>::allocsize;
using Array<T>::data; using Array<T>::data;
using Array<T>::mem_to_delete; using Array<T>::mem_to_delete;
using Array<T>::mt;
// using Array<T>::ownmem; // using Array<T>::ownmem;
public: public:
@ -1171,6 +1181,7 @@ namespace ngcore
data = new T[asize]; data = new T[asize];
allocsize = size; allocsize = size;
mem_to_delete = data; mem_to_delete = data;
mt.Alloc(sizeof(T)*asize);
} }
} }
@ -1191,6 +1202,7 @@ namespace ngcore
ArrayMem(ArrayMem && a2) ArrayMem(ArrayMem && a2)
: Array<T> (a2.Size(), (T*)mem) : Array<T> (a2.Size(), (T*)mem)
{ {
mt = std::move(a2.mt);
if (a2.mem_to_delete) if (a2.mem_to_delete)
{ {
mem_to_delete = a2.mem_to_delete; mem_to_delete = a2.mem_to_delete;
@ -1233,6 +1245,7 @@ namespace ngcore
ArrayMem & operator= (ArrayMem && a2) ArrayMem & operator= (ArrayMem && a2)
{ {
mt = std::move(a2.mt);
ngcore::Swap (mem_to_delete, a2.mem_to_delete); ngcore::Swap (mem_to_delete, a2.mem_to_delete);
ngcore::Swap (allocsize, a2.allocsize); ngcore::Swap (allocsize, a2.allocsize);
ngcore::Swap (size, a2.size); ngcore::Swap (size, a2.size);

View File

@ -40,12 +40,13 @@ namespace ngcore
if (owns_data) if (owns_data)
{ {
delete [] data; delete [] data;
mt.Free(Addr(size)+1); mt.Free(GetMemoryUsage());
} }
size = asize; size = asize;
data = new unsigned char [Addr (size)+1]; data = new unsigned char [Addr (size)+1];
mt.Alloc(Addr(size)+1); owns_data = true;
mt.Alloc(GetMemoryUsage());
} }
BitArray & BitArray :: Set () throw() BitArray & BitArray :: Set () throw()

View File

@ -49,6 +49,7 @@ public:
{ {
ba2.owns_data = false; ba2.owns_data = false;
ba2.data = nullptr; ba2.data = nullptr;
mt = std::move(ba2.mt);
} }
template <typename T> template <typename T>
@ -59,13 +60,17 @@ public:
int cnt = 0; int cnt = 0;
for (auto i = list.begin(); i < list.end(); i++, cnt++) for (auto i = list.begin(); i < list.end(); i++, cnt++)
if (*i) SetBit(cnt); if (*i) SetBit(cnt);
StartMemoryTracing();
} }
/// delete data /// delete data
~BitArray () ~BitArray ()
{ {
if (owns_data) if (owns_data)
{
delete [] data; delete [] data;
mt.Free(GetMemoryUsage());
}
} }
/// Set size, loose values /// Set size, loose values
@ -150,11 +155,11 @@ public:
NGCORE_API auto * Data() const { return data; } NGCORE_API auto * Data() const { return data; }
const size_t GetMemoryUsage() const { return owns_data ? (size+CHAR_BIT-1)/CHAR_BIT : 0; }
const MemoryTracer& GetMemoryTracer() const { return mt; } const MemoryTracer& GetMemoryTracer() const { return mt; }
void StartMemoryTracing() const void StartMemoryTracing() const
{ {
if(owns_data) mt.Alloc(GetMemoryUsage());
mt.Alloc(Addr(size)+1);
} }
private: private:

View File

@ -50,7 +50,7 @@ namespace ngcore
RangeException :: RangeException (// const std::string & where, RangeException :: RangeException (// const std::string & where,
const char * where, const char * where,
int ind, int imin, int imax) : Exception("") ptrdiff_t ind, ptrdiff_t imin, ptrdiff_t imax) : Exception("")
{ {
std::stringstream str; std::stringstream str;
str << where << ": index " << ind << " out of range [" << imin << "," << imax << ")\n"; str << where << ": index " << ind << " out of range [" << imin << "," << imax << ")\n";
@ -59,7 +59,7 @@ namespace ngcore
} }
void ThrowRangeException(const char * s, int ind, int imin, int imax) void ThrowRangeException(const char * s, ptrdiff_t ind, ptrdiff_t imin, ptrdiff_t imax)
{ {
throw RangeException(s, ind, imin, imax); throw RangeException(s, ind, imin, imax);
} }
@ -75,7 +75,7 @@ namespace ngcore
} }
void ThrowNotTheSameException(const char * s, long int a, long int b) void ThrowNotTheSameException(const char * s, ptrdiff_t a, ptrdiff_t b)
{ {
throw ngcore::Exception(std::string(s) + ", a="+ToString(a) + ", b="+ToString(b) + GetBackTrace()); throw ngcore::Exception(std::string(s) + ", a="+ToString(a) + ", b="+ToString(b) + GetBackTrace());
} }

View File

@ -1,6 +1,7 @@
#ifndef NETGEN_CORE_EXCEPTION_HPP #ifndef NETGEN_CORE_EXCEPTION_HPP
#define NETGEN_CORE_EXCEPTION_HPP #define NETGEN_CORE_EXCEPTION_HPP
#include <cstddef>
#include <sstream> // for stringstream #include <sstream> // for stringstream
#include <stdexcept> // for exception #include <stdexcept> // for exception
#include <string> // for string #include <string> // for string
@ -66,7 +67,7 @@ namespace ngcore
/// where it occurs, index, minimal and maximal indices /// where it occurs, index, minimal and maximal indices
RangeException (// const std::string & where, RangeException (// const std::string & where,
const char * where, const char * where,
int ind, int imin, int imax); ptrdiff_t ind, ptrdiff_t imin, ptrdiff_t imax);
/* /*
: Exception("") : Exception("")
{ {
@ -85,8 +86,8 @@ namespace ngcore
} }
}; };
[[noreturn]] NGCORE_API void ThrowRangeException(const char * s, int ind, int imin, int imax); [[noreturn]] NGCORE_API void ThrowRangeException(const char * s, ptrdiff_t ind, ptrdiff_t imin, ptrdiff_t imax);
[[noreturn]] NGCORE_API void ThrowNotTheSameException(const char * s, long int a, long int b); [[noreturn]] NGCORE_API void ThrowNotTheSameException(const char * s, ptrdiff_t a, ptrdiff_t b);
// Exception used if no simd implementation is available to fall back to standard evaluation // Exception used if no simd implementation is available to fall back to standard evaluation
@ -98,12 +99,12 @@ namespace ngcore
constexpr operator bool() const { return false; } }; constexpr operator bool() const { return false; } };
namespace detail { namespace detail {
template <typename T> template <typename T, typename Tmin, typename Tmax>
inline static void CheckRange(const char * s, const T& n, int first, int next) inline static void CheckRange(const char * s, const T& n, Tmin first, Tmax next)
{ {
if constexpr (!IsSafe<decltype(n)>()) if constexpr (!IsSafe<decltype(n)>())
if (n<first || n>=next) if (n<first || n>=next)
ThrowRangeException(s, int(n), first, next); ThrowRangeException(s, ptrdiff_t(n), ptrdiff_t(first), ptrdiff_t(next));
} }
template <typename Ta, typename Tb> template <typename Ta, typename Tb>
@ -112,10 +113,10 @@ namespace ngcore
if constexpr (!IsSafe<decltype(a)>() || !IsSafe<decltype(b)>()) if constexpr (!IsSafe<decltype(a)>() || !IsSafe<decltype(b)>())
if(a != b) if(a != b)
{ {
if constexpr(std::is_same<decltype(a),size_t>() && std::is_same<decltype(b),size_t>()) if constexpr(std::is_integral_v<decltype(a)> && std::is_same_v<decltype(a),decltype(b)>)
ThrowNotTheSameException(s, long(a), long(b)); \ ThrowNotTheSameException(s, long(a), long(b)); \
else else
throw Exception(std::string(s) + "\t: not the same"+ToString(a) + ", b="+ngcore::ToString(b) + GetBackTrace()); throw Exception(std::string(s) + "\t: not the same, a="+ToString(a) + ", b="+ngcore::ToString(b) + GetBackTrace());
} }
} }
} // namespace detail } // namespace detail
@ -128,7 +129,7 @@ namespace ngcore
#define NG_EXCEPTION(s) ngcore::Exception(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t"+std::string(s)) #define NG_EXCEPTION(s) ngcore::Exception(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t"+std::string(s))
#if defined(NETGEN_ENABLE_CHECK_RANGE) && !defined(__CUDA_ARCH__) #if defined(NETGEN_ENABLE_CHECK_RANGE) && !defined(__CUDA_ARCH__)
#define NETGEN_CHECK_RANGE(value, min, max_plus_one) ngcore::detail::CheckRange(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t", value, int(min), int(max_plus_one)); #define NETGEN_CHECK_RANGE(value, min, max_plus_one) ngcore::detail::CheckRange(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t", value, min, max_plus_one);
#define NETGEN_CHECK_SAME(a,b) ngcore::detail::CheckSame(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t", a, b); #define NETGEN_CHECK_SAME(a,b) ngcore::detail::CheckSame(__FILE__ ":" NETGEN_CORE_NGEXEPTION_STR(__LINE__) "\t", a, b);
#define NETGEN_NOEXCEPT #define NETGEN_NOEXCEPT

View File

@ -35,11 +35,16 @@ namespace ngcore
class MemoryTracer class MemoryTracer
{ {
#if defined(NETGEN_TRACE_MEMORY) && !defined(__CUDA_ARCH__) #if defined(NETGEN_TRACE_MEMORY) && !defined(__CUDA_ARCH__)
NGCORE_API static std::vector<std::string> names; NGCORE_API static std::vector<std::string> names;
NGCORE_API static std::vector<int> parents; NGCORE_API static std::vector<int> parents;
static int CreateId(const std::string& name) #if defined(NETGEN_CHECK_RANGE)
NGCORE_API static std::atomic<size_t> total_memory;
mutable size_t allocated_memory = 0;
#endif // NETGEN_CHECK_RANGE
static int CreateId(const std::string& name = "")
{ {
int id = names.size(); int id = names.size();
names.push_back(name); names.push_back(name);
@ -48,7 +53,7 @@ namespace ngcore
std::cerr << "Allocated " << id << " MemoryTracer objects" << std::endl; std::cerr << "Allocated " << id << " MemoryTracer objects" << std::endl;
return id; return id;
} }
int id; mutable int id = 0;
public: public:
@ -57,8 +62,33 @@ namespace ngcore
id = CreateId(name); id = CreateId(name);
} }
// not tracing MemoryTracer() { }
MemoryTracer() : id(0) {}
MemoryTracer(const MemoryTracer & tracer)
{
(*this) = tracer;
}
MemoryTracer(MemoryTracer && tracer)
{
(*this) = std::move(tracer);
}
MemoryTracer & operator=(const MemoryTracer & tracer) {
if(tracer.id)
id = CreateId(names[tracer.id]);
return *this;
}
MemoryTracer & operator=(MemoryTracer && tracer) {
ngcore::Swap(id, tracer.id);
#if defined(NETGEN_CHECK_RANGE)
ngcore::Swap(allocated_memory, tracer.allocated_memory);
#endif // NETGEN_CHECK_RANGE
return *this;
}
template <typename... TRest> template <typename... TRest>
MemoryTracer( std::string name, TRest & ... rest ) MemoryTracer( std::string name, TRest & ... rest )
@ -67,38 +97,48 @@ namespace ngcore
Track(rest...); Track(rest...);
} }
#if defined(NETGEN_CHECK_RANGE)
// check if all memory was freed when object is destroyed
~MemoryTracer()
{
NETGEN_CHECK_SAME(allocated_memory, 0);
}
#endif // NETGEN_CHECK_RANGE
NETGEN_INLINE void Alloc(size_t size) const NETGEN_INLINE void Alloc(size_t size) const
{ {
#if defined(NETGEN_CHECK_RANGE)
// Trace also nameless Memtracer objects if range checks are active
if(!id && size)
id = CreateId();
#endif // NETGEN_CHECK_RANGE
if(id && trace) if(id && trace)
trace->AllocMemory(id, size); trace->AllocMemory(id, size);
#if defined(NETGEN_CHECK_RANGE)
if(id)
{
allocated_memory += size;
total_memory += size;
}
#endif // NETGEN_CHECK_RANGE
} }
void Free(size_t size) const void Free(size_t size) const
{ {
if(id && trace) if(id && trace)
trace->FreeMemory(id, size); trace->FreeMemory(id, size);
}
void Swap(size_t mysize, MemoryTracer& other, size_t other_size) const #if defined(NETGEN_CHECK_RANGE)
{ if(id)
if(!trace || (id == 0 && other.id == 0)) {
return; // check if we have at least size bytes of memory currently allocated (such that allocated_memory doesn't get negative)
if(id == 0) NETGEN_CHECK_RANGE(allocated_memory, static_cast<ptrdiff_t>(size), std::numeric_limits<ptrdiff_t>::max());
return trace->ChangeMemory(other.id, mysize - other_size); allocated_memory -= size;
if(other.id == 0) total_memory -= size;
return trace->ChangeMemory(id, other_size - mysize); #endif // NETGEN_CHECK_RANGE
}
// first decrease memory, otherwise have artificial/wrong high peak memory usage
if(mysize<other_size)
{
trace->ChangeMemory(other.id, mysize-other_size);
trace->ChangeMemory(id, other_size-mysize);
}
else
{
trace->ChangeMemory(id, other_size-mysize);
trace->ChangeMemory(other.id, mysize-other_size);
}
} }
int GetId() const { return id; } int GetId() const { return id; }
@ -148,6 +188,14 @@ namespace ngcore
static const std::vector<std::string> & GetNames() { return names; } static const std::vector<std::string> & GetNames() { return names; }
static const std::vector<int> & GetParents() { return parents; } static const std::vector<int> & GetParents() { return parents; }
static size_t GetTotalMemory()
{
#if defined(NETGEN_CHECK_RANGE)
return total_memory;
#else
return 0;
#endif // NETGEN_CHECK_RANGE
}
#else // defined(NETGEN_TRACE_MEMORY) && !defined(__CUDA_ARCH__) #else // defined(NETGEN_TRACE_MEMORY) && !defined(__CUDA_ARCH__)
public: public:
MemoryTracer() {} MemoryTracer() {}
@ -157,7 +205,6 @@ namespace ngcore
void Alloc(size_t /* size */) const {} void Alloc(size_t /* size */) const {}
void Free(size_t /* size */) const {} void Free(size_t /* size */) const {}
void Swap(...) const {}
int GetId() const { return 0; } int GetId() const { return 0; }
template <typename... TRest> template <typename... TRest>
@ -166,6 +213,7 @@ namespace ngcore
static std::string GetName(int /* id */) { return ""; } static std::string GetName(int /* id */) { return ""; }
std::string GetName() const { return ""; } std::string GetName() const { return ""; }
void SetName(std::string /* name */) const {} void SetName(std::string /* name */) const {}
static size_t GetTotalMemory() { return 0; }
#endif // NETGEN_TRACE_MEMORY #endif // NETGEN_TRACE_MEMORY
}; };
} // namespace ngcore } // namespace ngcore

View File

@ -116,6 +116,7 @@ namespace ngcore
#ifdef NETGEN_TRACE_MEMORY #ifdef NETGEN_TRACE_MEMORY
std::vector<std::string> MemoryTracer::names{"all"}; std::vector<std::string> MemoryTracer::names{"all"};
std::vector<int> MemoryTracer::parents{-1}; std::vector<int> MemoryTracer::parents{-1};
std::atomic<size_t> MemoryTracer::total_memory{0};
#endif // NETGEN_TRACE_MEMORY #endif // NETGEN_TRACE_MEMORY
} // namespace ngcore } // namespace ngcore

View File

@ -324,6 +324,8 @@ threads : int
#endif // NETGEN_TRACE_MEMORY #endif // NETGEN_TRACE_MEMORY
; ;
m.def("GetTotalMemory", MemoryTracer::GetTotalMemory);
py::class_<Timer<>> (m, "Timer") py::class_<Timer<>> (m, "Timer")
.def(py::init<const string&>()) .def(py::init<const string&>())
.def("Start", static_cast<void (Timer<>::*)()const>(&Timer<>::Start), "start timer") .def("Start", static_cast<void (Timer<>::*)()const>(&Timer<>::Start), "start timer")

View File

@ -130,6 +130,7 @@ namespace ngcore
{ {
for (size_t i : IntRange(size+1)) for (size_t i : IntRange(size+1))
index[i] = i*entrysize; index[i] = i*entrysize;
mt.Alloc(GetMemUsage());
} }
/// Construct table of variable entrysize /// Construct table of variable entrysize
@ -141,6 +142,7 @@ namespace ngcore
index = TablePrefixSum (FlatArray<TI> (entrysize.Size(), entrysize.Data())); index = TablePrefixSum (FlatArray<TI> (entrysize.Size(), entrysize.Data()));
size_t cnt = index[size]; size_t cnt = index[size];
data = new T[cnt]; data = new T[cnt];
mt.Alloc(GetMemUsage());
} }
explicit NETGEN_INLINE Table (const FlatTable<T,IndexType> & tab2) explicit NETGEN_INLINE Table (const FlatTable<T,IndexType> & tab2)
@ -157,6 +159,7 @@ namespace ngcore
size_t cnt = index[size]; size_t cnt = index[size];
data = new T[cnt]; data = new T[cnt];
this->AsArray() = tab2.AsArray(); this->AsArray() = tab2.AsArray();
mt.Alloc(GetMemUsage());
/* /*
for (size_t i = 0; i < cnt; i++) for (size_t i = 0; i < cnt; i++)
data[i] = tab2.data[i]; data[i] = tab2.data[i];
@ -177,12 +180,14 @@ namespace ngcore
data = new T[cnt]; data = new T[cnt];
for (size_t i = 0; i < cnt; i++) for (size_t i = 0; i < cnt; i++)
data[i] = tab2.data[i]; data[i] = tab2.data[i];
mt.Alloc(GetMemUsage());
} }
NETGEN_INLINE Table (Table && tab2) NETGEN_INLINE Table (Table && tab2)
: FlatTable<T,IndexType>(0, nullptr, nullptr) : FlatTable<T,IndexType>(0, nullptr, nullptr)
{ {
tab2.mt.Free(tab2.GetMemUsage()); mt = std::move(tab2.mt);
Swap (size, tab2.size); Swap (size, tab2.size);
Swap (index, tab2.index); Swap (index, tab2.index);
Swap (data, tab2.data); Swap (data, tab2.data);
@ -210,7 +215,7 @@ namespace ngcore
NETGEN_INLINE Table & operator= (Table && tab2) NETGEN_INLINE Table & operator= (Table && tab2)
{ {
mt.Swap(GetMemUsage(), tab2.mt, tab2.GetMemUsage()); mt = std::move(tab2.mt);
Swap (size, tab2.size); Swap (size, tab2.size);
Swap (index, tab2.index); Swap (index, tab2.index);
Swap (data, tab2.data); Swap (data, tab2.data);

View File

@ -1211,7 +1211,7 @@ namespace netgen
PrintMessage (2, "Object ", i, " has ", tams->GetNT(), " triangles"); PrintMessage (2, "Object ", i, " has ", tams->GetNT(), " triangles");
} }
} }
catch (exception) catch (const std::exception &)
{ {
cerr << "*************************************************************" << endl cerr << "*************************************************************" << endl
<< "**** out of memory problem in CSG visualization ****" << endl << "**** out of memory problem in CSG visualization ****" << endl

View File

@ -109,7 +109,7 @@ namespace netgen
{ {
void * p = new char [(line.maxsize+5) * elsize]; void * p = new char [(line.maxsize+5) * elsize];
if (line.maxsize*elsize) if (line.maxsize && elsize)
memcpy (p, line.col, line.maxsize * elsize); memcpy (p, line.col, line.maxsize * elsize);
delete [] (char*)line.col; delete [] (char*)line.col;

View File

@ -2246,11 +2246,8 @@ int Ng_GetClosureNodes (int nt, int nodenr, int nodeset, int * nodes)
if (nodeset & 2) // Edges if (nodeset & 2) // Edges
{ {
int edges[12]; auto edges = mesh->GetTopology().GetEdges (ElementIndex(nodenr));
// int ned; for (int i = 0; i < edges.Size(); i++)
// ned = mesh->GetTopology().GetElementEdges (nodenr+1, edges, 0);
int ned = mesh->GetTopology().GetEdges (ElementIndex(nodenr)).Size();
for (int i = 0; i < ned; i++)
{ {
nodes[cnt++] = 1; nodes[cnt++] = 1;
nodes[cnt++] = edges[i]-1; nodes[cnt++] = edges[i]-1;

View File

@ -1041,7 +1041,7 @@ namespace netgen
build_searchtree); build_searchtree);
return ind - 1; return ind - 1;
} }
catch(NgException e) // quads not implemented curved yet catch(const NgException & e) // quads not implemented curved yet
{ {
for (SegmentIndex si = 0; si < mesh->GetNSeg(); si++) for (SegmentIndex si = 0; si < mesh->GetNSeg(); si++)
{ {

View File

@ -301,7 +301,7 @@ namespace netgen
in >> name; in >> name;
cout << IM(3) << len << " element are in group " << name << endl; cout << IM(3) << len << " element are in group " << name << endl;
int hi, index; int hi, index;
int fdnr, ednr; int fdnr=-1, ednr=-1;
in >> hi >> index >> hi >> hi; in >> hi >> index >> hi >> hi;
int codim = get<1>(element_map[index]); int codim = get<1>(element_map[index]);
@ -712,7 +712,7 @@ namespace netgen
if(!UserFormatRegister::HaveFormat(format)) if(!UserFormatRegister::HaveFormat(format))
throw Exception("Unknown format: " + format); throw Exception("Unknown format: " + format);
const auto & entry = UserFormatRegister::Get(format); const auto entry = UserFormatRegister::Get(format);
if(!entry.read) if(!entry.read)
throw Exception("Reading format " + format + " is not implemented"); throw Exception("Reading format " + format + " is not implemented");

View File

@ -53,8 +53,8 @@ void ReadMeditFormat (Mesh & mesh, const filesystem::path & filename, map<tuple<
for([[maybe_unused]] auto k : Range(nvert)) { for([[maybe_unused]] auto k : Range(nvert)) {
for(auto i : Range(dim)) for(auto i : Range(dim))
fin >> p[i]; fin >> p[i];
fin >> index; fin >> index;
mesh.AddPoint(p); mesh.AddPoint(p);
} }
} }
else if(token == "Edges") { else if(token == "Edges") {
@ -64,10 +64,10 @@ void ReadMeditFormat (Mesh & mesh, const filesystem::path & filename, map<tuple<
for([[maybe_unused]] auto k : Range(nedge)) { for([[maybe_unused]] auto k : Range(nedge)) {
for(auto i : Range(2)) for(auto i : Range(2))
fin >> seg[i]; fin >> seg[i];
fin >> seg.edgenr; fin >> seg.edgenr;
seg.edgenr = getIndex(1, seg.edgenr); seg.edgenr = getIndex(1, seg.edgenr);
seg.si = seg.edgenr; seg.si = seg.edgenr;
mesh.AddSegment(seg); mesh.AddSegment(seg);
} }
} }
else if(token == "Triangles") { else if(token == "Triangles") {
@ -77,9 +77,9 @@ void ReadMeditFormat (Mesh & mesh, const filesystem::path & filename, map<tuple<
for([[maybe_unused]] auto k : Range(ntrig)) { for([[maybe_unused]] auto k : Range(ntrig)) {
for(auto i : Range(3)) for(auto i : Range(3))
fin >> sel[i]; fin >> sel[i];
fin >> index; fin >> index;
sel.SetIndex(getIndex(2, index)); sel.SetIndex(getIndex(2, index));
mesh.AddSurfaceElement(sel); mesh.AddSurfaceElement(sel);
} }
} }
else if(token == "Tetrahedra") { else if(token == "Tetrahedra") {
@ -89,10 +89,10 @@ void ReadMeditFormat (Mesh & mesh, const filesystem::path & filename, map<tuple<
for([[maybe_unused]] auto k : Range(ntet)) { for([[maybe_unused]] auto k : Range(ntet)) {
for(auto i : Range(4)) for(auto i : Range(4))
fin >> el[i]; fin >> el[i];
fin >> index; fin >> index;
el.SetIndex(getIndex(3, index)); el.SetIndex(getIndex(3, index));
el.Invert(); el.Invert();
mesh.AddVolumeElement(el); mesh.AddVolumeElement(el);
} }
} }
else if(token == "Corners") { else if(token == "Corners") {

View File

@ -39,7 +39,7 @@ bool WriteUserFormat (const string & format,
if(!UserFormatRegister::HaveFormat(format)) if(!UserFormatRegister::HaveFormat(format))
return true; return true;
const auto & entry = UserFormatRegister::Get(format); const auto entry = UserFormatRegister::Get(format);
if(!entry.write) if(!entry.write)
return true; return true;

View File

@ -49,7 +49,7 @@ namespace netgen
{ {
data = NULL; height = width = 0; data = NULL; height = width = 0;
SetSize (m2.Height(), m2.Width()); SetSize (m2.Height(), m2.Width());
if (Height()*Width()) if (Height() && Width())
memcpy (data, m2.data, sizeof(double) * (Height() * Width())); memcpy (data, m2.data, sizeof(double) * (Height() * Width()));
} }
@ -70,7 +70,7 @@ namespace netgen
delete[] data; delete[] data;
if (h*w) if (h && w)
data = new double[h*w]; data = new double[h*w];
else else
data = NULL; data = NULL;

View File

@ -788,9 +788,9 @@ namespace netgen
if(fabs(mat.Det()) > 1e-6) if(fabs(mat.Det()) > 1e-6)
break; break;
} }
int maxpos1; int maxpos1 = 0;
int maxpos2; int maxpos2 = 1;
double val = 0; double val = ns[0] * ns[1];
for (auto i : Range(ns)) for (auto i : Range(ns))
{ {
for (auto j : Range(i + 1, ns.Size())) for (auto j : Range(i + 1, ns.Size()))

View File

@ -1066,7 +1066,8 @@ double MeshOptimize3d :: SwapImproveEdge (
bad3 += GetLegalPenalty(); bad3 += GetLegalPenalty();
} }
bool swap2, swap3; bool swap2=false;
bool swap3=false;
if (goal == OPT_CONFORM) if (goal == OPT_CONFORM)
{ {
@ -2608,7 +2609,7 @@ double MeshOptimize3d :: SplitImprove2Element (
// search for very flat tets, with two disjoint edges nearly crossing, like a rectangle with diagonals // search for very flat tets, with two disjoint edges nearly crossing, like a rectangle with diagonals
int minedge = -1; int minedge = -1;
double mindist = 1e99; double mindist = 1e99;
double minlam0, minlam1; double minlam0=0, minlam1=0;
for (int i : Range(3)) for (int i : Range(3))
{ {

View File

@ -196,8 +196,8 @@ namespace netgen
auto& el = mesh.SurfaceElement(velement); auto& el = mesh.SurfaceElement(velement);
if(el.GetType() == TRIG) if(el.GetType() == TRIG)
{ {
double seg_lam; double seg_lam=-1;
double lam; double lam=-1;
auto seg = mesh.LineSegment(segs[i]); auto seg = mesh.LineSegment(segs[i]);
for(auto k : Range(3)) for(auto k : Range(3))
{ {

View File

@ -396,8 +396,8 @@ namespace netgen
for (int i = 1; i <= mesh.GetNOpenElements(); i++) for (int i = 1; i <= mesh.GetNOpenElements(); i++)
md.meshing->AddBoundaryElement (mesh.OpenElement(i)); md.meshing->AddBoundaryElement (mesh.OpenElement(i));
if (mp.delaunay && mesh.GetNOpenElements()) if (mp.delaunay && mesh.GetNOpenElements())
{ {
int oldne = mesh.GetNE(); int oldne = mesh.GetNE();
md.meshing->Delaunay (mesh, domain, mp); md.meshing->Delaunay (mesh, domain, mp);
@ -408,22 +408,22 @@ namespace netgen
PrintMessage (3, mesh.GetNP(), " points, ", PrintMessage (3, mesh.GetNP(), " points, ",
mesh.GetNE(), " elements"); mesh.GetNE(), " elements");
mesh.FindOpenElements(domain); mesh.FindOpenElements(domain);
} }
Box<3> domain_bbox( Box<3>::EMPTY_BOX ); Box<3> domain_bbox( Box<3>::EMPTY_BOX );
for (auto & sel : mesh.SurfaceElements()) for (auto & sel : mesh.SurfaceElements())
{ {
if (sel.IsDeleted() ) continue; if (sel.IsDeleted() ) continue;
for (auto pi : sel.PNums()) for (auto pi : sel.PNums())
domain_bbox.Add (mesh[pi]); domain_bbox.Add (mesh[pi]);
} }
domain_bbox.Increase (0.01 * domain_bbox.Diam()); domain_bbox.Increase (0.01 * domain_bbox.Diam());
int cntsteps = 0; int cntsteps = 0;
int meshed; int meshed;
if (mesh.GetNOpenElements()) if (mesh.GetNOpenElements())
do do
{ {
if (multithread.terminate) if (multithread.terminate)
@ -515,24 +515,22 @@ namespace netgen
PrintMessage (1, "Success !"); PrintMessage (1, "Success !");
} }
} }
while (!meshed); while (!meshed);
{
PrintMessage (3, "Check subdomain ", domain, " / ", mesh.GetNDomains());
mesh.FindOpenElements(domain); PrintMessage (3, "Check subdomain ", domain, " / ", mesh.GetNDomains());
bool res = (mesh.CheckConsistentBoundary() != 0); mesh.FindOpenElements(domain);
if (res)
{ bool res = (mesh.CheckConsistentBoundary() != 0);
if(debugparam.write_mesh_on_error) if (res)
md.mesh->Save("inconsistent_surface_domain_"+ToString(md.domain)+".vol.gz"); {
PrintError ("Surface mesh not consistent"); if(debugparam.write_mesh_on_error)
throw NgException ("Stop meshing since surface mesh not consistent"); md.mesh->Save("inconsistent_surface_domain_"+ToString(md.domain)+".vol.gz");
} PrintError ("Surface mesh not consistent");
} throw NgException ("Stop meshing since surface mesh not consistent");
RemoveIllegalElements (mesh, domain); }
ConformToFreeSegments (mesh, domain); RemoveIllegalElements (mesh, domain);
ConformToFreeSegments (mesh, domain);
} }
void MergeMeshes( Mesh & mesh, Array<MeshingData> & md ) void MergeMeshes( Mesh & mesh, Array<MeshingData> & md )

View File

@ -175,7 +175,8 @@ namespace netgen
} }
constexpr PointIndex (t_invalid inv) : i(PointIndex::BASE-1) { ; } constexpr PointIndex (t_invalid inv) : i(PointIndex::BASE-1) { ; }
// PointIndex & operator= (const PointIndex &ai) { i = ai.i; return *this; } // PointIndex & operator= (const PointIndex &ai) { i = ai.i; return *this; }
constexpr operator int () const { return i; } constexpr operator const int& () const { return i; }
explicit constexpr operator int& () { return i; }
PointIndex operator++ (int) { PointIndex hi(*this); i++; return hi; } PointIndex operator++ (int) { PointIndex hi(*this); i++; return hi; }
PointIndex operator-- (int) { PointIndex hi(*this); i--; return hi; } PointIndex operator-- (int) { PointIndex hi(*this); i--; return hi; }
PointIndex & operator++ () { i++; return *this; } PointIndex & operator++ () { i++; return *this; }

View File

@ -267,7 +267,7 @@ DLL_HEADER void ExportNetgenMeshing(py::module &m)
.def(py::init<int>()) .def(py::init<int>())
.def("__repr__", &ToString<PointIndex>) .def("__repr__", &ToString<PointIndex>)
.def("__str__", &ToString<PointIndex>) .def("__str__", &ToString<PointIndex>)
.def_property_readonly("nr", &PointIndex::operator int) .def_property_readonly("nr", &PointIndex::operator const int&)
.def("__eq__" , FunctionPointer( [](PointIndex &self, PointIndex &other) .def("__eq__" , FunctionPointer( [](PointIndex &self, PointIndex &other)
{ return static_cast<int>(self)==static_cast<int>(other); }) ) { return static_cast<int>(self)==static_cast<int>(other); }) )
.def("__hash__" , FunctionPointer( [](PointIndex &self ) { return static_cast<int>(self); }) ) .def("__hash__" , FunctionPointer( [](PointIndex &self ) { return static_cast<int>(self); }) )

View File

@ -224,7 +224,7 @@ NGCORE_API_EXPORT void ExportSTL(py::module & m)
.def("GetVicinity", [] (shared_ptr<STLGeometry> self, int node, int size, string type) { .def("GetVicinity", [] (shared_ptr<STLGeometry> self, int node, int size, string type) {
NgArray<int> vic; NgArray<int> vic;
int trig; int trig=-1;
if(type == "trig") if(type == "trig")
trig = node; trig = node;

View File

@ -266,7 +266,7 @@ namespace netgen
} }
catch (bad_weak_ptr e) catch (const bad_weak_ptr & e)
{ {
// cout << "don't have a mesh to visualize" << endl; // cout << "don't have a mesh to visualize" << endl;
VisualScene::DrawScene(); VisualScene::DrawScene();
@ -895,7 +895,7 @@ namespace netgen
vstimestamp = meshtimestamp; vstimestamp = meshtimestamp;
} }
catch (bad_weak_ptr e) catch (const bad_weak_ptr & e)
{ {
PrintMessage (3, "vsmesh::buildscene: don't have a mesh to visualize"); PrintMessage (3, "vsmesh::buildscene: don't have a mesh to visualize");
VisualScene::BuildScene (zoomall); VisualScene::BuildScene (zoomall);

View File

@ -641,7 +641,7 @@ namespace netgen
// delete lock; // delete lock;
// mem_lock.UnLock(); // mem_lock.UnLock();
} }
catch (bad_weak_ptr e) catch (const bad_weak_ptr & e)
{ {
// cout << "don't have a mesh to visualize" << endl; // cout << "don't have a mesh to visualize" << endl;
VisualScene::DrawScene(); VisualScene::DrawScene();
@ -1120,7 +1120,7 @@ namespace netgen
clipplanetimestamp = max2 (vispar.clipping.timestamp, solutiontimestamp); clipplanetimestamp = max2 (vispar.clipping.timestamp, solutiontimestamp);
} }
catch (bad_weak_ptr e) catch (const bad_weak_ptr & e)
{ {
PrintMessage (3, "vssolution::buildscene: don't have a mesh to visualize"); PrintMessage (3, "vssolution::buildscene: don't have a mesh to visualize");
VisualScene::BuildScene (zoomall); VisualScene::BuildScene (zoomall);

View File

@ -194,7 +194,7 @@ namespace netgen
if(mesh->GetGeometry()) if(mesh->GetGeometry())
ng_geometry = mesh->GetGeometry(); ng_geometry = mesh->GetGeometry();
} }
catch (NgException e) catch (const NgException & e)
{ {
PrintMessage (3, e.What()); PrintMessage (3, e.What());
return TCL_ERROR; return TCL_ERROR;
@ -269,7 +269,7 @@ namespace netgen
geometry -> LoadSurfaces(infile); geometry -> LoadSurfaces(infile);
} }
} }
catch (NgException e) catch (const NgException & e)
{ {
PrintMessage (3, e.What()); PrintMessage (3, e.What());
return TCL_ERROR; return TCL_ERROR;
@ -551,7 +551,7 @@ namespace netgen
} }
} }
catch (NgException e) catch (const NgException & e)
{ {
Tcl_SetResult (interp, const_cast<char*> (e.What().c_str()), TCL_VOLATILE); Tcl_SetResult (interp, const_cast<char*> (e.What().c_str()), TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
@ -582,7 +582,7 @@ namespace netgen
{ {
ng_geometry -> Save (string (cfilename)); ng_geometry -> Save (string (cfilename));
} }
catch (NgException e) catch (const NgException & e)
{ {
Tcl_SetResult (interp, const_cast<char*> (e.What().c_str()), TCL_VOLATILE); Tcl_SetResult (interp, const_cast<char*> (e.What().c_str()), TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
@ -1440,7 +1440,7 @@ namespace netgen
PrintMessage (1, "Meshing done, time = ", GetTime(), " sec"); PrintMessage (1, "Meshing done, time = ", GetTime(), " sec");
} }
catch (NgException e) catch (const NgException & e)
{ {
cout << e.What() << endl; cout << e.What() << endl;
} }

View File

@ -8,6 +8,9 @@ try:
from webgui_jupyter_widgets import BaseWebGuiScene, WebGuiDocuWidget from webgui_jupyter_widgets import BaseWebGuiScene, WebGuiDocuWidget
import webgui_jupyter_widgets.widget as wg import webgui_jupyter_widgets.widget as wg
except ImportError: except ImportError:
class BaseWebGuiScene:
pass
wg = None wg = None
def encodeData( data, dtype=None, encoding='b64' ): def encodeData( data, dtype=None, encoding='b64' ):
@ -214,9 +217,13 @@ def GetData(mesh, args, kwargs):
d[name] = pnew d[name] = pnew
return d return d
base = object if wg is None else BaseWebGuiScene class WebGLScene(BaseWebGuiScene):
class WebGLScene(base): class Widget:
def __init__(self):
self.value = {}
def __init__(self, obj, args=[], kwargs={}): def __init__(self, obj, args=[], kwargs={}):
self.widget = self.Widget()
self.obj = obj self.obj = obj
self.args = args self.args = args
self.kwargs = kwargs self.kwargs = kwargs