Threading Model, basic concepts
This section provides basic information about the Carmenta Engine threading model and introduces some terminology that is used in the Carmenta Engine documentation to describe various aspects of the threading model.
Goals of the threading model
The main goal of the Carmenta Engine threading model is to allow applications to run faster in the future simply by upgrading to newer versions of Carmenta Engine where more operations will have been parallelized. For this to work without requiring any, or only a few, modifications to the code base of existing applications it is necessary for the application developer to understand the Carmenta Engine threading model and to synchronize access to shared data using the appropriate locks. This puts part of the burden on the developer but it also results in the best overall performance as well as the aforementioned opportunities for future performance improvements.
Accessing shared data
The main problem that must be taken care of when writing applications that execute on more than a single thread is to synchronize access to data that is shared between threads. If access to shared data is not synchronized then the application will be susceptible to race conditions that can lead to crashes or other undefined behavior.
Carmenta Engine, like many other modern class libraries and frameworks, is not thread safe (certain common operations have been made thread safe but this is the exception and not the rule). In addition some operations should only be performed on specific threads (in most cases this means the main GUI thread of the application). However, Carmenta Engine can start and use background threads to speed up processing or make some operations asynchronous. Because these background threads are used automatically, as opposed to being explicitly started by an application, it is very important that the application always synchronizes access to data that can be shared with Carmenta Engine's background threads.
So, what data can be shared with one of Carmenta Engine's background threads? The answer is two kinds, namely:
All objects that are accessed to generate a map. This includes views, layers, operators, visualizers, datasets etc. We often refer to this set of objects as the configuration since they are usually defined in a map configuration file that has been created with Carmenta Studio but they can also be created at runtime and access must be synchronized properly then as well. Access to the configuration objects, or configuration for short, must be synchronized with what is known as the global configuration lock or simply the configuration lock.
See Threading Model, configuration for more details.Features in a MemoryDataSet or a custom dataset. Access to features in a MemoryDataSet or custom dataset as well as modifications of the datasets themselves, for example inserting new features, are synchronized by locking the dataset.
See Threading Model, datasets and features for more details.
Each kind of lock, whether configuration or dataset, is acquired by creating an instance of the Guard class using the appropriate constructor. The lock is released when the Guard instance is destroyed (or disposed when using a garbage collected Carmenta Engine API).
GUI controls and event handlers
GUI elements like list boxes, tree controls or something like a Carmenta Engine MapControl, can only be accessed safely from the main GUI thread of the application, sometimes called the main thread or GUI thread. To update or access a GUI element from a background thread the application must marshal the call to the main thread which can then access and modify GUI elements safely.
How you marshal calls to the GUI thread differs between platforms and implementation languages. For example, in a native Windows C++ application you typically marshal a call by posting a message to a window while a .NET application can call the Control.Invoke method to execute a function on the GUI thread.
It might appear that this situation is uncommon but it occurs frequently due to the fact that event handlers typically do not specify on which thread they are called and it is often event handlers that are responsible for updating the GUI.
See Threading model, GUI for more details.
Custom objects
Custom objects are a way for an application to provide its own implementation of a visualizer, symbol, operator or dataset. Carmenta Engine calls custom objects in the same way as a built-in variant of the same type. For example, a custom operator is called in the same way as any other operator is called. This also means that custom objects can be called on background threads which makes it important that their implementation can be called safely on any thread.
See Threading model, custom objects for more details.
Thread Access Verifier
Carmenta Engine 5.9 introduced the thread access verifier that, if enabled, verifies that certain kinds of operations are executed on the correct thread and acquire the correct lock.
See Thread Access Verifier for more details.