blob: ed55cd3022bd84921be4d744bf36aeea50c01fc9 [file] [log] [blame]
module li_boost_shared_ptr_runme;
import tango.io.Stdout;
import tango.core.Exception;
import tango.core.Memory;
import tango.core.Thread;
import tango.text.convert.Integer;
import li_boost_shared_ptr.li_boost_shared_ptr;
import li_boost_shared_ptr.Klass;
import li_boost_shared_ptr.KlassDerived;
import li_boost_shared_ptr.Klass3rdDerived;
import li_boost_shared_ptr.MemberVariables;
import li_boost_shared_ptr.PairIntDouble;
// Debugging flag
const bool TRACE = false;
void main() {
if (TRACE)
Stdout("---> STARTED <---").newline;
debug_shared=TRACE;
// Change loop count to run for a long time to monitor memory
const int LOOP_COUNT = 1; // 50000;
for (int i = 0; i < LOOP_COUNT; ++i) {
runTest();
GC.collect();
}
if (TRACE)
Stdout("---> NEARLY FINISHED <---").newline;
// Try to get the GC to collect everything not referenced anymore.
int countdown = 100;
while (--countdown) {
GC.collect();
if (Klass.getTotal_count() == 1)
break;
Thread.sleep(0.01);
}
// A single remaining instance expected: the global variable (GlobalValue).
if (Klass.getTotal_count() != 1)
throw new Exception("Klass.total_count=" ~ toString(Klass.getTotal_count()));
// A single remaining instance expected: the global variable (GlobalSmartValue).
int wrapper_count = shared_ptr_wrapper_count();
if (wrapper_count != NOT_COUNTING)
if (wrapper_count != 1)
throw new Exception("shared_ptr wrapper count=" ~ toString(wrapper_count));
if (TRACE)
Stdout("---> FINISHED <---").newline;
}
void runTest() {
// simple shared_ptr usage - created in C++
{
auto k = new Klass("me oh my");
char[] val = k.getValue();
verifyValue("me oh my", val);
verifyCount(1, k);
}
// simple shared_ptr usage - not created in C++
{
auto k = factorycreate();
char[] val = k.getValue();
verifyValue("factorycreate", val);
verifyCount(1, k);
}
// pass by shared_ptr
{
auto k = new Klass("me oh my");
auto kret = smartpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointertest", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// pass by shared_ptr pointer
{
auto k = new Klass("me oh my");
auto kret = smartpointerpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerpointertest", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// pass by shared_ptr reference
{
auto k = new Klass("me oh my");
auto kret = smartpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerreftest", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// pass by shared_ptr pointer reference
{
auto k = new Klass("me oh my");
auto kret = smartpointerpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerpointerreftest", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// const pass by shared_ptr
{
auto k = new Klass("me oh my");
auto kret = constsmartpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// const pass by shared_ptr pointer
{
auto k = new Klass("me oh my");
auto kret = constsmartpointerpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// const pass by shared_ptr reference
{
auto k = new Klass("me oh my");
auto kret = constsmartpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my", val);
verifyCount(2, k);
verifyCount(2, kret);
}
// pass by value
{
auto k = new Klass("me oh my");
auto kret = valuetest(k);
char[] val = kret.getValue();
verifyValue("me oh my valuetest", val);
verifyCount(1, k);
verifyCount(1, kret);
}
// pass by pointer
{
auto k = new Klass("me oh my");
auto kret = pointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my pointertest", val);
verifyCount(1, k);
verifyCount(1, kret);
}
// pass by reference
{
auto k = new Klass("me oh my");
auto kret = reftest(k);
char[] val = kret.getValue();
verifyValue("me oh my reftest", val);
verifyCount(1, k);
verifyCount(1, kret);
}
// pass by pointer reference
{
auto k = new Klass("me oh my");
auto kret = pointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my pointerreftest", val);
verifyCount(1, k);
verifyCount(1, kret);
}
// null tests
{
Klass k = null;
// TODO: add in const versions too
if (smartpointertest(k) !is null)
throw new Exception("return was not null");
if (smartpointerpointertest(k) !is null)
throw new Exception("return was not null");
if (smartpointerreftest(k) !is null)
throw new Exception("return was not null");
if (smartpointerpointerreftest(k) !is null)
throw new Exception("return was not null");
if (nullsmartpointerpointertest(null) != "null pointer")
throw new Exception("not null smartpointer pointer");
try { valuetest(k); throw new Exception("Failed to catch null pointer"); } catch (IllegalArgumentException) {}
if (pointertest(k) !is null)
throw new Exception("return was not null");
try { reftest(k); throw new Exception("Failed to catch null pointer"); } catch (IllegalArgumentException) {}
}
// $owner
{
auto k = pointerownertest();
char[] val = k.getValue();
verifyValue("pointerownertest", val);
verifyCount(1, k);
}
{
auto k = smartpointerpointerownertest();
char[] val = k.getValue();
verifyValue("smartpointerpointerownertest", val);
verifyCount(1, k);
}
////////////////////////////////// Derived classes ////////////////////////////////////////
// derived pass by shared_ptr
{
auto k = new KlassDerived("me oh my");
auto kret = derivedsmartptrtest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedsmartptrtest-Derived", val);
verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
verifyCount(4, kret);
}
// derived pass by shared_ptr pointer
{
auto k = new KlassDerived("me oh my");
auto kret = derivedsmartptrpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedsmartptrpointertest-Derived", val);
verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
verifyCount(4, kret);
}
// derived pass by shared_ptr ref
{
auto k = new KlassDerived("me oh my");
auto kret = derivedsmartptrreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedsmartptrreftest-Derived", val);
verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
verifyCount(4, kret);
}
// derived pass by shared_ptr pointer ref
{
auto k = new KlassDerived("me oh my");
auto kret = derivedsmartptrpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedsmartptrpointerreftest-Derived", val);
verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
verifyCount(4, kret);
}
// derived pass by pointer
{
auto k = new KlassDerived("me oh my");
auto kret = derivedpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedpointertest-Derived", val);
verifyCount(2, k); // includes an extra reference for the upcast in the proxy class
verifyCount(2, kret);
}
// derived pass by ref
{
auto k = new KlassDerived("me oh my");
auto kret = derivedreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my derivedreftest-Derived", val);
verifyCount(2, k); // includes an extra reference for the upcast in the proxy class
verifyCount(2, kret);
}
////////////////////////////////// Derived and base class mixed ////////////////////////////////////////
// pass by shared_ptr (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = smartpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointertest-Derived", val);
verifyCount(3, k); // an extra reference for the upcast in the proxy class
verifyCount(3, kret);
}
// pass by shared_ptr pointer (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = smartpointerpointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerpointertest-Derived", val);
verifyCount(3, k); // an extra reference for the upcast in the proxy class
verifyCount(3, kret);
}
// pass by shared_ptr reference (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = smartpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerreftest-Derived", val);
verifyCount(3, k); // an extra reference for the upcast in the proxy class
verifyCount(3, kret);
}
// pass by shared_ptr pointer reference (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = smartpointerpointerreftest(k);
char[] val = kret.getValue();
verifyValue("me oh my smartpointerpointerreftest-Derived", val);
verifyCount(3, k); // an extra reference for the upcast in the proxy class
verifyCount(3, kret);
}
// pass by value (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = valuetest(k);
char[] val = kret.getValue();
verifyValue("me oh my valuetest", val); // note slicing
verifyCount(2, k); // an extra reference for the upcast in the proxy class
verifyCount(1, kret);
}
// pass by pointer (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = pointertest(k);
char[] val = kret.getValue();
verifyValue("me oh my pointertest-Derived", val);
verifyCount(2, k); // an extra reference for the upcast in the proxy class
verifyCount(1, kret);
}
// pass by ref (mixed)
{
auto k = new KlassDerived("me oh my");
auto kret = reftest(k);
char[] val = kret.getValue();
verifyValue("me oh my reftest-Derived", val);
verifyCount(2, k); // an extra reference for the upcast in the proxy class
verifyCount(1, kret);
}
// 3rd derived class
{
auto k = new Klass3rdDerived("me oh my");
char[] val = k.getValue();
verifyValue("me oh my-3rdDerived", val);
verifyCount(3, k); // 3 classes in inheritance chain == 3 swigCPtr values
val = test3rdupcast(k);
verifyValue("me oh my-3rdDerived", val);
verifyCount(3, k);
}
////////////////////////////////// Member variables ////////////////////////////////////////
// smart pointer by value
{
auto m = new MemberVariables();
auto k = new Klass("smart member value");
m.SmartMemberValue = k;
char[] val = k.getValue();
verifyValue("smart member value", val);
verifyCount(2, k);
auto kmember = m.SmartMemberValue;
val = kmember.getValue();
verifyValue("smart member value", val);
verifyCount(3, kmember);
verifyCount(3, k);
delete m;
verifyCount(2, kmember);
verifyCount(2, k);
}
// smart pointer by pointer
{
auto m = new MemberVariables();
auto k = new Klass("smart member pointer");
m.SmartMemberPointer = k;
char[] val = k.getValue();
verifyValue("smart member pointer", val);
verifyCount(1, k);
auto kmember = m.SmartMemberPointer;
val = kmember.getValue();
verifyValue("smart member pointer", val);
verifyCount(2, kmember);
verifyCount(2, k);
delete m;
verifyCount(2, kmember);
verifyCount(2, k);
}
// smart pointer by reference
{
auto m = new MemberVariables();
auto k = new Klass("smart member reference");
m.SmartMemberReference = k;
char[] val = k.getValue();
verifyValue("smart member reference", val);
verifyCount(2, k);
auto kmember = m.SmartMemberReference;
val = kmember.getValue();
verifyValue("smart member reference", val);
verifyCount(3, kmember);
verifyCount(3, k);
// The C++ reference refers to SmartMemberValue...
auto kmemberVal = m.SmartMemberValue;
val = kmember.getValue();
verifyValue("smart member reference", val);
verifyCount(4, kmemberVal);
verifyCount(4, kmember);
verifyCount(4, k);
delete m;
verifyCount(3, kmember);
verifyCount(3, k);
}
// plain by value
{
auto m = new MemberVariables();
auto k = new Klass("plain member value");
m.MemberValue = k;
char[] val = k.getValue();
verifyValue("plain member value", val);
verifyCount(1, k);
auto kmember = m.MemberValue;
val = kmember.getValue();
verifyValue("plain member value", val);
verifyCount(1, kmember);
verifyCount(1, k);
delete m;
verifyCount(1, kmember);
verifyCount(1, k);
}
// plain by pointer
{
auto m = new MemberVariables();
auto k = new Klass("plain member pointer");
m.MemberPointer = k;
char[] val = k.getValue();
verifyValue("plain member pointer", val);
verifyCount(1, k);
auto kmember = m.MemberPointer;
val = kmember.getValue();
verifyValue("plain member pointer", val);
verifyCount(1, kmember);
verifyCount(1, k);
delete m;
verifyCount(1, kmember);
verifyCount(1, k);
}
// plain by reference
{
auto m = new MemberVariables();
auto k = new Klass("plain member reference");
m.MemberReference = k;
char[] val = k.getValue();
verifyValue("plain member reference", val);
verifyCount(1, k);
auto kmember = m.MemberReference;
val = kmember.getValue();
verifyValue("plain member reference", val);
verifyCount(1, kmember);
verifyCount(1, k);
delete m;
verifyCount(1, kmember);
verifyCount(1, k);
}
// null member variables
{
auto m = new MemberVariables();
// shared_ptr by value
auto k = m.SmartMemberValue;
if (k !is null)
throw new Exception("expected null");
m.SmartMemberValue = null;
k = m.SmartMemberValue;
if (k !is null)
throw new Exception("expected null");
verifyCount(0, k);
// plain by value
try { m.MemberValue = null; throw new Exception("Failed to catch null pointer"); } catch (IllegalArgumentException) {}
}
////////////////////////////////// Global variables ////////////////////////////////////////
// smart pointer
{
auto kglobal = GlobalSmartValue;
if (kglobal !is null)
throw new Exception("expected null");
auto k = new Klass("smart global value");
GlobalSmartValue = k;
verifyCount(2, k);
kglobal = GlobalSmartValue;
char[] val = kglobal.getValue();
verifyValue("smart global value", val);
verifyCount(3, kglobal);
verifyCount(3, k);
verifyValue("smart global value", GlobalSmartValue.getValue());
GlobalSmartValue = null;
}
// plain value
{
Klass kglobal;
auto k = new Klass("global value");
GlobalValue = k;
verifyCount(1, k);
kglobal = GlobalValue;
char[] val = kglobal.getValue();
verifyValue("global value", val);
verifyCount(1, kglobal);
verifyCount(1, k);
verifyValue("global value", GlobalValue.getValue());
try { GlobalValue = null; throw new Exception("Failed to catch null pointer"); } catch (IllegalArgumentException) {}
}
// plain pointer
{
auto kglobal = GlobalPointer;
if (kglobal !is null)
throw new Exception("expected null");
auto k = new Klass("global pointer");
GlobalPointer = k;
verifyCount(1, k);
kglobal = GlobalPointer;
char[] val = kglobal.getValue();
verifyValue("global pointer", val);
verifyCount(1, kglobal);
verifyCount(1, k);
GlobalPointer = null;
}
// plain reference
{
Klass kglobal;
auto k = new Klass("global reference");
GlobalReference = k;
verifyCount(1, k);
kglobal = GlobalReference;
char[] val = kglobal.getValue();
verifyValue("global reference", val);
verifyCount(1, kglobal);
verifyCount(1, k);
try { GlobalReference = null; throw new Exception("Failed to catch null pointer"); } catch (IllegalArgumentException) {}
}
////////////////////////////////// Templates ////////////////////////////////////////
{
PairIntDouble pid = new PairIntDouble(10, 20.2);
if (pid.baseVal1 != 20 || pid.baseVal2 != 40.4)
throw new Exception("Base values wrong");
if (pid.val1 != 10 || pid.val2 != 20.2)
throw new Exception("Derived Values wrong");
}
}
private void verifyValue(char[] expected, char[] got) {
if (expected != got)
throw new Exception("verify value failed. Expected: " ~ expected ~ " Got: " ~ got);
}
private void verifyCount(int expected, Klass k) {
// We deliberately call the use_count(Klass) overload also for objects which
// are instances of a subclass of Klass (due to static dispatch); things still
// have to work.
int got = use_count(k);
if (expected != got)
throw new Exception("verify use_count failed. Expected: " ~ toString(expected) ~ " Got: " ~ toString(got));
}