After finding a plan, Fast Downward calls exit_with() which in turn calls exit(). The exit() function performs some cleanup (https://en.cppreference.com/w/c/program/exit), but doesn't call destructors (https://www.hackerearth.com/practice/notes/return-0-vs-exit0-in-main/). This might be unexpected for users who'd like to report some statistics when objects are destroyed. Also, this behaviour leads to Valgrind reporting memory leaks:
./build.py && valgrind --leak-check=full builds/release/bin/downward --search "astar(blind())" --internal-plan-file sas_plan < output.sas
[...]
Solution found.
Peak memory: 171880 KB
==338825==
==338825== HEAP SUMMARY:
==338825== in use at exit: 43,075 bytes in 223 blocks
==338825== total heap usage: 31,916 allocs, 31,693 frees, 1,712,735 bytes allocated
==338825==
==338825== LEAK SUMMARY:
==338825== definitely lost: 0 bytes in 0 blocks
==338825== indirectly lost: 0 bytes in 0 blocks
==338825== possibly lost: 0 bytes in 0 blocks
==338825== still reachable: 43,075 bytes in 223 blocks
==338825== suppressed: 0 bytes in 0 blocks
==338825== Reachable blocks (those to which a pointer was found) are not shown.
==338825== To see them, rerun with: --leak-check=full --show-leak-kinds=all
==338825==
==338825== For lists of detected and suppressed errors, rerun with: -s
==338825== ERROR SUMMARY: 0 errors from 0 contexts (suppressed: 0 from 0)
In Scorpion, I solved this problem by using the following code at the end of main() in planner.cc:
// exit_with() doesn't call destructors.
report_exit_code_reentrant(exitcode);
return static_cast<int>(exitcode);
|