PageRenderTime 56ms CodeModel.GetById 25ms RepoModel.GetById 0ms app.codeStats 0ms

/trunk/Examples/test-suite/d/li_boost_shared_ptr_runme.2.d

#
D | 602 lines | 471 code | 74 blank | 57 comment | 26 complexity | 1a0b36594b3a01d01454a515242b951b MD5 | raw file
Possible License(s): LGPL-2.1, Cube, GPL-3.0, 0BSD, GPL-2.0
  1. module li_boost_shared_ptr_runme;
  2. import core.memory;
  3. import core.thread;
  4. import std.conv;
  5. import std.exception;
  6. import std.stdio;
  7. import li_boost_shared_ptr.li_boost_shared_ptr;
  8. import li_boost_shared_ptr.Klass;
  9. import li_boost_shared_ptr.KlassDerived;
  10. import li_boost_shared_ptr.Klass3rdDerived;
  11. import li_boost_shared_ptr.MemberVariables;
  12. import li_boost_shared_ptr.PairIntDouble;
  13. // Debugging flag
  14. enum TRACE = false;
  15. void main() {
  16. if (TRACE)
  17. writeln("---> STARTED <---");
  18. debug_shared = TRACE;
  19. // Change loop count to run for a long time to monitor memory
  20. enum LOOP_COUNT = 1; // 50000;
  21. for (int i = 0; i < LOOP_COUNT; ++i) {
  22. runTest();
  23. GC.collect();
  24. }
  25. if (TRACE)
  26. writeln("---> NEARLY FINISHED <---");
  27. // Try to get the GC to collect everything not referenced anymore.
  28. int countdown = 100;
  29. while (--countdown) {
  30. GC.collect();
  31. if (Klass.getTotal_count() == 1)
  32. break;
  33. Thread.sleep(100);
  34. }
  35. // A single remaining instance expected: the global variable (GlobalValue).
  36. if (Klass.getTotal_count() != 1)
  37. throw new Exception("Klass.total_count=" ~ to!string(Klass.getTotal_count()));
  38. // A single remaining instance expected: the global variable (GlobalSmartValue).
  39. int wrapper_count = shared_ptr_wrapper_count();
  40. if (wrapper_count != NOT_COUNTING)
  41. if (wrapper_count != 1)
  42. throw new Exception("shared_ptr wrapper count=" ~ to!string(wrapper_count));
  43. if (TRACE)
  44. writeln("---> FINISHED <---");
  45. }
  46. void runTest() {
  47. // simple shared_ptr usage - created in C++
  48. {
  49. auto k = new Klass("me oh my");
  50. string val = k.getValue();
  51. verifyValue("me oh my", val);
  52. verifyCount(1, k);
  53. }
  54. // simple shared_ptr usage - not created in C++
  55. {
  56. auto k = factorycreate();
  57. string val = k.getValue();
  58. verifyValue("factorycreate", val);
  59. verifyCount(1, k);
  60. }
  61. // pass by shared_ptr
  62. {
  63. auto k = new Klass("me oh my");
  64. auto kret = smartpointertest(k);
  65. string val = kret.getValue();
  66. verifyValue("me oh my smartpointertest", val);
  67. verifyCount(2, k);
  68. verifyCount(2, kret);
  69. }
  70. // pass by shared_ptr pointer
  71. {
  72. auto k = new Klass("me oh my");
  73. auto kret = smartpointerpointertest(k);
  74. string val = kret.getValue();
  75. verifyValue("me oh my smartpointerpointertest", val);
  76. verifyCount(2, k);
  77. verifyCount(2, kret);
  78. }
  79. // pass by shared_ptr reference
  80. {
  81. auto k = new Klass("me oh my");
  82. auto kret = smartpointerreftest(k);
  83. string val = kret.getValue();
  84. verifyValue("me oh my smartpointerreftest", val);
  85. verifyCount(2, k);
  86. verifyCount(2, kret);
  87. }
  88. // pass by shared_ptr pointer reference
  89. {
  90. auto k = new Klass("me oh my");
  91. auto kret = smartpointerpointerreftest(k);
  92. string val = kret.getValue();
  93. verifyValue("me oh my smartpointerpointerreftest", val);
  94. verifyCount(2, k);
  95. verifyCount(2, kret);
  96. }
  97. // const pass by shared_ptr
  98. {
  99. auto k = new Klass("me oh my");
  100. auto kret = constsmartpointertest(k);
  101. string val = kret.getValue();
  102. verifyValue("me oh my", val);
  103. verifyCount(2, k);
  104. verifyCount(2, kret);
  105. }
  106. // const pass by shared_ptr pointer
  107. {
  108. auto k = new Klass("me oh my");
  109. auto kret = constsmartpointerpointertest(k);
  110. string val = kret.getValue();
  111. verifyValue("me oh my", val);
  112. verifyCount(2, k);
  113. verifyCount(2, kret);
  114. }
  115. // const pass by shared_ptr reference
  116. {
  117. auto k = new Klass("me oh my");
  118. auto kret = constsmartpointerreftest(k);
  119. string val = kret.getValue();
  120. verifyValue("me oh my", val);
  121. verifyCount(2, k);
  122. verifyCount(2, kret);
  123. }
  124. // pass by value
  125. {
  126. auto k = new Klass("me oh my");
  127. auto kret = valuetest(k);
  128. string val = kret.getValue();
  129. verifyValue("me oh my valuetest", val);
  130. verifyCount(1, k);
  131. verifyCount(1, kret);
  132. }
  133. // pass by pointer
  134. {
  135. auto k = new Klass("me oh my");
  136. auto kret = pointertest(k);
  137. string val = kret.getValue();
  138. verifyValue("me oh my pointertest", val);
  139. verifyCount(1, k);
  140. verifyCount(1, kret);
  141. }
  142. // pass by reference
  143. {
  144. auto k = new Klass("me oh my");
  145. auto kret = reftest(k);
  146. string val = kret.getValue();
  147. verifyValue("me oh my reftest", val);
  148. verifyCount(1, k);
  149. verifyCount(1, kret);
  150. }
  151. // pass by pointer reference
  152. {
  153. auto k = new Klass("me oh my");
  154. auto kret = pointerreftest(k);
  155. string val = kret.getValue();
  156. verifyValue("me oh my pointerreftest", val);
  157. verifyCount(1, k);
  158. verifyCount(1, kret);
  159. }
  160. // null tests
  161. {
  162. Klass k = null;
  163. // TODO: add in const versions too
  164. enforce(smartpointertest(k) is null, "return was not null");
  165. enforce(smartpointerpointertest(k) is null, "return was not null");
  166. enforce(smartpointerreftest(k) is null, "return was not null");
  167. enforce(smartpointerpointerreftest(k) is null, "return was not null");
  168. enforce(nullsmartpointerpointertest(null) == "null pointer",
  169. "not null smartpointer pointer");
  170. enforceThrows( (){ valuetest(k); }, "Failed to catch null pointer");
  171. enforce(pointertest(k) is null, "return was not null");
  172. enforceThrows( (){ reftest(k); }, "Failed to catch null pointer");
  173. }
  174. // $owner
  175. {
  176. auto k = pointerownertest();
  177. string val = k.getValue();
  178. verifyValue("pointerownertest", val);
  179. verifyCount(1, k);
  180. }
  181. {
  182. auto k = smartpointerpointerownertest();
  183. string val = k.getValue();
  184. verifyValue("smartpointerpointerownertest", val);
  185. verifyCount(1, k);
  186. }
  187. ////////////////////////////////// Derived classes ////////////////////////////////////////
  188. // derived pass by shared_ptr
  189. {
  190. auto k = new KlassDerived("me oh my");
  191. auto kret = derivedsmartptrtest(k);
  192. string val = kret.getValue();
  193. verifyValue("me oh my derivedsmartptrtest-Derived", val);
  194. verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
  195. verifyCount(4, kret);
  196. }
  197. // derived pass by shared_ptr pointer
  198. {
  199. auto k = new KlassDerived("me oh my");
  200. auto kret = derivedsmartptrpointertest(k);
  201. string val = kret.getValue();
  202. verifyValue("me oh my derivedsmartptrpointertest-Derived", val);
  203. verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
  204. verifyCount(4, kret);
  205. }
  206. // derived pass by shared_ptr ref
  207. {
  208. auto k = new KlassDerived("me oh my");
  209. auto kret = derivedsmartptrreftest(k);
  210. string val = kret.getValue();
  211. verifyValue("me oh my derivedsmartptrreftest-Derived", val);
  212. verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
  213. verifyCount(4, kret);
  214. }
  215. // derived pass by shared_ptr pointer ref
  216. {
  217. auto k = new KlassDerived("me oh my");
  218. auto kret = derivedsmartptrpointerreftest(k);
  219. string val = kret.getValue();
  220. verifyValue("me oh my derivedsmartptrpointerreftest-Derived", val);
  221. verifyCount(4, k); // includes two extra references for upcasts in the proxy classes
  222. verifyCount(4, kret);
  223. }
  224. // derived pass by pointer
  225. {
  226. auto k = new KlassDerived("me oh my");
  227. auto kret = derivedpointertest(k);
  228. string val = kret.getValue();
  229. verifyValue("me oh my derivedpointertest-Derived", val);
  230. verifyCount(2, k); // includes an extra reference for the upcast in the proxy class
  231. verifyCount(2, kret);
  232. }
  233. // derived pass by ref
  234. {
  235. auto k = new KlassDerived("me oh my");
  236. auto kret = derivedreftest(k);
  237. string val = kret.getValue();
  238. verifyValue("me oh my derivedreftest-Derived", val);
  239. verifyCount(2, k); // includes an extra reference for the upcast in the proxy class
  240. verifyCount(2, kret);
  241. }
  242. ////////////////////////////////// Derived and base class mixed ////////////////////////////////////////
  243. // pass by shared_ptr (mixed)
  244. {
  245. auto k = new KlassDerived("me oh my");
  246. auto kret = smartpointertest(k);
  247. string val = kret.getValue();
  248. verifyValue("me oh my smartpointertest-Derived", val);
  249. verifyCount(3, k); // an extra reference for the upcast in the proxy class
  250. verifyCount(3, kret);
  251. }
  252. // pass by shared_ptr pointer (mixed)
  253. {
  254. auto k = new KlassDerived("me oh my");
  255. auto kret = smartpointerpointertest(k);
  256. string val = kret.getValue();
  257. verifyValue("me oh my smartpointerpointertest-Derived", val);
  258. verifyCount(3, k); // an extra reference for the upcast in the proxy class
  259. verifyCount(3, kret);
  260. }
  261. // pass by shared_ptr reference (mixed)
  262. {
  263. auto k = new KlassDerived("me oh my");
  264. auto kret = smartpointerreftest(k);
  265. string val = kret.getValue();
  266. verifyValue("me oh my smartpointerreftest-Derived", val);
  267. verifyCount(3, k); // an extra reference for the upcast in the proxy class
  268. verifyCount(3, kret);
  269. }
  270. // pass by shared_ptr pointer reference (mixed)
  271. {
  272. auto k = new KlassDerived("me oh my");
  273. auto kret = smartpointerpointerreftest(k);
  274. string val = kret.getValue();
  275. verifyValue("me oh my smartpointerpointerreftest-Derived", val);
  276. verifyCount(3, k); // an extra reference for the upcast in the proxy class
  277. verifyCount(3, kret);
  278. }
  279. // pass by value (mixed)
  280. {
  281. auto k = new KlassDerived("me oh my");
  282. auto kret = valuetest(k);
  283. string val = kret.getValue();
  284. verifyValue("me oh my valuetest", val); // note slicing
  285. verifyCount(2, k); // an extra reference for the upcast in the proxy class
  286. verifyCount(1, kret);
  287. }
  288. // pass by pointer (mixed)
  289. {
  290. auto k = new KlassDerived("me oh my");
  291. auto kret = pointertest(k);
  292. string val = kret.getValue();
  293. verifyValue("me oh my pointertest-Derived", val);
  294. verifyCount(2, k); // an extra reference for the upcast in the proxy class
  295. verifyCount(1, kret);
  296. }
  297. // pass by ref (mixed)
  298. {
  299. auto k = new KlassDerived("me oh my");
  300. auto kret = reftest(k);
  301. string val = kret.getValue();
  302. verifyValue("me oh my reftest-Derived", val);
  303. verifyCount(2, k); // an extra reference for the upcast in the proxy class
  304. verifyCount(1, kret);
  305. }
  306. // 3rd derived class
  307. {
  308. auto k = new Klass3rdDerived("me oh my");
  309. string val = k.getValue();
  310. verifyValue("me oh my-3rdDerived", val);
  311. verifyCount(3, k); // 3 classes in inheritance chain == 3 swigCPtr values
  312. val = test3rdupcast(k);
  313. verifyValue("me oh my-3rdDerived", val);
  314. verifyCount(3, k);
  315. }
  316. ////////////////////////////////// Member variables ////////////////////////////////////////
  317. // smart pointer by value
  318. {
  319. auto m = new MemberVariables();
  320. auto k = new Klass("smart member value");
  321. m.SmartMemberValue = k;
  322. string val = k.getValue();
  323. verifyValue("smart member value", val);
  324. verifyCount(2, k);
  325. auto kmember = m.SmartMemberValue;
  326. val = kmember.getValue();
  327. verifyValue("smart member value", val);
  328. verifyCount(3, kmember);
  329. verifyCount(3, k);
  330. delete m;
  331. verifyCount(2, kmember);
  332. verifyCount(2, k);
  333. }
  334. // smart pointer by pointer
  335. {
  336. auto m = new MemberVariables();
  337. auto k = new Klass("smart member pointer");
  338. m.SmartMemberPointer = k;
  339. string val = k.getValue();
  340. verifyValue("smart member pointer", val);
  341. verifyCount(1, k);
  342. auto kmember = m.SmartMemberPointer;
  343. val = kmember.getValue();
  344. verifyValue("smart member pointer", val);
  345. verifyCount(2, kmember);
  346. verifyCount(2, k);
  347. delete m;
  348. verifyCount(2, kmember);
  349. verifyCount(2, k);
  350. }
  351. // smart pointer by reference
  352. {
  353. auto m = new MemberVariables();
  354. auto k = new Klass("smart member reference");
  355. m.SmartMemberReference = k;
  356. string val = k.getValue();
  357. verifyValue("smart member reference", val);
  358. verifyCount(2, k);
  359. auto kmember = m.SmartMemberReference;
  360. val = kmember.getValue();
  361. verifyValue("smart member reference", val);
  362. verifyCount(3, kmember);
  363. verifyCount(3, k);
  364. // The C++ reference refers to SmartMemberValue...
  365. auto kmemberVal = m.SmartMemberValue;
  366. val = kmember.getValue();
  367. verifyValue("smart member reference", val);
  368. verifyCount(4, kmemberVal);
  369. verifyCount(4, kmember);
  370. verifyCount(4, k);
  371. delete m;
  372. verifyCount(3, kmember);
  373. verifyCount(3, k);
  374. }
  375. // plain by value
  376. {
  377. auto m = new MemberVariables();
  378. auto k = new Klass("plain member value");
  379. m.MemberValue = k;
  380. string val = k.getValue();
  381. verifyValue("plain member value", val);
  382. verifyCount(1, k);
  383. auto kmember = m.MemberValue;
  384. val = kmember.getValue();
  385. verifyValue("plain member value", val);
  386. verifyCount(1, kmember);
  387. verifyCount(1, k);
  388. delete m;
  389. verifyCount(1, kmember);
  390. verifyCount(1, k);
  391. }
  392. // plain by pointer
  393. {
  394. auto m = new MemberVariables();
  395. auto k = new Klass("plain member pointer");
  396. m.MemberPointer = k;
  397. string val = k.getValue();
  398. verifyValue("plain member pointer", val);
  399. verifyCount(1, k);
  400. auto kmember = m.MemberPointer;
  401. val = kmember.getValue();
  402. verifyValue("plain member pointer", val);
  403. verifyCount(1, kmember);
  404. verifyCount(1, k);
  405. delete m;
  406. verifyCount(1, kmember);
  407. verifyCount(1, k);
  408. }
  409. // plain by reference
  410. {
  411. auto m = new MemberVariables();
  412. auto k = new Klass("plain member reference");
  413. m.MemberReference = k;
  414. string val = k.getValue();
  415. verifyValue("plain member reference", val);
  416. verifyCount(1, k);
  417. auto kmember = m.MemberReference;
  418. val = kmember.getValue();
  419. verifyValue("plain member reference", val);
  420. verifyCount(1, kmember);
  421. verifyCount(1, k);
  422. delete m;
  423. verifyCount(1, kmember);
  424. verifyCount(1, k);
  425. }
  426. // null member variables
  427. {
  428. auto m = new MemberVariables();
  429. // shared_ptr by value
  430. auto k = m.SmartMemberValue;
  431. if (k !is null)
  432. throw new Exception("expected null");
  433. m.SmartMemberValue = null;
  434. k = m.SmartMemberValue;
  435. if (k !is null)
  436. throw new Exception("expected null");
  437. verifyCount(0, k);
  438. // plain by value
  439. enforceThrows( (){ m.MemberValue = null; }, "Failed to catch null pointer");
  440. }
  441. ////////////////////////////////// Global variables ////////////////////////////////////////
  442. // smart pointer
  443. {
  444. auto kglobal = GlobalSmartValue;
  445. enforce(kglobal is null, "expected null");
  446. auto k = new Klass("smart global value");
  447. GlobalSmartValue = k;
  448. verifyCount(2, k);
  449. kglobal = GlobalSmartValue;
  450. string val = kglobal.getValue();
  451. verifyValue("smart global value", val);
  452. verifyCount(3, kglobal);
  453. verifyCount(3, k);
  454. verifyValue("smart global value", GlobalSmartValue.getValue());
  455. GlobalSmartValue = null;
  456. }
  457. // plain value
  458. {
  459. Klass kglobal;
  460. auto k = new Klass("global value");
  461. GlobalValue = k;
  462. verifyCount(1, k);
  463. kglobal = GlobalValue;
  464. string val = kglobal.getValue();
  465. verifyValue("global value", val);
  466. verifyCount(1, kglobal);
  467. verifyCount(1, k);
  468. verifyValue("global value", GlobalValue.getValue());
  469. enforceThrows((){ GlobalValue = null; }, "Failed to catch null pointer");
  470. }
  471. // plain pointer
  472. {
  473. auto kglobal = GlobalPointer;
  474. enforce(kglobal is null, "expected null");
  475. auto k = new Klass("global pointer");
  476. GlobalPointer = k;
  477. verifyCount(1, k);
  478. kglobal = GlobalPointer;
  479. string val = kglobal.getValue();
  480. verifyValue("global pointer", val);
  481. verifyCount(1, kglobal);
  482. verifyCount(1, k);
  483. GlobalPointer = null;
  484. }
  485. // plain reference
  486. {
  487. Klass kglobal;
  488. auto k = new Klass("global reference");
  489. GlobalReference = k;
  490. verifyCount(1, k);
  491. kglobal = GlobalReference;
  492. string val = kglobal.getValue();
  493. verifyValue("global reference", val);
  494. verifyCount(1, kglobal);
  495. verifyCount(1, k);
  496. enforceThrows((){ GlobalReference = null; }, "Failed to catch null pointer");
  497. }
  498. ////////////////////////////////// Templates ////////////////////////////////////////
  499. {
  500. auto pid = new PairIntDouble(10, 20.2);
  501. enforce(pid.baseVal1 == 20 && pid.baseVal2== 40.4, "Base values wrong");
  502. enforce(pid.val1 == 10 && pid.val2 == 20.2, "Derived Values wrong");
  503. }
  504. }
  505. private void verifyValue(string expected, string got) {
  506. if (expected != got)
  507. throw new Exception("verify value failed. Expected: " ~ expected ~ " Got: " ~ got);
  508. }
  509. private void verifyCount(int expected, Klass k) {
  510. // We deliberately call the use_count(Klass) overload also for objects which
  511. // are instances of a subclass of Klass (due to static dispatch); things still
  512. // have to work.
  513. auto got = use_count(k);
  514. if (expected != got)
  515. throw new Exception("verify use_count failed. Expected: " ~ to!string(expected) ~ " Got: " ~ to!string(got));
  516. }
  517. private void enforceThrows(void delegate() dg, string errorMessage) {
  518. bool hasThrown;
  519. try {
  520. dg();
  521. } catch (Exception) {
  522. hasThrown = true;
  523. } finally {
  524. if (!hasThrown) {
  525. throw new Exception(errorMessage);
  526. }
  527. }
  528. }