Memory Management Model Description
In C#, memory management rules are simple: all objects allocated on heap exist until references to them exist, and are collected by GC afterwards. In C++, this is not the case.
To enable executing translated code in C++ environment, CodePorting.Translator Cs2Cpp wraps all references to objects created on heap into smart pointers of custom System::SmartPtr type. These pointers follow intruisive pointer semantics which means that reference counter resides in allocated object itself and all smart pointers share same reference counter regardless how they were created.
Having same wrapping rules across whole environment allows following native C# memory management rules in translated code. This means, any function defined in any module can allocate any object and use it any way required; the object will be deleted when and only when no more shared references to it exist. The developer shouldn’t make any special arrangements to track down object deletion points.
On the other hand, those types which are usually allocated on stack in C#, are considered value types in translated code. No references to them are created by CodePorting.Translator Cs2Cpp as they are always passed by value.
When working with translated code, same rules should apply to manually-written one. This means:
- Use System::MakeObject() function to allocate object if it is allocated on heap in C#.
- Use System::SmartPtr class to keep references to such objects.
- Do not allocate C# heap types on stack as this can lead to memory management issues (e. g. calling ‘delete’ for stack-allocated objects).
- Allocate C# value types on stack.
There are some exceptions as well.
- Exception classes should always be allocated on stack in C++. Never create instance of exception class using ‘new’ or ‘System::MakeObject’.
- Some special C# types such as System::String become unconditional value types in C++.
Managing memory via shared pointers introduces loop reference issues. If two or more objects hold shared pointers to each other, reference loop is created which prevents all objects in charge from being destroyed. To break such a reference loop, use weak pointers. If reference loop comes from C#, use CppWeakPtr
attribute to do so (see CodePorting.Translator Cs2Cpp attributes for details). In C++ you may use System::WeakPtr class directly.
Unfortunately, some usecases that look normal in C# can’t be reproduced in terms of shared and weak pointers. For example, consider two classes called Actor and Config. Config class represents Actor configuration, while Actor performs some actions based on referenced Config. Both classes have references to each other. If your API allows the user to create any of these two classes in such a manner that the second one is created automatically, you can’t tell which reference is weak and which is strong. In such cases, code refactoring is required to fix memory management issues.