This topic provides a startup help for developing Communicator plug-ins for the use with the process integration mode of MERLIC. If you develop a Communicator plug-in for the first time or if you are not sure how to start, you can use this guide to get information about the first steps and the general workflow for the plug-in development up to building and starting the plug-in.
The easiest way to develop a custom Communicator plug-in is to use one of the provided example plug-ins as a basis and customize it as required to your use case. The general workflow for this is described in the first two sections of this topic. The last sections provides basic information about how to build and start the Communicator plug-in.
Using an Example Plug-in as Basis
Customizing the Code of the Plug-in
Building and Providing the Plug-in for the Communicator
You can use one of our example Communicator plug-ins as basis for your plug-in and customize the existing code as required. This way, you do not have to start from scratch and save a lot of time.
"%PROGRAMFILES%\MVTec\MERLIC-5.5"."%LOCALAPPDATA%\Programs\MVTec\MERLIC-5.5".Check the implementation of the plug-in functions in your plug-in. Some of the functions are required for each Communicator plug-in, i.e., MVInfo, MVInit_V2(), MVOpen, MVClose, MVStart, and MVStop. Therefore, you can already find an implementation of those in your copied plug-in. You just need to customize the code for your plug-in.
In addition, you can make use of the optional functions MVExpose and MVValidate which are required if you want to define plug-in parameters that can be configured via the graphical user interface of the MERLIC RTE Setup and if you want to provide a proper validation of the parameter changes.
Expand the following drop-down sections to get more detailed information about the function declarations and individual functions that are required for each Communicator plug-in. The sections show the respective code excerpts based on the example plug-in "event-logger" and describe which parts of the code need to be customized when developing your plug-in. The drop-down sections for the optional functions also give more detailed information about their basic structure and for which use cases they should be implemented.
See also the topic Life Cycle of a Plug-in Instance for specific information about the life cycle of a plug-in instance, especially regarding the implementation and use of the plug-in API functions MVStart, MVStop, MVOpen, and MVClose. You can get information how to ensure that your plug-in is valid and compatible for the use with the Communicator and MERLIC. It also contains information about the internal processes when starting the Communicator and describes the steps that are performed when an instance of a plug-in is used, e.g., which functions are called for certain situations.
Make sure to keep the following function declarations for the API functions symbols, which are expected by the Communicator to be present in the plug-in shared library, in your .cpp file. They are required for each Communicator plug-in.
If the plug-in is written in a language different from C, these functions need to be given C linkage, e.g., by marking them as extern "C" in the case of plug-ins written in C++. All further API functions need not have C linkage as the MVInit_V2() function is expected to set up pointers to the appropriate functions.
In MVInfo, you just have to set the capability of the plug-in, i.e., the value of the placeholder <CAPABILITY> in the code. You can choose between "monitor", "control" or both capabilities, i.e., eMVCapabilities_Monitor, eMVCapabilities_Control, or "eMVCapabilities_Monitor | eMVCapabilities_Control".
Plug-ins with the "monitor" capability can receive "events" that are provided by MERLIC whereas plug-ins with the "control" capability are allowed to send actions to MERLIC using either the MV_QueueAction() or MV_QueueAction_WithInfo() function. If you want to allow your plug-in to receive "events" and send "actions", you have to set the capability to "eMVCapabilities_Monitor | eMVCapabilities_Control".
Define the capability of the plug-in:
In MVInit_V2(), you have to set the function pointers to the remaining plug-in API functions to make them known to the Communicator. In our example plug-ins, the mandatory plug-in functions MVOpen, MVStart, MVStop, and MVClose are used. In addition, the function pointers to the optional functions MVExpose and MVValidate are set in MVInit_V2().
To customize the MVInit_V2() function for your plug-in, it is sufficient to set up the pointers for the optional functions MVExpose and MVValidate depending on whether you want to implement the respective function or not. The function MVExpose is required if you want to define parameters for the plug-in and to enable the configuration of your plug-in parameters via the MERLIC RTE Setup. The function MVValidate is required if you also want to provide a proper validation of the parameter changes. Please see also the sections for MVExpose and MVValidate for more information.
Set up MVExpose and MVValidate functions:
It is possible to use arbitrary names for the functions. However, you have to make sure to set up the function pointer members of the MVClass_V2_t data structure accordingly as shown in the example code below:
Example: Set up the functions with arbitrary names
The name of the MVInit_V2() function indicates the major version number of the Communicator API that is used. This mechanism allows a plug-in to support multiple, otherwise incompatible, major versions of the Communicator API in the future by implementing MVInit_Vx for each supported major version and setting up the function pointers to either the same or different implementations as required.
In MVOpen, a "user data" structure can be created which aggregates the collective state of one plug-in instance. The use of such a user data structure is the recommended way to ensure that the plug-in can manage some internal state which is then accessed and modified by each of the plug-in API functions as well as any additional threads that the plug-in might create. Typically, the data structure can be allocated in MVOpen and is then associated with the plug-in handle via MV_Plugin_SetUserData() to make sure that it is available for the entire existence of the plug-in instance. The pointer to the user data can then be retrieved from the plug-in handle in each of the other plug-in API functions via MV_Plugin_GetUserData(), e.g., in MVClose to free up the memory again.
Using the user data mechanism ensures that the same plug-in can be instantiated multiple times and that the internal state of the plug-in instances does not interfere with one another as would be the case if globals or singletons were used instead. In case global state cannot be avoided due to requirements from third-party dependencies and a plug-in cannot be instantiated safely multiple times, it is recommended to detect a repeated instantiation and return from MVOpen with a return code which is different from MV_CODE_OK.
If there are no restrictions that keep you from using this user data mechanism, you can customize the functions of the example plug-in by adjusting the type of the user data structure as required. In our example plug-ins, the pointer to the user data is called *ud.
Adjust the user data struct:
In MVClose, the pointer to the user data is retrieved to free up the memory again. For this, you have to adjust the data type for the user data pointer as defined in MVOpen.
Adjust the user data struct:
MVStart and MVStop are invoked when a plug-in instance is started and stopped, respectively. In between, the plug-in is considered to be running and is allowed to queue actions (in case of "control" capability) and to receive events (in case of the "monitor" capability). The same plug-in may be started and stopped multiple times and its configuration may be changed while it is not running. Hence, MVStart is typically the place where you can read the configuration options using MV_Plugin_GetConfig() and MV_PluginConfig_GetUserParameter().
To access the internal state of the plug-in instance, you can recall the pointer to the user data. In the code snippet below, it is called *ud. You can then use and modify the field of <USER_DATA_STRUCT_TYPE> in the custom implementation.
In addition, you can remove the code that defines how the example plug-in reacts to "events" ( in case, you are using the "event-logger" plug-in as basis ) and add the desired implementation of your custom plug-in instead.
In the implementation of your plug-in, the plug-in should be able to process events as fast as MERLIC is sending them and must be able to handle dropped events. If a plug-in is processing events slower than MERLIC is sending them, events are queued which may lead to delays. If the queue is full, additional events will be dropped. This may happen if MERLIC is running in continuous mode and the plug-in takes longer to process events than MERLIC needs for a single iteration, or when a plug-in is triggering single executions faster than another plug-in is handling the results. Therefore, this issue should already be considered during the development of the plug-in.
Adjust the user data struct and insert your code:
For plug-ins with the capability eMVCapabilities_Monitor, i.e., plug-ins that handle "events", it is quite typical to use the user data to store the handle to a thread which continuously polls for new events, i.e., an "event loop". In order to be able to terminate such an event loop thread again, a special event of type eMVEvent_Shutdown will be enqueued when a plug-in is stopped. This guarantees to be the last event that the plug-in will receive before it is restarted and thus it is possible to break out of the event loop at that point.
In the "event-logger" example, this is done by setting "stop_event_thread" to "true":
Terminate the event loop in case of the Shutdown event:
In case an event loop thread has been started in MVStart, you have to join that thread again in MVStop. MVStop is called by the Communicator at the same time the Shutdown event is enqueued, so there is no need to take any additional action to stop the thread. In case your plug-in manages additional resources from third-party libraries, such as an I/O context or a database connection, it may be necessary to terminate those explicitly in MVStop.
Adjust the type of the user data struct:
This function must be implemented if you want to define parameters for the plug-in and enable the configuration of these parameters via the graphical user interface of the MERLIC RTE Setup.
You have to define the set of plug-in parameters and their meta data ("user parameter description") for the configuration and use the family of MV_PluginUserParameterDescription_*() API functions to expose the user parameter description. If only predefined values should be allowed for the parameters, you also have to impose the required constraints via MV_PluginUserParameterDescription_ImposeConstraint().
This function is optional. However, if it is not implemented, the configuration of the plug-in is not possible, neither via the MERLIC RTE Setup nor manually via JSON files because no parameters have been defined for the plug-in.
Define the user parameter description:
This function can be implemented if you want to provide a validation for more complex cases of parameter changes that are made in the MERLIC RTE Setup, e.g., if the allowed value range of a parameter depends on other parameter values. MVValidate also enables you to provide feedback to the user about the state of the parameters, e.g., to highlight invalid user parameter values or show specific error messages. This data structure is called "validation result".
For the implementation you have to use the API functions MV_PluginConfig_* and MV_PluginConfigValidation_*.
Implement the validation of parameter changes and feedback of the parameter state:
After the implementation of a plug-in, you have to build the plug-in and make sure that the plug-in is available for the Communicator. The following section gives an overview of how to do this. However, to get more detailed information on the build process and how to provide a plug-in for the Communicator, see the topic Building an Example Plug-in.
With CMake, the build system artifacts (e.g. Makefiles) are generated automatically according to the respective platform, toolchain, and generator.
To build the release version of the plug-in, you can use the following commands:
To get more detailed information on how to build a plug-in and additional arguments, see the section How to Build a Plug-in in the topic Building an Example Plug-in.
To provide a plug-in for the Communicator, the following prerequisites must be fulfilled:
"%PROGRAMFILES%\MVTec\MERLIC-5.5\bin\x64-win64" by default.Therefore, you have to make sure to either copy the plug-in to the predefined plug-in directory or to add the directory in which the plug-in is stored to the list of plug-in directories. In addition, you have to make sure to add the required prefix to the file name of the dynamic library. For more detailed information, see the section Providing the Plug-in for the Communicator in the topic Building an Example Plug-in.
As soon as your plug-in is stored in the defined plug-in directory with the required prefix, you can start using the plug-in with the Communicator. For more information on how to start a Communicator plug-in, see the topic "Starting the Communicator and Plug-ins" in the Communicator Manual.