-
-
Notifications
You must be signed in to change notification settings - Fork 895
Push EnTT across boundaries
EnTT
has historically had a limit when used across boundaries on Windows in
general and on GNU/Linux when default visibility was set to hidden. The
limitation was mainly due to a custom utility used to assign unique, sequential
identifiers with different types.
Fortunately, nowadays EnTT
works smoothly across boundaries.
Many classes in EnTT
make extensive use of type erasure for their purposes.
This raises the need to identify objects whose type has been erased.
The type_hash
class template is how identifiers are generated and thus made
available to the rest of the library. In general, this class doesn't arouse much
interest. The only exception is when a conflict between identifiers occurs
(definitely uncommon though) or when the default solution proposed by EnTT
isn't suitable for the user's purposes.
The section dedicated to type_info
contains all the details to get around the
issue in a concise and elegant way. Please refer to the specific documentation.
When working with linked libraries, compile definitions ENTT_API_EXPORT
and
ENTT_API_IMPORT
are to import or export symbols, so as to make everything work
nicely across boundaries.
On the other hand, everything should run smoothly when working with plugins or
shared libraries that don't export any symbols.
For those who need more details, the test suite contains many examples covering
the most common cases (see the lib
directory for all details).
It goes without saying that it's impossible to cover all possible cases.
However, what is offered should hopefully serve as a basis for all of them.
The runtime reflection system deserves a special mention when it comes to using
it across boundaries.
Since it's linked already to a static context to which the elements are attached
and different contexts don't relate to each other, they must be shared to
allow the use of meta types across boundaries.
Fortunately, sharing a context is also trivial to do. First of all, the local one is acquired in the main space:
auto handle = entt::locator<entt::meta_ctx>::handle();
Then, it's passed to the receiving space that sets it as its default context, thus discarding or storing aside the local one:
entt::locator<entt::meta_ctx>::reset(handle);
From now on, both spaces refer to the same context and on it are attached all
new meta types, no matter where they are created.
Note that replacing the main context doesn't also propagate changes across
boundaries. In other words, replacing a context results in the decoupling of the
two sides and therefore a divergence in the contents.
There is another subtle problem due to memory management that can lead to
headaches.
It can occur where there are pools of objects (such as components or events)
dynamically created on demand. This is usually not a problem when working with
linked libraries that rely on the same dynamic runtime. However, it can occur in
the case of plugins or statically linked runtimes.
As an example, imagine creating an instance of registry
in the main executable
and sharing it with a plugin. If the latter starts working with a component that
is unknown to the former, a dedicated pool is created within the registry on
first use.
As one can guess, this pool is instantiated on a different side of the boundary
from the registry
. Therefore, the instance is now managing memory from
different spaces and this can quickly lead to crashes if not properly addressed.
To overcome the risk, it's recommended to use well-defined interfaces that make
fundamental types pass through the boundaries, isolating the instances of the
EnTT
classes from time to time and as appropriate.
Refer to the test suite for some examples, read the documentation available
online about this type of issues or consult someone who has already had such
experiences to avoid problems.