mirror of
https://github.com/NGSolve/netgen.git
synced 2025-01-18 17:00:33 +05:00
437 lines
10 KiB
C++
437 lines
10 KiB
C++
|
|
#include <catch2/catch.hpp>
|
|
#include <../core/ngcore.hpp>
|
|
#include <core/register_archive.hpp>
|
|
#include <core/logging.hpp>
|
|
#include <meshing.hpp>
|
|
using namespace ngcore;
|
|
using namespace std;
|
|
|
|
class CommonBase
|
|
{
|
|
public:
|
|
int a;
|
|
virtual ~CommonBase() {}
|
|
|
|
virtual void DoArchive(Archive& archive) { archive & a; }
|
|
};
|
|
|
|
// pure abstract base class
|
|
class SharedPtrHolder : virtual public CommonBase
|
|
{
|
|
public:
|
|
vector<shared_ptr<string>> names;
|
|
virtual ~SharedPtrHolder()
|
|
{ }
|
|
|
|
virtual void abstract() = 0;
|
|
virtual void DoArchive(Archive& archive)
|
|
{
|
|
CommonBase::DoArchive(archive);
|
|
archive & names;
|
|
}
|
|
};
|
|
|
|
class PtrHolder : virtual public CommonBase
|
|
{
|
|
public:
|
|
vector<int*> numbers;
|
|
virtual ~PtrHolder() {}
|
|
|
|
virtual void DoArchive(Archive& archive)
|
|
{
|
|
CommonBase::DoArchive(archive);
|
|
archive & numbers;
|
|
}
|
|
};
|
|
|
|
class SharedPtrAndPtrHolder : public SharedPtrHolder, public PtrHolder
|
|
{
|
|
public:
|
|
virtual ~SharedPtrAndPtrHolder() {}
|
|
virtual void DoArchive(Archive& archive)
|
|
{
|
|
SharedPtrHolder::DoArchive(archive);
|
|
PtrHolder::DoArchive(archive);
|
|
}
|
|
virtual void abstract() {}
|
|
};
|
|
|
|
// Classes without virt. or multiple inheritance do not need to be registered
|
|
class SimpleClass : public CommonBase
|
|
{
|
|
public:
|
|
double d;
|
|
virtual void DoArchive(Archive& ar)
|
|
{
|
|
CommonBase::DoArchive(ar);
|
|
ar & d;
|
|
}
|
|
};
|
|
|
|
class NotRegisteredForArchive : public SharedPtrAndPtrHolder {};
|
|
|
|
class ClassWithConstPtr
|
|
{
|
|
private:
|
|
const int* ptr;
|
|
public:
|
|
ClassWithConstPtr(const int* aptr) : ptr(aptr) { }
|
|
// constructor only for archive
|
|
ClassWithConstPtr() {}
|
|
void DoArchive(Archive& ar)
|
|
{
|
|
ar & ptr;
|
|
}
|
|
const int* getPtr() { return ptr; }
|
|
};
|
|
|
|
class ClassWithoutDefaultConstructor
|
|
{
|
|
public:
|
|
int a;
|
|
double b;
|
|
double c;
|
|
ClassWithoutDefaultConstructor(int aa, double c) : a(aa), c(c) {}
|
|
|
|
void DoArchive(Archive& ar)
|
|
{
|
|
ar & b;
|
|
}
|
|
|
|
auto GetCArgs()
|
|
{
|
|
return make_tuple(a, c);
|
|
}
|
|
};
|
|
|
|
static RegisterClassForArchive<ClassWithoutDefaultConstructor> regwdc;
|
|
|
|
class OneMoreDerivedClass : public SharedPtrAndPtrHolder {
|
|
};
|
|
|
|
static RegisterClassForArchive<CommonBase> regb;
|
|
static RegisterClassForArchive<SharedPtrHolder, CommonBase> regsp;
|
|
static RegisterClassForArchive<PtrHolder, CommonBase> regp;
|
|
static RegisterClassForArchive<SharedPtrAndPtrHolder, tuple<SharedPtrHolder, PtrHolder>> regspp;
|
|
static RegisterClassForArchive<OneMoreDerivedClass, SharedPtrAndPtrHolder> regom;
|
|
|
|
void testNullPtr(Archive& in, Archive& out)
|
|
{
|
|
SharedPtrHolder* p = nullptr;
|
|
shared_ptr<string> sp = nullptr;
|
|
out & p & sp;
|
|
out.FlushBuffer();
|
|
SharedPtrHolder* pin = nullptr;
|
|
shared_ptr<string> spin = nullptr;
|
|
in & pin & spin;
|
|
CHECK(pin == nullptr);
|
|
CHECK(spin == nullptr);
|
|
}
|
|
|
|
void testSharedPointer(Archive& in, Archive& out)
|
|
{
|
|
SECTION("Same shared ptr")
|
|
{
|
|
static_assert(detail::has_DoArchive<SharedPtrHolder>::value, "");
|
|
SharedPtrAndPtrHolder holder, holder2;
|
|
holder.names.push_back(make_shared<string>("name"));
|
|
holder2.names = holder.names; // same shared ptr
|
|
out & holder & holder2;
|
|
out.FlushBuffer();
|
|
SharedPtrAndPtrHolder inholder, inholder2;
|
|
in & inholder & inholder2;
|
|
CHECK(inholder.names.size() == 1);
|
|
CHECK(inholder.names[0] == inholder2.names[0]);
|
|
CHECK(inholder.names[0].use_count() == 3); // one shared ptr is still kept in the archive
|
|
CHECK(*inholder.names[0] == "name");
|
|
}
|
|
}
|
|
|
|
void testPointer(Archive& in, Archive& out)
|
|
{
|
|
PtrHolder holder, holder2;
|
|
holder.numbers.push_back(new int(3));
|
|
holder2.numbers = holder.numbers; // same shared ptr
|
|
out & holder & holder2;
|
|
out.FlushBuffer();
|
|
PtrHolder inholder, inholder2;
|
|
in & inholder & inholder2;
|
|
CHECK(inholder.numbers.size() == 1);
|
|
CHECK(inholder.numbers[0] == inholder2.numbers[0]);
|
|
CHECK(*inholder.numbers[0] == 3);
|
|
}
|
|
|
|
void testConstPointer(Archive& in, Archive& out)
|
|
{
|
|
SECTION("Const pointer")
|
|
{
|
|
int* iptr = new int(4);
|
|
double d = 0.1;
|
|
ClassWithConstPtr cls(iptr);
|
|
out & cls & iptr & d;
|
|
out.FlushBuffer();
|
|
ClassWithConstPtr incls;
|
|
int* iniptr;
|
|
double ind;
|
|
in & incls & iniptr & ind;
|
|
CHECK(*incls.getPtr() == 4);
|
|
CHECK(incls.getPtr() == iniptr);
|
|
CHECK(ind == 0.1);
|
|
delete iptr;
|
|
delete iniptr;
|
|
}
|
|
}
|
|
|
|
void testMultipleInheritance(Archive& in, Archive& out)
|
|
{
|
|
PtrHolder* p = new OneMoreDerivedClass;
|
|
p->numbers.push_back(new int(2));
|
|
p->a = 5;
|
|
auto p2 = dynamic_cast<SharedPtrHolder*>(p);
|
|
p2->names.push_back(make_shared<string>("test"));
|
|
auto sp1 = shared_ptr<PtrHolder>(p);
|
|
auto sp2 = dynamic_pointer_cast<SharedPtrHolder>(sp1);
|
|
auto checkPtr = [] (auto pin, auto pin2)
|
|
{
|
|
CHECK(typeid(*pin) == typeid(*pin2));
|
|
CHECK(typeid(*pin) == typeid(OneMoreDerivedClass));
|
|
CHECK(*pin2->names[0] == "test");
|
|
CHECK(*pin->numbers[0] == 2);
|
|
CHECK(dynamic_cast<SharedPtrAndPtrHolder*>(pin) == dynamic_cast<SharedPtrAndPtrHolder*>(pin2));
|
|
CHECK(pin->a == pin2->a);
|
|
CHECK(pin->a == 5);
|
|
REQUIRE(dynamic_cast<SharedPtrAndPtrHolder*>(pin2) != nullptr);
|
|
CHECK(*dynamic_cast<SharedPtrAndPtrHolder*>(pin2)->numbers[0] == 2);
|
|
CHECK(*pin->numbers[0] == *dynamic_cast<SharedPtrAndPtrHolder*>(pin2)->numbers[0]);
|
|
REQUIRE(dynamic_cast<SharedPtrAndPtrHolder*>(pin) != nullptr);
|
|
CHECK(dynamic_cast<SharedPtrAndPtrHolder*>(pin)->names[0] == pin2->names[0]);
|
|
};
|
|
SECTION("Archive ptrs to leaves of mult. inh.")
|
|
{
|
|
out & p & p2;
|
|
out.FlushBuffer();
|
|
PtrHolder* pin = nullptr;
|
|
SharedPtrHolder* pin2 = nullptr;
|
|
in & pin & pin2;
|
|
checkPtr(pin, pin2);
|
|
}
|
|
SECTION("Archive shared ptrs to leaves of mult. inh.")
|
|
{
|
|
out & sp1 & sp2;
|
|
out.FlushBuffer();
|
|
shared_ptr<PtrHolder> pin;
|
|
shared_ptr<SharedPtrHolder> pin2;
|
|
in & pin & pin2;
|
|
checkPtr(pin.get(), pin2.get());
|
|
}
|
|
SECTION("Virtual base class")
|
|
{
|
|
CommonBase* b = dynamic_cast<CommonBase*>(p);
|
|
out & b & p;
|
|
PtrHolder* pin;
|
|
CommonBase* bin;
|
|
in & bin & pin;
|
|
checkPtr(pin, dynamic_cast<SharedPtrHolder*>(bin));
|
|
}
|
|
SECTION("Simple class without register")
|
|
{
|
|
auto a = new SimpleClass;
|
|
a->a = 5;
|
|
a->d = 2.3;
|
|
SECTION("check pointer")
|
|
{
|
|
out & a;
|
|
out.FlushBuffer();
|
|
SimpleClass* ain;
|
|
in & ain;
|
|
CHECK(ain->a == 5);
|
|
CHECK(ain->d == 2.3);
|
|
}
|
|
SECTION("check shared pointer")
|
|
{
|
|
auto spa = shared_ptr<SimpleClass>(a);
|
|
out & spa;
|
|
out.FlushBuffer();
|
|
shared_ptr<SimpleClass> spain;
|
|
in & spain;
|
|
CHECK(spain->a == 5);
|
|
CHECK(spain->d == 2.3);
|
|
}
|
|
}
|
|
}
|
|
|
|
void testMap(Archive& in, Archive& out)
|
|
{
|
|
map<string, VersionInfo> map1;
|
|
map1["netgen"] = "v6.2.1901";
|
|
out & map1;
|
|
out.FlushBuffer();
|
|
map<string, VersionInfo> map2;
|
|
in & map2;
|
|
CHECK(map2.size() == 1);
|
|
CHECK(map2["netgen"] == "v6.2.1901");
|
|
}
|
|
|
|
enum MyEnum
|
|
{
|
|
CASE1,
|
|
CASE2
|
|
};
|
|
|
|
void testEnum(Archive& in, Archive& out)
|
|
{
|
|
MyEnum en = CASE2;
|
|
out & en;
|
|
out.FlushBuffer();
|
|
MyEnum enin;
|
|
in & enin;
|
|
CHECK(enin == CASE2);
|
|
}
|
|
|
|
void testOptional(Archive& in, Archive& out)
|
|
{
|
|
{
|
|
std::optional<int> no_value;
|
|
std::optional<int> myint = 42;
|
|
std::optional<string> mystr = "have an optional string";
|
|
out & no_value & myint & mystr;
|
|
out.FlushBuffer();
|
|
}
|
|
{
|
|
std::optional<int> no_value_;
|
|
std::optional<int> myint_;
|
|
std::optional<string> mystr_;
|
|
in & no_value_ & myint_ & mystr_;
|
|
CHECK(no_value_.has_value() == false);
|
|
CHECK(myint_.has_value() == true);
|
|
CHECK(*myint_ == 42);
|
|
CHECK(mystr_.has_value() == true);
|
|
CHECK(*mystr_ == "have an optional string");
|
|
}
|
|
}
|
|
|
|
void testArchive(Archive& in, Archive& out)
|
|
{
|
|
SECTION("Empty String")
|
|
{
|
|
char* cstr = nullptr;
|
|
char* empty = new char[1];
|
|
char* simple = new char[7] {'s','i','m','p','l','e','\0'};
|
|
empty[0] = '\0';
|
|
out << string("") << cstr << empty << simple << string("simple") << long(1);
|
|
out.FlushBuffer();
|
|
string str; long i; char* readempty; char* readsimple;
|
|
string simplestr;
|
|
in & str & cstr & readempty & readsimple & simplestr & i;
|
|
CHECK(str == "");
|
|
CHECK(cstr == nullptr);
|
|
CHECK(strcmp(readempty,"") == 0);
|
|
CHECK(strcmp(readsimple,"simple") == 0);
|
|
CHECK(i == 1);
|
|
CHECK(simplestr == "simple");
|
|
delete[] readempty;
|
|
delete[] empty;
|
|
delete[] simple;
|
|
delete[] readsimple;
|
|
}
|
|
SECTION("SharedPtr")
|
|
{
|
|
testSharedPointer(in, out);
|
|
}
|
|
SECTION("Pointer")
|
|
{
|
|
testPointer(in, out);
|
|
}
|
|
SECTION("Const Pointer")
|
|
{
|
|
testConstPointer(in, out);
|
|
}
|
|
SECTION("Multiple inheritance")
|
|
{
|
|
testMultipleInheritance(in, out);
|
|
}
|
|
SECTION("Not registered")
|
|
{
|
|
SharedPtrAndPtrHolder* p = new NotRegisteredForArchive;
|
|
REQUIRE_THROWS(out & p, Catch::Contains("not registered for archive"));
|
|
}
|
|
SECTION("Non-default constructor")
|
|
{
|
|
ClassWithoutDefaultConstructor c(5, 2.2);
|
|
c.b = 3.2;
|
|
auto p = &c;
|
|
out & p;
|
|
out.FlushBuffer();
|
|
ClassWithoutDefaultConstructor* cin;
|
|
in & cin;
|
|
CHECK(cin->a == 5);
|
|
CHECK(cin->b == 3.2);
|
|
CHECK(cin->c == 2.2);
|
|
}
|
|
SECTION("nullptr")
|
|
{
|
|
testNullPtr(in, out);
|
|
}
|
|
SECTION("map")
|
|
{
|
|
testMap(in, out);
|
|
}
|
|
SECTION("enum")
|
|
{
|
|
testEnum(in, out);
|
|
}
|
|
SECTION("optional")
|
|
{
|
|
testOptional(in, out);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("BinaryArchive")
|
|
{
|
|
auto stream = make_shared<stringstream>();
|
|
BinaryOutArchive out(stream);
|
|
BinaryInArchive in(stream);
|
|
testArchive(in, out);
|
|
}
|
|
|
|
TEST_CASE("TextArchive")
|
|
{
|
|
auto stream = make_shared<stringstream>();
|
|
TextOutArchive out(stream);
|
|
TextInArchive in(stream);
|
|
testArchive(in, out);
|
|
}
|
|
|
|
|
|
template <typename T>
|
|
auto CheckCopyWithArchive(const T * v) {
|
|
T * tcopy = nullptr;
|
|
auto tstream = make_shared<stringstream>();
|
|
TextOutArchive tout(tstream);
|
|
tout & v;
|
|
TextInArchive tin(tstream);
|
|
tin & tcopy;
|
|
|
|
T *bcopy = nullptr;
|
|
auto bstream = make_shared<stringstream>();
|
|
BinaryOutArchive bout(bstream);
|
|
bout & v;
|
|
bout.FlushBuffer();
|
|
BinaryInArchive in(bstream);
|
|
in & bcopy;
|
|
|
|
CHECK(*v == *tcopy);
|
|
CHECK(*v == *bcopy);
|
|
CHECK(*bcopy == *bcopy);
|
|
}
|
|
|
|
TEST_CASE("CopyWithArchive")
|
|
{
|
|
Array<int> aint{1,2,5,67,23252};
|
|
CheckCopyWithArchive(&aint);
|
|
|
|
std::vector<byte> abyte{byte(1), byte(3), byte(255), byte(0), byte(76)};
|
|
CheckCopyWithArchive(&abyte);
|
|
}
|