Use templates to implement the legacy dynamic array classes as much as possible instead of doing it in macros. This makes the code much more maintainable and readable as well as easier to debug. It also allows to avoid casts between function pointers of incompatible types, which triggered many -Wcast-function-type warnings from g++ 8.