diff --git a/Makefile b/Makefile index a5526e1..72f4fb8 100644 --- a/Makefile +++ b/Makefile @@ -19,7 +19,7 @@ export DSE_SCHEMA_URL ?= $(DSE_SCHEMA_REPO)/releases/download/v$(DSE_SCHEMA_VERS ############### ## DSE C Library. DSE_CLIB_REPO ?= https://github.com/boschglobal/dse.clib -DSE_CLIB_VERSION ?= 1.0.7 +DSE_CLIB_VERSION ?= 1.0.8 export DSE_CLIB_URL ?= $(DSE_CLIB_REPO)/archive/refs/tags/v$(DSE_CLIB_VERSION).zip diff --git a/doc/Makefile b/doc/Makefile index a4bca91..29e4b56 100644 --- a/doc/Makefile +++ b/doc/Makefile @@ -45,6 +45,13 @@ DOC_OUTPUT_model := doc/content/apis/modelc/model/index.md DOC_LINKTITLE_model := Model DOC_TITLE_model := "Model API Reference" +# Module "runtime" +DOC_INPUT_runtime := dse/modelc/runtime.h +DOC_CDIR_runtime := dse/modelc/controller/modelc.c,dse/modelc/controller/modelc_args.c,dse/modelc/controller/modelc_debug.c, +DOC_OUTPUT_runtime := doc/content/apis/modelc/runtime/index.md +DOC_LINKTITLE_runtime := Runtime +DOC_TITLE_runtime := "Runtime API Reference" + # Module "schema" DOC_INPUT_schema := dse/modelc/schema.h DOC_CDIR_schema := dse/modelc/model/schema.c @@ -53,15 +60,15 @@ DOC_LINKTITLE_schema := Schema DOC_TITLE_schema := "Schema API Reference" # Module "simmock" -DOC_INPUT_simmock := dse/modelc/mocks/simmock.h -DOC_CDIR_simmock := dse/modelc/mocks/simmock.c +DOC_INPUT_simmock := dse/mocks/simmock.h +DOC_CDIR_simmock := dse/mocks/simmock.c DOC_OUTPUT_simmock := doc/content/apis/modelc/simmock/index.md DOC_LINKTITLE_simmock := SimMock DOC_TITLE_simmock := "SimMock API Reference" # Targets -DOC_C_MODULES := gateway mcl model schema simmock +DOC_C_MODULES := gateway mcl model runtime schema simmock #DOC_C_MODULES := model .PHONY: examples diff --git a/doc/content/apis/modelc/examples/gateway.c b/doc/content/apis/modelc/examples/gateway.c index 61b6e16..f081151 100644 --- a/doc/content/apis/modelc/examples/gateway.c +++ b/doc/content/apis/modelc/examples/gateway.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 #include +#include #include #include #include diff --git a/doc/content/apis/modelc/examples/model_create.c b/doc/content/apis/modelc/examples/model_create.c new file mode 100644 index 0000000..715e53c --- /dev/null +++ b/doc/content/apis/modelc/examples/model_create.c @@ -0,0 +1,44 @@ +#include +#include +#include +#include + +typedef struct { + ModelDesc model; + /* Signal Pointers. */ + struct { + double* counter; + } signals; +} ExtendedModelDesc; + +static inline double* _index(ExtendedModelDesc* m, const char* v, const char* s) +{ + ModelSignalIndex idx = m->model.index((ModelDesc*)m, v, s); + if (idx.scalar == NULL) log_fatal("Signal not found (%s:%s)", v, s); + return idx.scalar; +} + +ModelDesc* model_create(ModelDesc* model) +{ + /* Extend the ModelDesc object (using a shallow copy). */ + ExtendedModelDesc* m = calloc(1, sizeof(ExtendedModelDesc)); + memcpy(m, model, sizeof(ModelDesc)); + + /* Index the signals that are used by this model. */ + m->signals.counter = _index(m, "data", "counter"); + + /* Set initial values. */ + *(m->signals.counter) = 42; + + /* Return the extended object. */ + return (ModelDesc*)m; +} + +int model_step(ModelDesc* model, double* model_time, double stop_time) +{ + ExtendedModelDesc* m = (ExtendedModelDesc*)model; + *(m->signals.counter) += 1; + + *model_time = stop_time; + return 0; +} diff --git a/doc/content/apis/modelc/examples/model_interface.c b/doc/content/apis/modelc/examples/model_interface.c index 9c4453e..b3570d8 100644 --- a/doc/content/apis/modelc/examples/model_interface.c +++ b/doc/content/apis/modelc/examples/model_interface.c @@ -1,50 +1,25 @@ +#include +#include #include -#include - #define UNUSED(x) ((void)x) - -/* Signal Vector definition. - Note: Signal order should match order in related SignalGroup (YAML). */ -typedef enum signal_name_index { - SIGNAL_FOO, - SIGNAL_BAR, - __SIGNAL__COUNT__ -} signal_name_index; - -static double* signal_value; - - -int model_step(double* model_time, double stop_time) +ModelDesc* model_create(ModelDesc* m) { - signal_value[SIGNAL_FOO] += 1.2; - signal_value[SIGNAL_BAR] += 4.2; - - *model_time = stop_time; - return 0; + return (ModelDesc*)m; } -int model_setup(ModelInstanceSpec* mi) +int model_step(ModelDesc* m, double* model_time, double stop_time) { - int rc = model_function_register(mi, "example", - 0.005, model_step); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = "model_channel", - .function_name = "example", - }; - rc = model_configure_channel(mi, &channel_desc); - if (rc != 0) return rc; - signal_value = channel_desc.vector_double; + ModelSignalIndex counter = m->index(m, "data", "counter"); + if (counter.scalar == NULL) return -EINVAL; + *(counter.scalar) += 1; + *model_time = stop_time; return 0; } -int model_exit(ModelInstanceSpec* mi) +void model_destroy(ModelDesc* m) { - UNUSED(mi); - return 0; + UNUSED(m); } diff --git a/doc/content/apis/modelc/examples/model_sv_create.c b/doc/content/apis/modelc/examples/model_sv_create.c deleted file mode 100644 index 49ac8be..0000000 --- a/doc/content/apis/modelc/examples/model_sv_create.c +++ /dev/null @@ -1,13 +0,0 @@ -#include -#include - -void print_signal_names(ModelInstanceSpec* mi) -{ - SignalVector* sv_save = model_sv_create(mi); - for (SignalVector* sv = sv_save; sv->name; sv++) { - for (uint32_t i = 0; i < sv->count; i++) { - log_debug(" signal : %s", sv->signal[i]); - } - } - model_sv_destroy(sv_save); -} diff --git a/doc/content/apis/modelc/examples/signalvector_annotation.c b/doc/content/apis/modelc/examples/signalvector_annotation.c new file mode 100644 index 0000000..d2709a6 --- /dev/null +++ b/doc/content/apis/modelc/examples/signalvector_annotation.c @@ -0,0 +1,15 @@ +#include +#include + +ModelDesc* model_create(ModelDesc* m) +{ + ModelSignalIndex idx = m->index(m, "data", "counter"); + + if (idx.scalar) { + /* Set initial value. */ + const char* v = idx.sv->annotation(idx.sv, idx.signal, "initial_value"); + if (v) *(idx.scalar) = atoi(v); + } + + return m; +} diff --git a/doc/content/apis/modelc/examples/model_signalvector.c b/doc/content/apis/modelc/examples/signalvector_interface.c similarity index 84% rename from doc/content/apis/modelc/examples/model_signalvector.c rename to doc/content/apis/modelc/examples/signalvector_interface.c index 5e95506..efcf2a4 100644 --- a/doc/content/apis/modelc/examples/model_signalvector.c +++ b/doc/content/apis/modelc/examples/signalvector_interface.c @@ -1,9 +1,12 @@ +#include #include #include -void print_signal_vectors(ModelInstanceSpec* mi) +#define UNUSED(x) ((void)x) + +int model_step(ModelDesc* m, double* model_time, double stop_time) { - SignalVector* sv = model_sv_create(mi); + SignalVector* sv = m->sv; while (sv && sv->name) { log_debug("Signal Vector : %s", sv->name); @@ -30,4 +33,7 @@ void print_signal_vectors(ModelInstanceSpec* mi) // Next signal vector. sv++; } -} \ No newline at end of file + + *model_time = stop_time; + return 0; +} diff --git a/doc/content/apis/modelc/examples/simmock_configure.c b/doc/content/apis/modelc/examples/simmock_configure.c index 3ce29f5..fc0a31d 100644 --- a/doc/content/apis/modelc/examples/simmock_configure.c +++ b/doc/content/apis/modelc/examples/simmock_configure.c @@ -1,5 +1,5 @@ #include -#include +#include int test_setup(void** state) { @@ -19,7 +19,7 @@ int test_setup(void** state) }; SimMock* mock = simmock_alloc(inst_names, ARRAY_SIZE(inst_names)); simmock_configure(mock, argv, ARRAY_SIZE(argv), ARRAY_SIZE(inst_names)); - simmock_load(mock, true); + simmock_load(mock); simmock_setup(mock, "signal", "network"); /* Return the mock. */ diff --git a/doc/content/apis/modelc/examples/simmock_frame_check.c b/doc/content/apis/modelc/examples/simmock_frame_check.c index 340ec15..5f80e76 100644 --- a/doc/content/apis/modelc/examples/simmock_frame_check.c +++ b/doc/content/apis/modelc/examples/simmock_frame_check.c @@ -1,5 +1,5 @@ #include -#include +#include void test_network__frame_check(void** state) { diff --git a/doc/content/apis/modelc/examples/simmock_signal_check.c b/doc/content/apis/modelc/examples/simmock_signal_check.c index 501ae0c..9a5b2b0 100644 --- a/doc/content/apis/modelc/examples/simmock_signal_check.c +++ b/doc/content/apis/modelc/examples/simmock_signal_check.c @@ -1,5 +1,5 @@ #include -#include +#include #define SIG_task_init_done 1 #define SIG_task_5_active 2 diff --git a/doc/content/apis/modelc/mcl/index.md b/doc/content/apis/modelc/mcl/index.md index b33448f..ddc246c 100644 --- a/doc/content/apis/modelc/mcl/index.md +++ b/doc/content/apis/modelc/mcl/index.md @@ -71,7 +71,7 @@ typedef struct MclAdapterDesc { ```c typedef struct MclInstanceDesc { int * model_instance; - int * channel; + int * mcl_channel_sv; MclStrategyDesc * strategy; int models; } diff --git a/doc/content/apis/modelc/model/index.md b/doc/content/apis/modelc/model/index.md index 418ec43..087ab1d 100644 --- a/doc/content/apis/modelc/model/index.md +++ b/doc/content/apis/modelc/model/index.md @@ -5,67 +5,39 @@ linkTitle: Model ## Model API -The Model API allows model developers and integrators to interface with a -Dynamic Simulation Environment via a connection with a Simulation Bus. +The Model API allows model developers and integrators to implement models which +can be connected to a Simulation Bus. +Models are able to exchange signals with other models via this connection to +a Simulation Bus. +A runtime environment, such as the ModelC Runtime/Importer, will load the +model and also manages the connection with the Simulation Bus. +The Model API provides two simple interfaces which facilitate the development +of models; the Model Interface which is concerned with the model lifecycle; and +the Signal Interface which facilitates signal exchange. -## Signal Vector Interface +### Model Interface -Models exchange signals via the Simulation Bus using a Signal Vector. Signal -Vectors represent a logical grouping of signals (i.e. a collection of signals -belonging to an ECU interface or bus), they are defined by a `SignalGroup` -schema kind, and a Signal Vector can represent either scalar or binary values. - - -### Component Diagram - - - -![](model-signal-vector.png) - +The Model Interface provides the necessary types, methods and objects required +for implementing a model. Such a model can easily participate in a simulation +by being connecting to a Simulation Bus (using the ModelC Importer) and then +exchanging signals with other models in that simulation by using the +provided SignalVector objects (which represent those signals). -### Example +Additionally, model implementers may extend or modify the Model Interface +to support more complex integrations. -{{< readfile file="../examples/model_signalvector.c" code="true" lang="c" >}} +### Signal Vector Interface - - -## Model Interface - - -The Model Interface must be implemented by a Model. It includes the functions -necessary for a Model to be loaded and executed in the Dynamic Simulation -Environment. +Models exchange signals via the Simulation Bus using a Signal Vector. Signal +Vectors represent a logical grouping of signals (e.g. a collection of signals +belonging to an ECU interface or bus). They are defined by a `SignalGroup` +schema kind and may be configured to represent either either scalar +(double, int, bool) or binary values. ### Component Diagram @@ -87,14 +59,14 @@ m1 -left-> SBif m2 -right-> SBif package "Model" { - component "ModelC Lib" as ModelC - interface "ModelInterfaceVTable" as MIvt - component "Model" as Mdl + component "Runtime" as ModelC + interface "ModelVTable" as Mvt + component "Model" as Mdl } SBif <-down- ModelC -Mdl -up- MIvt -MIvt )-up- ModelC +Mdl -up- Mvt +Mvt )-up- ModelC center footer Dynamic Simulation Environment @@ -106,100 +78,54 @@ center footer Dynamic Simulation Environment ![](model-interface.png) -### Example +### Example (Model Interface) {{< readfile file="../examples/model_interface.c" code="true" lang="c" >}} +### Example (Signal Vector Interface) -## Typedefs +{{< readfile file="../examples/signalvector_interface.c" code="true" lang="c" >}} -### ChannelSpec -```c -typedef struct ChannelSpec { - const char * name; - const char * alias; - void * private; -} -``` -### ModelCArguments -```c -typedef struct ModelCArguments { - const char * transport; - char * uri; - const char * host; - uint32_t port; - double timeout; - uint8_t log_level; - double step_size; - double end_time; - uint32_t uid; - const char * name; - const char * file; - const char * path; - int * yaml_doc_list; - int timeout_set_by_cli; - int log_level_set_by_cli; - uint32_t steps; -} -``` - -### ModelChannelDesc - -```c -typedef struct ModelChannelDesc { - const char * name; - const char * function_name; - const char ** signal_names; - uint32_t signal_count; - bool propagator_source_channel; - bool propagator_target_channel; - double * vector_double; - void ** vector_binary; - uint32_t * vector_binary_size; - uint32_t * vector_binary_buffer_size; -} -``` +## Typedefs -### ModelDefinitionSpec +### ModelDesc ```c -typedef struct ModelDefinitionSpec { - const char * name; - const char * path; - const char * file; - char * full_path; - int * doc; - int * channels; +typedef struct ModelDesc { + ModelVTable vtable; + ModelIndex index; + SimulationSpec * sim; + ModelInstanceSpec * mi; + SignalVector * sv; } ``` -### ModelInstanceSpec +### ModelSignalIndex ```c -typedef struct ModelInstanceSpec { - uint32_t uid; - char * name; - ModelDefinitionSpec model_definition; - int * spec; - int * propagators; - int * yaml_doc_list; - void * private; +typedef struct ModelSignalIndex { + SignalVector * sv; + double * scalar; + void ** binary; + uint32_t vector; + uint32_t signal; } ``` -### ModelInterfaceVTable +### ModelVTable ```c -typedef struct ModelInterfaceVTable { - ModelSetupHandler setup; - ModelDoStepHandler step; - ModelExitHandler exit; +typedef struct ModelVTable { + ModelCreate create; + ModelStep step; + ModelDestroy destroy; + ModelIndex index; } ``` @@ -210,7 +136,7 @@ typedef struct SignalVector { const char * name; const char * alias; const char * function_name; - bool is_binary; + _Bool is_binary; uint32_t count; const char ** signal; BinarySignalAppendFunc append; @@ -234,115 +160,294 @@ typedef struct SignalVectorVTable { } ``` -### SimulationSpec +## Functions -```c -typedef struct SimulationSpec { - const char * transport; - char * uri; - uint32_t uid; - double timeout; - double step_size; - double end_time; - ModelInstanceSpec * instance_list; -} -``` +### model_create -## Functions +> Optional method of `ModelVTable` interface. + +Called by the Model Runtime to create a new instance of this model. + +The `model_create()` method may extend or mutilate the provided Model +Descriptor. When extending the Model Descriptor _and_ allocating additional +resources then the `model_destroy()` method should also be implemented. + +Fault conditions can be communicated to the caller by setting variable +`errno` to a non-zero value. Additionally, `log_fatal()` can be used to +immediately halt execution of a model. + +#### Parameters + +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +#### Returns + +NULL +: The Channel was configured. + +(ModelDesc*) +: Pointer to a new, or mutilated, version of the Model Descriptor object. The + original Model Descriptor object will be released by the Model Runtime (i.e. + don't call `free()`). + +errno <> 0 (indirect) +: Indicates an error condition. + +#### Example + + +{{< readfile file="../examples/model_create.c" code="true" lang="c" >}} + + + + + +### model_destroy -### model_configure_channel +> Optional method of `ModelVTable` interface. -Configure a connection from this Model to a Channel on the Simulation Bus. The -Channel can then be represented by a Signal Vector making access to individual -Signals and their configuration (annotations) easy. +Called by the Model Runtime at the end of a simulation, the `model_destroy()` +function may be implemented by a Model Integrator to perform any custom +cleanup operations (e.g. releasing instance related resources, such as open +files or allocated memory). #### Parameters -model_instance (ModelInstanceSpec*) -: The Model Instance object (provided via the `model_setup()` function of the - Model API). +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. -channel_desc (ModelChannelDesc*) -: A channel descriptor object which defines the Channel and Model Function names - which should be configured. + + + +### model_index_ + +> Provided method (by the Runtime). Model implementers may specify + a different index method by mutilating the Model Descriptor in the + `model_create()` method, or even at runtime. + +A model may use this method to index a signal that is contained within the +Signal Vectors of the Model Descriptor. + +#### Parameters + +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +vname (const char*) +: The name (alias) of the Signal Vector. + +sname (const char*) +: The name of the signal within the Signal Vector. + +#### Returns + +ModelSignalIndex +: An index. When valid, either the `scalar` or `binary` fields will be set to + a valid pointer (i.e. not NULL). + + + + +### model_step + +> Mandatory method of `ModelVTable` interface. Alternatively, Model implementers + may specify the `ModelVTable.step` method dynamically by mutilating the + Model Descriptor in the `model_create()` method, or even at runtime. + +Called by the Model Runtime to step the model for a time interval. + +#### Parameters + +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +model_time (double*) +: (in/out) Specifies the model time for this step of the model. + +stop_time (double) +: Specifies the stop time for this step of the model. The model step should not + exceed this time. #### Returns 0 -: The Channel was configured. +: The step completed without error. -+VE -: An error occurred during the registration of the Channel. +<>0 +: An error occurred at some point during the step execution. +model_time (via parameter) +: The final model time reached for this step. This value may be less than + `stop_time` if a step decides to return early. -### model_function_register +### signal_annotation -Register a Model Function. A Model may register one or more Model Functions -with repeated calls to this function. +Get an annotation from a signal definition. #### Parameters -model_instance (ModelInstanceSpec*) -: The Model Instance object (provided via the `model_setup()` function of the - Model API). +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. name (const char*) -: The name of the Model Function. +: The name of the annotation. + +#### Returns + +const char* +: The annotation value. + +#### Example (Annotation Specification) + + +```yaml +kind: SignalGroup +metadata: + name: data +spec: + signals: + - signal: counter + annotations: + initial_value: 10 +``` + +#### Example (Code Usage) + + +{{< readfile file="../examples/signalvector_annotation.c" code="true" lang="c" >}} + -step_size (double) -: The step size of the Model Function. +NULL +: The requested annotation was not found. -do_step_handler (ModelDoStepHandler) -: The "do step" function of the Model Function. + + +### signal_append + +Append data to the end of the specified binary signal. The append method will +resize the buffers of the binary signal as required. + +#### Parameters + +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +data (void*) +: Address/pointer to the data which should be appended to the binary signal. + +len (uint32_t) +: Length of the provided data buffer being appended. #### Returns 0 -: The model function was registered. +: The operation completed without error. -(errno) -: An error occurred during registration of the model function. The return - value is the `errno` which may indicate the reason for the failure. +<>0 +: Indicates an error condition. Inspect `errno` for additional information. +### signal_codec -### model_sv_create +Return a pointer to the Codec object associated with a binary signal. -This is Model User API replacing modelc_debug.c::modelc_get_model_vectors(). +Codec objects are created when a binary signal is specified with a `mime_type` +annotation. #### Parameters -mi (ModelInstanceSpec*) -: The model instance, which holds references to the registered channels. +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. #### Returns -SignalVector (pointer to NULL terminated list) -: A list of SignalVector objects representing the signals assigned to a model. - The list is NULL terminated (sv->name == NULL). Caller to free. +void* +: The Codec object associated with the binary signal. -#### Example +NULL +: The binary signal does not have an associated Codec object. +#### Example (Codec Specification) -{{< readfile file="../examples/model_sv_create.c" code="true" lang="c" >}} +```yaml +kind: SignalGroup +metadata: + name: network + labels: + channel: network_vector + annotations: + vector_type: binary +spec: + signals: + - signal: can_bus + annotations: + mime_type: application/x-automotive-bus; interface=stream; type=frame; bus=can; schema=fbs; bus_id=1; node_id=2; interface_id=3 +``` + +#### Reference +[Network Codec API](https://github.com/boschglobal/dse.standards/tree/main/dse/ncodec) -### model_sv_destroy -The underlying objects of a SignalVector object (e.g. from ModelC object) -are not affected by calling this method. + + +### signal_release + +Release the resources allocated to a binary signal (e.g. free the buffer). #### Parameters sv (SignalVector*) -: The SignalVector object to destroy. Should be the same object as returned - from the call to `model_sv_create()`. +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +#### Returns + +0 +: The operation completed without error. + +<>0 +: Indicates an error condition. Inspect `errno` for additional information. + + + +### signal_reset + +Reset a binary signal (e.g. sets its buffer length to 0). The buffers of the +binary signal are not released (see `signal_release()`). + +#### Parameters + +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +#### Returns + +0 +: The operation completed without error. +<>0 +: Indicates an error condition. Inspect `errno` for additional information. diff --git a/doc/content/apis/modelc/model/model-interface.png b/doc/content/apis/modelc/model/model-interface.png index 97e6201..10c0007 100644 Binary files a/doc/content/apis/modelc/model/model-interface.png and b/doc/content/apis/modelc/model/model-interface.png differ diff --git a/doc/content/apis/modelc/model/model-signal-vector.png b/doc/content/apis/modelc/model/model-signal-vector.png deleted file mode 100644 index b6ef0b3..0000000 Binary files a/doc/content/apis/modelc/model/model-signal-vector.png and /dev/null differ diff --git a/doc/content/apis/modelc/runtime/index.md b/doc/content/apis/modelc/runtime/index.md new file mode 100644 index 0000000..241facf --- /dev/null +++ b/doc/content/apis/modelc/runtime/index.md @@ -0,0 +1,127 @@ +--- +title: Runtime API Reference +linkTitle: Runtime +--- +## Runtime API + + +The Runtime API provides methods for implementing a model Runtime/Importer +which can be used to load, configure and execute a model. + + + + +## Typedefs + +### ChannelSpec + +```c +typedef struct ChannelSpec { + const char * name; + const char * alias; + void * private; +} +``` + +### ModelCArguments + +```c +typedef struct ModelCArguments { + const char * transport; + char * uri; + const char * host; + int port; + double timeout; + int log_level; + double step_size; + double end_time; + int uid; + const char * name; + const char * file; + const char * path; + void * yaml_doc_list; + int timeout_set_by_cli; + int log_level_set_by_cli; + int steps; +} +``` + +### ModelChannelDesc + +```c +typedef struct ModelChannelDesc { + const char * name; + const char * function_name; + const char ** signal_names; + int signal_count; + int propagator_source_channel; + int propagator_target_channel; + double * vector_double; + void ** vector_binary; + int * vector_binary_size; + int * vector_binary_buffer_size; +} +``` + +### ModelDefinitionSpec + +```c +typedef struct ModelDefinitionSpec { + const char * name; + const char * path; + const char * file; + char * full_path; + void * doc; + void * channels; +} +``` + +### ModelInstanceSpec + +```c +typedef struct ModelInstanceSpec { + int uid; + char * name; + int * model_desc; + ModelDefinitionSpec model_definition; + void * spec; + void * yaml_doc_list; + void * private; +} +``` + +### SimulationSpec + +```c +typedef struct SimulationSpec { + const char * transport; + char * uri; + int uid; + double timeout; + double step_size; + double end_time; + ModelInstanceSpec * instance_list; +} +``` + +## Functions + +### modelc_step + +Execute a simulation step with the provided step size for all model +functions of the given model instance. + +#### Parameters + +model_instance : ModelInstanceSpec (pointer to) + The model instance, which holds references to the registered channels +and model functions. step_size : double The duration simulation step to be +performed (in seconds). + +#### Returns + + 0 : Success. + +ve/-ve : Failure, inspect `errno` for the failing condition. + + + diff --git a/doc/content/apis/modelc/simmock/index.md b/doc/content/apis/modelc/simmock/index.md index 2a6c112..aa5f29f 100644 --- a/doc/content/apis/modelc/simmock/index.md +++ b/doc/content/apis/modelc/simmock/index.md @@ -91,6 +91,16 @@ center footer Dynamic Simulation Environment ## Typedefs +### BinaryCheck + +```c +typedef struct BinaryCheck { + int index; + int * buffer; + int len; +} +``` + ### FrameCheck ```c @@ -98,6 +108,7 @@ typedef struct FrameCheck { int frame_id; int offset; int value; + bool not_present; } ``` @@ -110,8 +121,7 @@ typedef struct ModelMock { int * sv_signal; int * sv_network; int * sv_save; - int model_setup_func; - int model_exit_func; + int vtable; } ``` @@ -161,6 +171,29 @@ SimMock* +### simmock_binary_check + +Check the content of a binary signal. + +#### Parameters + +mock (SimMock*) +: A SimMock object. + +model_name (const char*) +: The name of the model to check. + +checks (BinaryCheck*) +: Array of BinaryCheck objects. + +count (size_t) +: The number elements in the checks array. + +func (BinaryCheckFunc) +: Optional function pointer for performing the binary check. + + + ### simmock_configure Configure a SimMock object with a list of command line arguments (as would be @@ -270,8 +303,40 @@ Load all of the models referenced by a SimMock object. mock (SimMock*) : A SimMock object. -expect_exit_function (bool) -: Indicate that model libraries should contain a `model_exit` function. + + + +### simmock_load_model_check + +Check the condition/state of a loaded model. + +#### Parameters + +mock (SimMock*) +: A SimMock object. + +expect_create_func (bool) +: Indicate that model libraries should contain a `model_create` function. + +expect_step_func (bool) +: Indicate that model libraries should contain a `model_step` function. + +expect_destroy_func (bool) +: Indicate that model libraries should contain a `model_destroy` function. + + + +### simmock_print_binary_signals + +Print the binary signals contained in each network vector of each model. + +#### Parameters + +mock (SimMock*) +: A SimMock object. + +level (int) +: The log level to print at. diff --git a/dse/mocks/simmock.c b/dse/mocks/simmock.c index aab8aa1..6c2b591 100644 --- a/dse/mocks/simmock.c +++ b/dse/mocks/simmock.c @@ -7,13 +7,14 @@ #include #include #include +#include #include #include /* Private interface from ncodec.c */ -extern void* _create_stream(SignalVector* sv, uint32_t idx); -extern void _free_stream(void* stream); +extern void* model_sv_stream_create(SignalVector* sv, uint32_t idx); +extern void model_sv_stream_destroy(void* stream); /* Private interface from dse/modelc/controller/step.c */ extern int step_model(ModelInstanceSpec* mi, double* model_time); @@ -85,7 +86,7 @@ static void __free_stub_sv(SignalVector* sv) if (sv->binary[i]) free(sv->binary[i]); NCodecInstance* nc = sv->ncodec[i]; if (nc) { - if (nc->stream) _free_stream(nc->stream); + if (nc->stream) model_sv_stream_destroy(nc->stream); ncodec_close((NCODEC*)nc); } } @@ -183,10 +184,8 @@ Parameters mock (SimMock*) : A SimMock object. -expect_exit_function (bool) -: Indicate that model libraries should contain a `model_exit` function. */ -void simmock_load(SimMock* mock, bool expect_exit_func) +void simmock_load(SimMock* mock) { assert_non_null(mock); @@ -197,11 +196,11 @@ void simmock_load(SimMock* mock, bool expect_exit_func) void* handle = dlopen( model->mi->model_definition.full_path, RTLD_NOW | RTLD_LOCAL); + if (handle == NULL) log_notice("ERROR: dlopen call: %s", dlerror()); assert_non_null(handle); - model->model_setup_func = dlsym(handle, MODEL_SETUP_FUNC_STR); - model->model_exit_func = dlsym(handle, MODEL_EXIT_FUNC_STR); - assert_non_null(model->model_setup_func); - if (expect_exit_func) assert_non_null(model->model_exit_func); + model->vtable.create = dlsym(handle, MODEL_CREATE_FUNC_NAME); + model->vtable.step = dlsym(handle, MODEL_STEP_FUNC_NAME); + model->vtable.destroy = dlsym(handle, MODEL_DESTROY_FUNC_NAME); } /* Save a doc_list reference for simmock_free(). */ @@ -209,8 +208,43 @@ void simmock_load(SimMock* mock, bool expect_exit_func) } +/** +simmock_load_model_check +======================== + +Check the condition/state of a loaded model. + +Parameters +---------- +mock (SimMock*) +: A SimMock object. + +expect_create_func (bool) +: Indicate that model libraries should contain a `model_create` function. + +expect_step_func (bool) +: Indicate that model libraries should contain a `model_step` function. + +expect_destroy_func (bool) +: Indicate that model libraries should contain a `model_destroy` function. +*/ +void simmock_load_model_check(ModelMock* model, bool expect_create_func, + bool expect_step_func, bool expect_destroy_func) +{ + assert_non_null(model); + + log_debug("Load model check: %s", model->name); + assert_non_null(model->mi); + + if (expect_create_func) assert_non_null(model->vtable.create); + if (expect_step_func) assert_non_null(model->vtable.step); + if (expect_destroy_func) assert_non_null(model->vtable.destroy); +} + + static SignalVector* __stub_sv(SignalVector* sv) { + if (sv == NULL) return NULL; SignalVector* stub_sv = calloc(1, sizeof(SignalVector)); stub_sv->mi = sv->mi; @@ -232,12 +266,12 @@ static SignalVector* __stub_sv(SignalVector* sv) for (uint32_t i = 0; i < stub_sv->count; i++) { stub_sv->signal[i] = sv->signal[i]; stub_sv->mime_type[i] = sv->mime_type[i]; - void* stream = _create_stream(stub_sv, i); + void* stream = model_sv_stream_create(stub_sv, i); NCODEC* nc = ncodec_open(stub_sv->mime_type[i], stream); if (nc) { stub_sv->ncodec[i] = nc; } else { - _free_stream(stream); + model_sv_stream_destroy(stream); } } @@ -266,13 +300,25 @@ sig_name (const char*) void simmock_setup(SimMock* mock, const char* sig_name, const char* net_name) { assert_non_null(mock); + SignalVector* sv = NULL; for (ModelMock* model = mock->model; model->name; model++) { - log_debug("Call model_setup(): %s", model->name); - int rc = model->model_setup_func(model->mi); + log_debug("Setup the Model: %s", model->name); + log_debug("Call modelc_model_create(): %s", model->name); + int rc = modelc_model_create(&mock->sim, model->mi, &model->vtable); assert_int_equal(rc, 0); - SignalVector* sv = model_sv_create(model->mi); - model->sv_save = sv; + + assert_non_null(model->mi->model_desc); + assert_ptr_equal(&mock->sim, model->mi->model_desc->sim); + assert_ptr_equal(model->mi, model->mi->model_desc->mi); + assert_ptr_equal( + model->vtable.create, model->mi->model_desc->vtable.create); + assert_ptr_equal( + model->vtable.step, model->mi->model_desc->vtable.step); + assert_ptr_equal( + model->vtable.destroy, model->mi->model_desc->vtable.destroy); + assert_non_null(model->mi->model_desc->sv); + model->sv_save = sv = model->mi->model_desc->sv; /* Locate the vectors ... */ while (sv && sv->name) { @@ -286,14 +332,18 @@ void simmock_setup(SimMock* mock, const char* sig_name, const char* net_name) } /* Setup the mocked signal exchange. */ - mock->sv_signal = calloc(1, sizeof(SignalVector)); - mock->sv_signal->count = mock->model->sv_signal->count; - mock->sv_signal->scalar = calloc(mock->sv_signal->count, sizeof(double)); - memcpy(mock->sv_signal->scalar, mock->model->sv_signal->scalar, - (mock->sv_signal->count * sizeof(double))); - - mock->sv_network_rx = __stub_sv(mock->model->sv_network); - mock->sv_network_tx = __stub_sv(mock->model->sv_network); + if (mock->model->sv_signal) { + mock->sv_signal = calloc(1, sizeof(SignalVector)); + mock->sv_signal->count = mock->model->sv_signal->count; + mock->sv_signal->scalar = + calloc(mock->sv_signal->count, sizeof(double)); + memcpy(mock->sv_signal->scalar, mock->model->sv_signal->scalar, + (mock->sv_signal->count * sizeof(double))); + } + if (mock->model->sv_network) { + mock->sv_network_rx = __stub_sv(mock->model->sv_network); + mock->sv_network_tx = __stub_sv(mock->model->sv_network); + } } @@ -323,11 +373,13 @@ int simmock_step(SimMock* mock, bool assert_rc) assert_non_null(mock); /* Copy simmock->binary_tx to simmock->binary_rx */ - for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { - mock->sv_network_rx->reset(mock->sv_network_rx, i); - mock->sv_network_rx->append(mock->sv_network_rx, i, - mock->sv_network_tx->binary[i], mock->sv_network_tx->length[i]); - mock->sv_network_tx->reset(mock->sv_network_tx, i); + if (mock->sv_network_rx && mock->sv_network_tx) { + for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { + mock->sv_network_rx->reset(mock->sv_network_rx, i); + mock->sv_network_rx->append(mock->sv_network_rx, i, + mock->sv_network_tx->binary[i], mock->sv_network_tx->length[i]); + mock->sv_network_tx->reset(mock->sv_network_tx, i); + } } int rc = 0; @@ -337,24 +389,31 @@ int simmock_step(SimMock* mock, bool assert_rc) model->sv_signal->scalar[i] = mock->sv_signal->scalar[i]; } /* Copy binary from simmock->binary_rx. */ - for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { - model->sv_network->reset(model->sv_network, i); - model->sv_network->append(model->sv_network, i, - mock->sv_network_rx->binary[i], mock->sv_network_rx->length[i]); + if (mock->sv_network_rx && mock->sv_network_tx) { + for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { + model->sv_network->reset(model->sv_network, i); + model->sv_network->append(model->sv_network, i, + mock->sv_network_rx->binary[i], + mock->sv_network_rx->length[i]); + } } rc |= modelc_step(model->mi, mock->step_size); - for (uint32_t i = 0; i < model->sv_network->count; i++) { - mock->sv_network_tx->append(mock->sv_network_tx, i, - model->sv_network->binary[i], model->sv_network->length[i]); + if (mock->sv_network_rx && mock->sv_network_tx) { + for (uint32_t i = 0; i < model->sv_network->count; i++) { + mock->sv_network_tx->append(mock->sv_network_tx, i, + model->sv_network->binary[i], model->sv_network->length[i]); + } } /* Copy scalars to simmock->scalars. */ for (uint32_t i = 0; i < mock->sv_signal->count; i++) { mock->sv_signal->scalar[i] = model->sv_signal->scalar[i]; } /* Copy binary to simmock->binary_tx. */ - for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { - mock->sv_network_tx->append(mock->sv_network_tx, i, - model->sv_network->binary[i], model->sv_network->length[i]); + if (mock->sv_network_rx && mock->sv_network_tx) { + for (uint32_t i = 0; i < mock->sv_network_tx->count; i++) { + mock->sv_network_tx->append(mock->sv_network_tx, i, + model->sv_network->binary[i], model->sv_network->length[i]); + } } if (assert_rc) assert_int_equal(rc, 0); } @@ -377,15 +436,6 @@ mock (SimMock*) void simmock_exit(SimMock* mock) { assert_non_null(mock); - - for (ModelMock* model = mock->model; model->name; model++) { - log_debug("Call model_exit(): %s", model->name); - if (model->model_exit_func) { - int rc = model->model_exit_func(model->mi); - assert_int_equal(rc, 0); - } - if (model->sv_save) model_sv_destroy(model->sv_save); - } modelc_exit(&mock->sim); } @@ -468,6 +518,53 @@ void simmock_signal_check(SimMock* mock, const char* model_name, } +/** +simmock_binary_check +==================== + +Check the content of a binary signal. + +Parameters +---------- +mock (SimMock*) +: A SimMock object. + +model_name (const char*) +: The name of the model to check. + +checks (BinaryCheck*) +: Array of BinaryCheck objects. + +count (size_t) +: The number elements in the checks array. + +func (BinaryCheckFunc) +: Optional function pointer for performing the binary check. +*/ +void simmock_binary_check(SimMock* mock, const char* model_name, + BinaryCheck* checks, size_t count, BinaryCheckFunc func) +{ + assert_non_null(mock); + ModelMock* check_model = simmock_find_model(mock, model_name); + assert_non_null(check_model); + + SignalVector* sv = check_model->sv_network; + for (size_t i = 0; i < count; i++) { + if (func) { + int rc = func(&checks[i], sv); + assert_int_equal(rc, 0); + } else { + assert_true(checks[i].index < sv->count); + assert_int_equal(sv->length[checks[i].index], checks[i].len); + if (sv->binary[checks[i].index]) { + assert_memory_equal(sv->binary[checks[i].index], + checks[i].buffer, checks[i].len); + } + } + } +} + + /** simmock_frame_check =================== @@ -692,6 +789,47 @@ void simmock_print_scalar_signals(SimMock* mock, int level) } +/** +simmock_print_binary_signals +============================ + +Print the binary signals contained in each network vector of each model. + +Parameters +---------- +mock (SimMock*) +: A SimMock object. + +level (int) +: The log level to print at. +*/ +void simmock_print_binary_signals(SimMock* mock, int level) +{ + assert_non_null(mock); + + log_at(level, "SignalVector (Network) at %f:", mock->model_time); + for (ModelMock* model = mock->model; model->name; model++) { + log_at(level, " Model Instance: %s", model->name); + SignalVector* sv = model->sv_network; + for (uint32_t idx = 0; idx < sv->count; idx++) { + log_at(level, " Network Signal: %s (%d)", sv->signal[idx], + sv->length[idx]); + uint8_t* b = sv->binary[idx]; + for (uint32_t i = 0; i < sv->length[idx]; i += 8) { + char buffer[100] = {}; + for (uint32_t j = i; j < i + 8; j++) { + if (j < sv->length[idx]) { + snprintf(buffer + strlen(buffer), + sizeof(buffer + strlen(buffer)), "%02x ", b[j]); + } + } + log_at(level, "%s", buffer); + } + } + } +} + + /** simmock_print_network_frames ============================ diff --git a/dse/mocks/simmock.h b/dse/mocks/simmock.h index 49b450c..f16a962 100644 --- a/dse/mocks/simmock.h +++ b/dse/mocks/simmock.h @@ -8,6 +8,7 @@ #include #include #include +#include #define UNUSED(x) ((void)x) @@ -115,8 +116,7 @@ typedef struct ModelMock { SignalVector* sv_signal; SignalVector* sv_network; SignalVector* sv_save; - ModelSetupHandler model_setup_func; - ModelExitHandler model_exit_func; + ModelVTable vtable; } ModelMock; typedef struct SimMock { @@ -137,7 +137,9 @@ typedef struct SimMock { void* simmock_alloc(const char* inst_names[], size_t count); void simmock_configure( SimMock* mock, char** argv, size_t argc, size_t expect_model_count); -void simmock_load(SimMock* mock, bool expect_exit_func); +void simmock_load(SimMock* mock); +void simmock_load_model_check(ModelMock* model, bool expect_create_func, + bool expect_step_func, bool expect_destroy_func); void simmock_setup(SimMock* mock, const char* sig_name, const char* net_name); int simmock_step(SimMock* mock, bool assert_rc); void simmock_exit(SimMock* mock); @@ -155,6 +157,12 @@ typedef struct SignalCheck { double value; } SignalCheck; +typedef struct BinaryCheck { + uint32_t index; + uint8_t* buffer; + uint32_t len; +} BinaryCheck; + typedef struct FrameCheck { uint32_t frame_id; uint8_t offset; @@ -163,15 +171,19 @@ typedef struct FrameCheck { } FrameCheck; typedef int (*SignalCheckFunc)(SignalCheck* check, SignalVector* sv); +typedef int (*BinaryCheckFunc)(BinaryCheck* check, SignalVector* sv); void simmock_signal_check(SimMock* mock, const char* model_name, SignalCheck* checks, size_t count, SignalCheckFunc func); +void simmock_binary_check(SimMock* mock, const char* model_name, + BinaryCheck* checks, size_t count, BinaryCheckFunc func); void simmock_frame_check(SimMock* mock, const char* model_name, const char* sig_name, FrameCheck* checks, size_t count); /* Information API. */ void simmock_print_scalar_signals(SimMock* mock, int level); +void simmock_print_binary_signals(SimMock* mock, int level); void simmock_print_network_frames(SimMock* mock, int level); diff --git a/dse/modelc/CMakeLists.txt b/dse/modelc/CMakeLists.txt index f483c55..4f55235 100644 --- a/dse/modelc/CMakeLists.txt +++ b/dse/modelc/CMakeLists.txt @@ -189,6 +189,7 @@ set(DSE_MODELC_PUBLIC_HEADERS gateway.h mcl.h model.h + runtime.h schema.h ) add_subdirectory(adapter) diff --git a/dse/modelc/controller/controller.c b/dse/modelc/controller/controller.c index a6c156a..37313a7 100644 --- a/dse/modelc/controller/controller.c +++ b/dse/modelc/controller/controller.c @@ -310,16 +310,13 @@ void controller_exit(SimulationSpec* sim) ModelInstanceSpec* _instptr = sim->instance_list; while (_instptr && _instptr->name) { - ModelInstancePrivate* mip = _instptr->private; - ControllerModel* cm = mip->controller_model; - if (cm->model_exit_func == NULL) goto exit_next; - - log_notice("Call symbol: %s ...", MODEL_EXIT_FUNC_STR); - errno = 0; - int rc = cm->model_exit_func(_instptr); - if (rc) { - if (errno == 0) errno = rc; - log_error("model_exit_func() failed"); + if (_instptr->model_desc) { + if (_instptr->model_desc->vtable.destroy == NULL) goto exit_next; + + log_notice("Call symbol: %s ...", MODEL_DESTROY_FUNC_NAME); + errno = 0; + _instptr->model_desc->vtable.destroy(_instptr->model_desc); + if (errno) log_error(MODEL_DESTROY_FUNC_NAME "() failed"); } exit_next: /* Next instance? */ diff --git a/dse/modelc/controller/controller.h b/dse/modelc/controller/controller.h index 7ad38c7..435926f 100644 --- a/dse/modelc/controller/controller.h +++ b/dse/modelc/controller/controller.h @@ -30,23 +30,23 @@ typedef struct ModelFunctionChannel { typedef struct ModelFunction { - const char* name; - double step_size; - ModelDoStepHandler do_step_handler; + const char* name; + double step_size; - HashMap - channels; /* Collection of ModelFunctionChannel, Key is channel_name. */ + /* Collection of ModelFunctionChannel, Key is channel_name. */ + HashMap channels; } ModelFunction; typedef struct ControllerModel { /* Controller specific objects (placed in Model instance). */ const char* model_dynlib_filename; - HashMap model_functions; /* Collection of ModelFunction, Key is Model - Function name. */ - ModelSetupHandler model_setup_func; - ModelExitHandler model_exit_func; + /* Collection of ModelFunction, Key is Model Function name. */ + HashMap model_functions; + + /* Model interface vTable. */ + ModelVTable vtable; } ControllerModel; @@ -67,7 +67,7 @@ DLL_PRIVATE int controller_init(Endpoint* endpoint); DLL_PRIVATE int controller_init_channel(ModelInstanceSpec* model_instance, const char* channel_name, const char** signal_name, uint32_t signal_count); -/* These are called indirectly from the Model, via model_function_register() +/* These are called indirectly from the Model, via _model_function_register() and model_configure_channel_*(). */ DLL_PRIVATE int controller_register_model_function( ModelInstanceSpec* model_instance, ModelFunction* model_function); diff --git a/dse/modelc/controller/loader.c b/dse/modelc/controller/loader.c index 01cf763..ac4ca77 100644 --- a/dse/modelc/controller/loader.c +++ b/dse/modelc/controller/loader.c @@ -10,15 +10,15 @@ #include #include #include +#include extern Controller* controller_object_ref(void); -static ModelSetupHandler __model_setup_func = NULL; -static ModelExitHandler __model_exit_func = NULL; -extern int __model_gw_setup__(ModelInstanceSpec* mi); -extern int __model_gw_exit__(ModelInstanceSpec* mi); +extern ModelDesc* __model_gw_create__(ModelDesc* m); +extern int __model_gw_step__(ModelDesc* m, double* model_time, double stop_time); +extern void __model_gw_destroy__(ModelDesc* m); int controller_load_model(ModelInstanceSpec* model_instance) @@ -27,8 +27,6 @@ int controller_load_model(ModelInstanceSpec* model_instance) ModelInstancePrivate* mip = model_instance->private; ControllerModel* controller_model = mip->controller_model; assert(controller_model); - ModelSetupHandler model_setup_func = NULL; - ModelExitHandler model_exit_func = NULL; const char* dynlib_filename = model_instance->model_definition.full_path; errno = 0; @@ -40,32 +38,28 @@ int controller_load_model(ModelInstanceSpec* model_instance) log_notice("ERROR: dlopen call: %s", dlerror()); goto error_dl; } - log_notice("Loading symbol: %s ...", MODEL_SETUP_FUNC_STR); - model_setup_func = dlsym(handle, MODEL_SETUP_FUNC_STR); - if (model_setup_func == NULL) { - log_notice("ERROR: dlsym call: %s", dlerror()); - goto error_dl; - } - log_notice("Loading optional symbol: %s ...", MODEL_EXIT_FUNC_STR); - model_exit_func = dlsym(handle, MODEL_EXIT_FUNC_STR); - if (model_exit_func) { - log_debug("... symbol loaded"); - } + /* Load the model interface.*/ + controller_model->vtable.create = dlsym(handle, MODEL_CREATE_FUNC_NAME); + log_notice("Loading symbol: %s ... %s", MODEL_CREATE_FUNC_NAME, + controller_model->vtable.create ? "ok" : "not found"); + controller_model->vtable.step = dlsym(handle, MODEL_STEP_FUNC_NAME); + log_notice("Loading symbol: %s ... %s", MODEL_STEP_FUNC_NAME, + controller_model->vtable.step ? "ok" : "not found"); + controller_model->vtable.destroy = + dlsym(handle, MODEL_DESTROY_FUNC_NAME); + log_notice("Loading symbol: %s ... %s", MODEL_DESTROY_FUNC_NAME, + controller_model->vtable.destroy ? "ok" : "not found"); + } else { if (dse_yaml_find_node( model_instance->model_definition.doc, "spec/runtime/gateway")) { log_notice("Using gateway symbols: ..."); - model_setup_func = __model_gw_setup__; - model_exit_func = __model_gw_exit__; - } else { - log_notice("Using linked/registered symbols: ..."); - model_setup_func = __model_setup_func; - model_exit_func = __model_exit_func; + controller_model->vtable.create = __model_gw_create__; + controller_model->vtable.step = __model_gw_step__; + controller_model->vtable.destroy = __model_gw_destroy__; } } - controller_model->model_setup_func = model_setup_func; - controller_model->model_exit_func = model_exit_func; return 0; error_dl: @@ -106,18 +100,18 @@ int controller_load_models(SimulationSpec* sim) log_error("controller_load_model() failed!"); break; } - /* Call model_setup(), inversion of control. */ - log_notice("Call symbol: %s ...", MODEL_SETUP_FUNC_STR); - if (cm->model_setup_func == NULL) { - rc = errno = EINVAL; - log_error("model_setup_func() not loaded!"); + /* Create/Setup the model. */ + if (cm->vtable.create == NULL && cm->vtable.step == NULL) { + log_error("Model interface not complete!"); + log_error(" %s (%p)", MODEL_CREATE_FUNC_NAME, cm->vtable.create); + log_error(" %s (%p)", MODEL_STEP_FUNC_NAME, cm->vtable.step); + log_error(" %s (%p)", MODEL_DESTROY_FUNC_NAME, cm->vtable.destroy); break; } - errno = 0; - rc = cm->model_setup_func(_instptr); + rc = modelc_model_create(sim, _instptr, &cm->vtable); if (rc) { if (errno == 0) errno = EINVAL; - log_error("model_setup_func() failed!"); + log_error("modelc_model_create() failed!"); break; } /* Next instance? */ diff --git a/dse/modelc/controller/modelc.c b/dse/modelc/controller/modelc.c index 91928be..543258c 100644 --- a/dse/modelc/controller/modelc.c +++ b/dse/modelc/controller/modelc.c @@ -28,6 +28,9 @@ static double __stop_request = 0; /* Very private, indicate stop request. */ +extern ModelSignalIndex __model_index__(ModelDesc* m, const char* vname, + const char* sname); + static int _destroy_model_function(void* mf, void* additional_data) { @@ -46,6 +49,11 @@ static void _destroy_model_instances(SimulationSpec* sim) ModelInstancePrivate* mip = _instptr->private; free(_instptr->name); free(_instptr->model_definition.full_path); + if (_instptr->model_desc) { + if (_instptr->model_desc->sv) + model_sv_destroy(_instptr->model_desc->sv); + free(_instptr->model_desc); + } /* ControllerModel */ ControllerModel* cm = mip->controller_model; if (cm) { @@ -137,11 +145,6 @@ int modelc_configure_model( args->yaml_doc_list = dse_yaml_load_file(md_file, args->yaml_doc_list); free(md_file); } - - /* Propagators list. */ - model_instance->propagators = - dse_yaml_find_node(model_instance->spec, "propagators"); - /* Model Definition. */ const char* selector[] = { "metadata/name" }; const char* value[] = { model_name }; @@ -285,6 +288,139 @@ int modelc_configure(ModelCArguments* args, SimulationSpec* sim) } +static int _configure_model_channel(YamlNode* ch_node, ModelInstanceSpec* mi) +{ + const char* name; + + /* Locate the channel by alias name. */ + dse_yaml_get_string(ch_node, "alias", &name); + if (name == NULL) { + /* Fallback to name. */ + dse_yaml_get_string(ch_node, "name", &name); + if (name == NULL) { + errno = EINVAL; + log_error("Could not find channel alias or name!"); + return -EINVAL; + } + } + + /* Configure the channel. */ + ModelChannelDesc channel_desc = { + .name = name, + .function_name = MODEL_STEP_FUNC_NAME, + }; + int rc = model_configure_channel(mi, &channel_desc); + if (rc != 0) { + if (errno == 0) errno = rc; + log_error("Channel could not be configured! Check stack alias."); + return -EINVAL; + } + + return 0; +} + + +static int _model_function_register( + ModelInstanceSpec* model_instance, const char* name, double step_size) +{ + int rc; + errno = 0; + + /* Create the Model Function object. */ + ModelFunction* mf = calloc(1, sizeof(ModelFunction)); + if (mf == NULL) { + log_error("ModelFunction malloc failed!"); + goto error_clean_up; + } + mf->name = name; + mf->step_size = step_size; + rc = hashmap_init(&mf->channels); + if (rc) { + log_error("Hashmap init failed for channels!"); + if (errno == 0) errno = rc; + goto error_clean_up; + } + /* Register the object with the Controller. */ + rc = controller_register_model_function(model_instance, mf); + if (rc && (errno != EEXIST)) goto error_clean_up; + return 0; + +error_clean_up: + model_function_destroy(mf); + return errno; +} + +int modelc_model_create( + SimulationSpec* sim, ModelInstanceSpec* mi, ModelVTable* model_vtable) +{ + /* Create the Model Function (to represent model_step). */ + if (model_vtable->step == NULL) { + errno = EINVAL; + log_error("Model has no " MODEL_STEP_FUNC_NAME "() function"); + return -errno; + } + int rc = _model_function_register(mi, MODEL_STEP_FUNC_NAME, sim->step_size); + if (rc != 0) { + if (errno == 0) errno = rc; + log_error("Model function registration failed!"); + return rc; + } + + /* Setup the channels and signal vector. */ + YamlNode* c_node; + if (dse_yaml_find_node(mi->model_definition.doc, "spec/runtime/gateway")) { + log_debug("Select channels based on Model Instance (Gateway)"); + c_node = dse_yaml_find_node(mi->spec, "channels"); + } else { + c_node = dse_yaml_find_node(mi->model_definition.doc, "spec/channels"); + } + if (c_node) { + for (uint32_t i = 0; i < hashlist_length(&c_node->sequence); i++) { + YamlNode* ch_node = hashlist_at(&c_node->sequence, i); + int rc = _configure_model_channel(ch_node, mi); + if (rc) { + if (errno == 0) errno = EINVAL; + log_error("Model has no " MODEL_STEP_FUNC_NAME "() function"); + return -errno; + } + } + } + SignalVector* sv = model_sv_create(mi); + + /* Setup the initial ModelDesc object. */ + ModelDesc* model_desc = calloc(1, sizeof(ModelDesc)); + memcpy(&model_desc->vtable, model_vtable, sizeof(ModelVTable)); + model_desc->index = model_desc->vtable.index = __model_index__; + model_desc->sim = sim; + model_desc->mi = mi; + model_desc->sv = sv; + + /* Call create (if it exists). */ + if (model_desc->vtable.create) { + errno = 0; + ModelDesc* extended_model_desc = model_desc->vtable.create(model_desc); + if (errno) { + log_error("Error condition while calling " MODEL_CREATE_FUNC_NAME); + } + if (extended_model_desc) { + if (extended_model_desc != model_desc) { + /* The model returned an extended (new) ModelDesc object. */ + free(model_desc); + model_desc = extended_model_desc; + } + } + /* Reset some elements of the ModelDesc (don't trust the model). */ + model_desc->sim = sim; + model_desc->mi = mi; + model_desc->sv = sv; + } + + /* Finalise the ModelDesc object. */ + mi->model_desc = model_desc; + return 0; +} + + /** * modelc_get_model_instance * diff --git a/dse/modelc/controller/modelc_args.c b/dse/modelc/controller/modelc_args.c index 99a896f..3d8c1df 100644 --- a/dse/modelc/controller/modelc_args.c +++ b/dse/modelc/controller/modelc_args.c @@ -10,7 +10,7 @@ #include #include #include -#include +#include #include @@ -69,7 +69,7 @@ static void print_usage(const char* doc_string) } -/** +/* * _args_extract_environment * * Extract arguments from environment variables. diff --git a/dse/modelc/controller/modelc_debug.c b/dse/modelc/controller/modelc_debug.c index be76cda..d52610f 100644 --- a/dse/modelc/controller/modelc_debug.c +++ b/dse/modelc/controller/modelc_debug.c @@ -12,7 +12,7 @@ #include -/** +/* * The ModelC Debug API * ==================== */ diff --git a/dse/modelc/controller/step.c b/dse/modelc/controller/step.c index 5669fcb..a9e30ac 100644 --- a/dse/modelc/controller/step.c +++ b/dse/modelc/controller/step.c @@ -27,8 +27,9 @@ static int _do_step_func(void* _mf, void* _step_data) { ModelFunction* mf = _mf; mf_step_data* step_data = _step_data; + ModelDesc* md = step_data->mi->model_desc; double model_time = step_data->model_time; - int rc = mf->do_step_handler(&model_time, step_data->stop_time); + int rc = md->vtable.step(md, &model_time, step_data->stop_time); if (rc) log_error( "Model Function %s:%s (rc=%d)", step_data->mi->name, mf->name, rc); diff --git a/dse/modelc/examples/CMakeLists.txt b/dse/modelc/examples/CMakeLists.txt index 1101514..60b7564 100644 --- a/dse/modelc/examples/CMakeLists.txt +++ b/dse/modelc/examples/CMakeLists.txt @@ -2,11 +2,15 @@ # # SPDX-License-Identifier: Apache-2.0 -add_subdirectory(dynamic) + +# Example Models. +add_subdirectory(minimal) +add_subdirectory(extended) add_subdirectory(binary) +# TODO add_subdirectory(ncodec) + add_subdirectory(gateway) -add_subdirectory(stacked) -add_subdirectory(benchmark) +#add_subdirectory(benchmark) # Code examples for documentation. if(UNIX) diff --git a/dse/modelc/examples/apis/CMakeLists.txt b/dse/modelc/examples/apis/CMakeLists.txt index e2babc4..ee73a29 100644 --- a/dse/modelc/examples/apis/CMakeLists.txt +++ b/dse/modelc/examples/apis/CMakeLists.txt @@ -45,9 +45,10 @@ set(DSE_CLIB_INCLUDE_DIR "${DSE_CLIB_SOURCE_DIR}/..") # APIs # ---- add_library(apis OBJECT + model_create.c model_interface.c - model_signalvector.c - model_sv_create.c + signalvector_annotation.c + signalvector_interface.c schema_object_search.c simmock_configure.c simmock_frame_check.c diff --git a/dse/modelc/examples/apis/model_create.c b/dse/modelc/examples/apis/model_create.c new file mode 100644 index 0000000..715e53c --- /dev/null +++ b/dse/modelc/examples/apis/model_create.c @@ -0,0 +1,44 @@ +#include +#include +#include +#include + +typedef struct { + ModelDesc model; + /* Signal Pointers. */ + struct { + double* counter; + } signals; +} ExtendedModelDesc; + +static inline double* _index(ExtendedModelDesc* m, const char* v, const char* s) +{ + ModelSignalIndex idx = m->model.index((ModelDesc*)m, v, s); + if (idx.scalar == NULL) log_fatal("Signal not found (%s:%s)", v, s); + return idx.scalar; +} + +ModelDesc* model_create(ModelDesc* model) +{ + /* Extend the ModelDesc object (using a shallow copy). */ + ExtendedModelDesc* m = calloc(1, sizeof(ExtendedModelDesc)); + memcpy(m, model, sizeof(ModelDesc)); + + /* Index the signals that are used by this model. */ + m->signals.counter = _index(m, "data", "counter"); + + /* Set initial values. */ + *(m->signals.counter) = 42; + + /* Return the extended object. */ + return (ModelDesc*)m; +} + +int model_step(ModelDesc* model, double* model_time, double stop_time) +{ + ExtendedModelDesc* m = (ExtendedModelDesc*)model; + *(m->signals.counter) += 1; + + *model_time = stop_time; + return 0; +} diff --git a/dse/modelc/examples/apis/model_interface.c b/dse/modelc/examples/apis/model_interface.c index 9c4453e..b3570d8 100644 --- a/dse/modelc/examples/apis/model_interface.c +++ b/dse/modelc/examples/apis/model_interface.c @@ -1,50 +1,25 @@ +#include +#include #include -#include - #define UNUSED(x) ((void)x) - -/* Signal Vector definition. - Note: Signal order should match order in related SignalGroup (YAML). */ -typedef enum signal_name_index { - SIGNAL_FOO, - SIGNAL_BAR, - __SIGNAL__COUNT__ -} signal_name_index; - -static double* signal_value; - - -int model_step(double* model_time, double stop_time) +ModelDesc* model_create(ModelDesc* m) { - signal_value[SIGNAL_FOO] += 1.2; - signal_value[SIGNAL_BAR] += 4.2; - - *model_time = stop_time; - return 0; + return (ModelDesc*)m; } -int model_setup(ModelInstanceSpec* mi) +int model_step(ModelDesc* m, double* model_time, double stop_time) { - int rc = model_function_register(mi, "example", - 0.005, model_step); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = "model_channel", - .function_name = "example", - }; - rc = model_configure_channel(mi, &channel_desc); - if (rc != 0) return rc; - signal_value = channel_desc.vector_double; + ModelSignalIndex counter = m->index(m, "data", "counter"); + if (counter.scalar == NULL) return -EINVAL; + *(counter.scalar) += 1; + *model_time = stop_time; return 0; } -int model_exit(ModelInstanceSpec* mi) +void model_destroy(ModelDesc* m) { - UNUSED(mi); - return 0; + UNUSED(m); } diff --git a/dse/modelc/examples/apis/model_sv_create.c b/dse/modelc/examples/apis/model_sv_create.c deleted file mode 100644 index 49ac8be..0000000 --- a/dse/modelc/examples/apis/model_sv_create.c +++ /dev/null @@ -1,13 +0,0 @@ -#include -#include - -void print_signal_names(ModelInstanceSpec* mi) -{ - SignalVector* sv_save = model_sv_create(mi); - for (SignalVector* sv = sv_save; sv->name; sv++) { - for (uint32_t i = 0; i < sv->count; i++) { - log_debug(" signal : %s", sv->signal[i]); - } - } - model_sv_destroy(sv_save); -} diff --git a/dse/modelc/examples/apis/signalvector_annotation.c b/dse/modelc/examples/apis/signalvector_annotation.c new file mode 100644 index 0000000..d2709a6 --- /dev/null +++ b/dse/modelc/examples/apis/signalvector_annotation.c @@ -0,0 +1,15 @@ +#include +#include + +ModelDesc* model_create(ModelDesc* m) +{ + ModelSignalIndex idx = m->index(m, "data", "counter"); + + if (idx.scalar) { + /* Set initial value. */ + const char* v = idx.sv->annotation(idx.sv, idx.signal, "initial_value"); + if (v) *(idx.scalar) = atoi(v); + } + + return m; +} diff --git a/dse/modelc/examples/apis/model_signalvector.c b/dse/modelc/examples/apis/signalvector_interface.c similarity index 84% rename from dse/modelc/examples/apis/model_signalvector.c rename to dse/modelc/examples/apis/signalvector_interface.c index 5e95506..efcf2a4 100644 --- a/dse/modelc/examples/apis/model_signalvector.c +++ b/dse/modelc/examples/apis/signalvector_interface.c @@ -1,9 +1,12 @@ +#include #include #include -void print_signal_vectors(ModelInstanceSpec* mi) +#define UNUSED(x) ((void)x) + +int model_step(ModelDesc* m, double* model_time, double stop_time) { - SignalVector* sv = model_sv_create(mi); + SignalVector* sv = m->sv; while (sv && sv->name) { log_debug("Signal Vector : %s", sv->name); @@ -30,4 +33,7 @@ void print_signal_vectors(ModelInstanceSpec* mi) // Next signal vector. sv++; } -} \ No newline at end of file + + *model_time = stop_time; + return 0; +} diff --git a/dse/modelc/examples/apis/simmock_configure.c b/dse/modelc/examples/apis/simmock_configure.c index 3586390..fc0a31d 100644 --- a/dse/modelc/examples/apis/simmock_configure.c +++ b/dse/modelc/examples/apis/simmock_configure.c @@ -19,7 +19,7 @@ int test_setup(void** state) }; SimMock* mock = simmock_alloc(inst_names, ARRAY_SIZE(inst_names)); simmock_configure(mock, argv, ARRAY_SIZE(argv), ARRAY_SIZE(inst_names)); - simmock_load(mock, true); + simmock_load(mock); simmock_setup(mock, "signal", "network"); /* Return the mock. */ diff --git a/dse/modelc/examples/binary/CMakeLists.txt b/dse/modelc/examples/binary/CMakeLists.txt index 831f2fe..06482ec 100644 --- a/dse/modelc/examples/binary/CMakeLists.txt +++ b/dse/modelc/examples/binary/CMakeLists.txt @@ -1,79 +1,45 @@ -# Copyright 2023 Robert Bosch GmbH +# Copyright 2024 Robert Bosch GmbH # # SPDX-License-Identifier: Apache-2.0 cmake_minimum_required(VERSION 3.21) -project(BinaryModel - DESCRIPTION "ModelC - Example Binary Model." - HOMEPAGE_URL "${PROJECT_URL}" -) -set(PROJECT_VERSION ${VERSION}) - -include(FetchContent) - +project(Binary) set(MODEL_PATH "examples/binary") -set(CMAKE_SHARED_LIBRARY_PREFIX "") - -set(CMAKE_C_STANDARD 99) -set(CMAKE_C_STANDARD_REQUIRED TRUE) -set(CMAKE_POSITION_INDEPENDENT_CODE ON) -set(CMAKE_C_FLAGS_DEBUG "-g -ggdb") -set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -O3") -list(APPEND C_CXX_WARNING_FLAGS - -Wall - -W - -Wwrite-strings - -Wno-missing-field-initializers - -Wno-misleading-indentation -) -add_compile_options(${C_CXX_WARNING_FLAGS}) - -# External Project - DSE C Lib -# ---------------------------- +include(FetchContent) FetchContent_Declare(dse_clib URL $ENV{DSE_CLIB_URL} HTTP_USERNAME $ENV{GHE_USER} HTTP_PASSWORD $ENV{GHE_TOKEN} ) FetchContent_MakeAvailable(dse_clib) -set(DSE_CLIB_SOURCE_DIR ${dse_clib_SOURCE_DIR}/dse) -set(DSE_CLIB_SOURCE_FILES ) -set(DSE_CLIB_INCLUDE_DIR "${DSE_CLIB_SOURCE_DIR}/..") - - - -# Targets -# ======= +set(DSE_CLIB_INCLUDE_DIR ${dse_clib_SOURCE_DIR}) -# Target - Binary Model -# ---------------------- -add_library(binary_model SHARED - binary_model.c +add_library(binary SHARED + model.c ) -target_include_directories(binary_model +target_include_directories(binary PRIVATE ${DSE_CLIB_INCLUDE_DIR} ../../../.. ) -target_link_libraries(binary_model +target_link_libraries(binary PRIVATE $<$:${modelc_link_lib}> ) -install(TARGETS binary_model +install(TARGETS binary LIBRARY DESTINATION ${MODEL_PATH}/lib COMPONENT - binary_model + binary ) install( FILES model.yaml - stack.yaml - signal_group.yaml + simulation.yaml DESTINATION ${MODEL_PATH}/data COMPONENT - binary_model + binary ) diff --git a/dse/modelc/examples/binary/binary_model.c b/dse/modelc/examples/binary/binary_model.c deleted file mode 100644 index 54facb4..0000000 --- a/dse/modelc/examples/binary/binary_model.c +++ /dev/null @@ -1,116 +0,0 @@ -// Copyright 2023 Robert Bosch GmbH -// -// SPDX-License-Identifier: Apache-2.0 - -#include -#include -#include -#include -#include - - -#define ARRAY_LENGTH(array) (sizeof((array)) / sizeof((array)[0])) - - -/* Model Function definitions. A single Model Function is defined. */ -#define MODEL_FUNCTION_NAME "binary_example" -#define MODEL_FUNCTION_STEP_SIZE 0.005 -#define MODEL_FUNCTION_CHANNEL "binary_channel" -#define MODEL_FUNCTION_DO_STEP_FUNC do_step -#define MODEL_FUNCTION_DO_STEP_FUNC_STR "do_step" - - -/* Signals are defined in dynamic_model.yaml, order must match here! */ -typedef enum signal_name_index { - SIGNAL_RAW, - __SIGNAL__COUNT__ -} signal_name_index; - - -/* Signal Vector (binary). */ -static void** signal_value; -static uint32_t* signal_value_size; -static uint32_t* signal_value_buffer_size; - - -/* Model Function do_step() definition. Each Model Function has its own - do_step() and they are registered during model_setup(). */ -int MODEL_FUNCTION_DO_STEP_FUNC(double* model_time, double stop_time) -{ - log_simbus("Model function " MODEL_FUNCTION_DO_STEP_FUNC_STR " called"); - - /* Define some test data. */ - static int index = 0; - static const char* test_data[4] = { - "one", - "two", - "three", - "four", - }; - - /* Print the incoming content of the binary signal. */ - if (signal_value_size[SIGNAL_RAW]) { - /* The binary string may have embedded NULL characters, replace - them before printing the buffer. */ - char* buffer = (char*)signal_value[SIGNAL_RAW]; - for (uint32_t i = 0; i < signal_value_size[SIGNAL_RAW] - 1; i++) { - if (buffer[i] == 0) { - buffer[i] = ' '; - } - } - /* Now print the buffer*/ - log_notice("RECV: %s (buffer size=%u)", signal_value[SIGNAL_RAW], - signal_value_buffer_size[SIGNAL_RAW]); - } - - /* Indicate the binary object/signal was consumed (important!). */ - signal_value_size[SIGNAL_RAW] = 0; - - /* Write some test data to the binary signal. */ - if (index < 4) { - dse_buffer_append(&signal_value[SIGNAL_RAW], - &signal_value_size[SIGNAL_RAW], - &signal_value_buffer_size[SIGNAL_RAW], test_data[index], - strlen(test_data[index]) + 1); - index += 1; - } - - /* Advance the model time. */ - *model_time = stop_time; - - return 0; -} - - -/* Model Setup: a handle to this function will be dynamically loaded by the - Controller and it (this function) will be called to initialise the Model - and its Model Functions. */ -int MODEL_SETUP_FUNC(ModelInstanceSpec* model_instance) -{ - log_notice("Model function " MODEL_SETUP_FUNC_STR " called"); - assert(model_instance); - - int rc = model_function_register(model_instance, MODEL_FUNCTION_NAME, - MODEL_FUNCTION_STEP_SIZE, MODEL_FUNCTION_DO_STEP_FUNC); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = MODEL_FUNCTION_CHANNEL, - .function_name = MODEL_FUNCTION_NAME, - }; - rc = model_configure_channel(model_instance, &channel_desc); - if (rc != 0) return rc; - if (channel_desc.vector_binary == NULL) { - log_fatal("Binary vector not allocated!"); - } - assert(channel_desc.signal_count == __SIGNAL__COUNT__); - assert(channel_desc.vector_binary); - assert(channel_desc.vector_binary_size); - assert(channel_desc.vector_binary_buffer_size); - signal_value = channel_desc.vector_binary; - signal_value_size = channel_desc.vector_binary_size; - signal_value_buffer_size = channel_desc.vector_binary_buffer_size; - - return 0; -} diff --git a/dse/modelc/examples/binary/model.c b/dse/modelc/examples/binary/model.c new file mode 100644 index 0000000..876b65f --- /dev/null +++ b/dse/modelc/examples/binary/model.c @@ -0,0 +1,106 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 + +#include +#include +#include +#include +#include + + +typedef struct { + SignalVector* sv; + uint32_t index; + char* buffer; + uint32_t buffer_size; +} BinarySignalDesc; + +typedef struct { + ModelDesc model; + /* Scalar Signal Pointers. */ + struct { + double* counter; + } scalars; + /* Binary Signal Indexes. */ + struct { + BinarySignalDesc message; + } binary; +} ExtendedModelDesc; + + +static inline double* _index_scalar( + ExtendedModelDesc* m, const char* v, const char* s) +{ + ModelSignalIndex idx = m->model.index((ModelDesc*)m, v, s); + if (idx.scalar == NULL) log_fatal("Signal not found (%s:%s)", v, s); + return idx.scalar; +} + + +static inline BinarySignalDesc _index_binary( + ExtendedModelDesc* m, const char* v, const char* s) +{ + ModelSignalIndex idx = m->model.index((ModelDesc*)m, v, s); + if (idx.binary == NULL) log_fatal("Signal not found (%s:%s)", v, s); + + BinarySignalDesc ret = { + .sv = &(m->model.sv[idx.vector]), + .index = idx.signal, + }; + return ret; +} + + +ModelDesc* model_create(ModelDesc* model) +{ + /* Extend the ModelDesc object (using a shallow copy). */ + ExtendedModelDesc* m = calloc(1, sizeof(ExtendedModelDesc)); + memcpy(m, model, sizeof(ModelDesc)); + + /* Index the signals that are used by this model. */ + m->scalars.counter = _index_scalar(m, "scalar", "counter"); + m->binary.message = _index_binary(m, "binary", "message"); + + /* Set initial values. */ + *(m->scalars.counter) = 42; + m->binary.message.buffer = calloc(10, sizeof(char)); + m->binary.message.buffer_size = 10; + + /* Return the extended object. */ + return (ModelDesc*)m; +} + + +static inline int _format_message(BinarySignalDesc* b, int v) +{ + return snprintf(b->buffer, b->buffer_size, "count is %d", v); +} + +int model_step(ModelDesc* model, double* model_time, double stop_time) +{ + ExtendedModelDesc* m = (ExtendedModelDesc*)model; + + /* Scalar signals. */ + *(m->scalars.counter) += 1; + /* Binary signals. */ + int len = _format_message(&(m->binary.message), (int)*(m->scalars.counter)); + if (len >= (int)(m->binary.message.buffer_size - 1)) { + m->binary.message.buffer = realloc(m->binary.message.buffer, len + 1); + m->binary.message.buffer_size = len + 1; + _format_message(&(m->binary.message), (int)*(m->scalars.counter)); + } + m->binary.message.sv->reset(m->binary.message.sv, m->binary.message.index); + m->binary.message.sv->append(m->binary.message.sv, m->binary.message.index, + m->binary.message.buffer, strlen(m->binary.message.buffer) + 1); + + *model_time = stop_time; + return 0; +} + + +void model_destroy(ModelDesc* model) +{ + ExtendedModelDesc* m = (ExtendedModelDesc*)model; + if (m->binary.message.buffer) free(m->binary.message.buffer); +} diff --git a/dse/modelc/examples/binary/model.yaml b/dse/modelc/examples/binary/model.yaml index 71b9828..a09a572 100644 --- a/dse/modelc/examples/binary/model.yaml +++ b/dse/modelc/examples/binary/model.yaml @@ -1,21 +1,30 @@ -# Copyright 2023 Robert Bosch GmbH +# Copyright 2024 Robert Bosch GmbH # # SPDX-License-Identifier: Apache-2.0 --- kind: Model metadata: - name: BinaryModel + name: Binary spec: runtime: dynlib: - os: linux arch: amd64 - path: lib/binary_model.so + path: lib/libbinary.so - os: linux arch: x86 - path: lib/binary_model.so + path: lib/libbinary.so + - os: windows + arch: x64 + path: bin/binary.dll + - os: windows + arch: x86 + path: bin/binary.dll channels: - - alias: binary_channel + - alias: scalar + selectors: + channel: scalar_channel + - alias: binary selectors: - channel: network + channel: binary_channel diff --git a/dse/modelc/examples/binary/signal_group.yaml b/dse/modelc/examples/binary/signal_group.yaml deleted file mode 100644 index 0be06f1..0000000 --- a/dse/modelc/examples/binary/signal_group.yaml +++ /dev/null @@ -1,19 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: SignalGroup -metadata: - name: network_signals - labels: - channel: network - model: BinaryModel - annotations: - vector_type: binary - vector_name: network_vector -spec: - signals: - - signal: RAW - annotations: - mime_type: 'application/octet-stream' diff --git a/dse/modelc/examples/binary/simulation.yaml b/dse/modelc/examples/binary/simulation.yaml new file mode 100644 index 0000000..361b4ae --- /dev/null +++ b/dse/modelc/examples/binary/simulation.yaml @@ -0,0 +1,58 @@ +# Copyright 2024 Robert Bosch GmbH +# +# SPDX-License-Identifier: Apache-2.0 + +--- +kind: Stack +metadata: + name: binary_stack +spec: + connection: + transport: + redispubsub: + uri: redis://localhost:6379 + timeout: 60 + models: + - name: simbus + model: + name: simbus + channels: + - name: scalar_channel + expectedModelCount: 1 + - name: binary_channel + expectedModelCount: 1 + - name: binary_inst + uid: 42 + model: + name: Binary + channels: + - name: scalar_channel + alias: scalar + - name: binary_channel + alias: binary +--- +kind: Model +metadata: + name: simbus +--- +kind: SignalGroup +metadata: + name: scalar_channel + labels: + channel: scalar_channel +spec: + signals: + - signal: counter +--- +kind: SignalGroup +metadata: + name: binary_channel + labels: + channel: binary_channel + annotations: + vector_type: binary +spec: + signals: + - signal: message + annotations: + mime_type: 'application/octet-stream' diff --git a/dse/modelc/examples/binary/stack.yaml b/dse/modelc/examples/binary/stack.yaml deleted file mode 100644 index a3efc06..0000000 --- a/dse/modelc/examples/binary/stack.yaml +++ /dev/null @@ -1,39 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: Stack -metadata: - name: binary_model_stack -spec: - connection: - transport: - redispubsub: - uri: redis://localhost:6379 - timeout: 60 - models: - - name: simbus - model: - name: simbus - channels: - - name: network - expectedModelCount: 2 - - name: binary_model_instance - uid: 42 - model: - name: BinaryModel - channels: - - name: network - alias: binary_channel - - name: second_binary_model_instance - uid: 24 - model: - name: BinaryModel - channels: - - name: network - alias: binary_channel ---- -kind: Model -metadata: - name: simbus diff --git a/dse/modelc/examples/dynamic/CMakeLists.txt b/dse/modelc/examples/dynamic/CMakeLists.txt deleted file mode 100644 index 424fae3..0000000 --- a/dse/modelc/examples/dynamic/CMakeLists.txt +++ /dev/null @@ -1,79 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - -cmake_minimum_required(VERSION 3.21) - -project(DynamicModel - DESCRIPTION "ModelC - Example Dynamic Model." - HOMEPAGE_URL "${PROJECT_URL}" -) -set(PROJECT_VERSION ${VERSION}) - -include(FetchContent) - -set(MODEL_PATH "examples/dynamic") -set(CMAKE_SHARED_LIBRARY_PREFIX "") - -set(CMAKE_C_STANDARD 99) -set(CMAKE_C_STANDARD_REQUIRED TRUE) -set(CMAKE_POSITION_INDEPENDENT_CODE ON) -set(CMAKE_C_FLAGS_DEBUG "-g -ggdb") -set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -O3") -list(APPEND C_CXX_WARNING_FLAGS - -Wall - -W - -Wwrite-strings - -Wno-missing-field-initializers - -Wno-misleading-indentation -) -add_compile_options(${C_CXX_WARNING_FLAGS}) - - -# External Project - DSE C Lib -# ---------------------------- -FetchContent_Declare(dse_clib - URL $ENV{DSE_CLIB_URL} - HTTP_USERNAME $ENV{GHE_USER} - HTTP_PASSWORD $ENV{GHE_TOKEN} -) -FetchContent_MakeAvailable(dse_clib) -set(DSE_CLIB_SOURCE_DIR ${dse_clib_SOURCE_DIR}/dse) -set(DSE_CLIB_SOURCE_FILES ) -set(DSE_CLIB_INCLUDE_DIR "${DSE_CLIB_SOURCE_DIR}/..") - - - -# Targets -# ======= - -# Target - Dynamic Model -# ---------------------- -add_library(dynamic_model SHARED - dynamic_model.c -) -target_include_directories(dynamic_model - PRIVATE - ${DSE_CLIB_INCLUDE_DIR} - ../../../.. -) -target_link_libraries(dynamic_model - PRIVATE - $<$:${modelc_link_lib}> -) -install(TARGETS dynamic_model - LIBRARY DESTINATION - ${MODEL_PATH}/lib - COMPONENT - dynamic_model -) -install( - FILES - model.yaml - stack.yaml - signal_group.yaml - DESTINATION - ${MODEL_PATH}/data - COMPONENT - dynamic_model -) diff --git a/dse/modelc/examples/dynamic/README.md b/dse/modelc/examples/dynamic/README.md deleted file mode 100644 index be350ce..0000000 --- a/dse/modelc/examples/dynamic/README.md +++ /dev/null @@ -1,36 +0,0 @@ - - -### Run the Example - -##### Terminal 1: Redis - -```bash -$ docker run --rm -it --name redis -p 6379:6379 redis & -``` - -##### Terminal 2: Standalone SimBus - -```bash -$ export SIMBUS_EXE=$(pwd)/libmodelcapi/build/_out/bin/simbus -$ export MODEL_SANDBOX_DIR=$(pwd)/examples/dynamic_model/build/_out -$ export YAML_DIR=${MODEL_SANDBOX_DIR}/data/yaml/dynamic_model - -$ cd ${MODEL_SANDBOX_DIR} -$ ${SIMBUS_EXE} --name simbus ${YAML_DIR}/dynamic_model.yaml -``` - -##### Terminal 3: Dynamically Linked Model - -```bash -$ export MODELC_EXE=$(pwd)/libmodelcapi/build/_out/bin/modelc -$ export MODEL_SANDBOX_DIR=$(pwd)/examples/dynamic_model/build/_out -$ export YAML_DIR=${MODEL_SANDBOX_DIR}/data/yaml/dynamic_model - -$ cd ${MODEL_SANDBOX_DIR}; \ - ${MODELC_EXE} --name dynamic_model_instance ${YAML_DIR}/dynamic_model.yaml; \ - cd - -``` diff --git a/dse/modelc/examples/dynamic/dynamic_model.c b/dse/modelc/examples/dynamic/dynamic_model.c deleted file mode 100644 index 70c45f3..0000000 --- a/dse/modelc/examples/dynamic/dynamic_model.c +++ /dev/null @@ -1,86 +0,0 @@ -// Copyright 2023 Robert Bosch GmbH -// -// SPDX-License-Identifier: Apache-2.0 - -#include -#include -#include -#include - - -#define ARRAY_LENGTH(array) (sizeof((array)) / sizeof((array)[0])) - - -/* Model Function definitions. */ -#define MODEL_FUNCTION_NAME "example" -#define MODEL_FUNCTION_DO_STEP do_step -#define MODEL_FUNCTION_STEP_SIZE 0.005 -#define MODEL_FUNCTION_CHANNEL "model_channel" - - -/* Signals are defined in dynamic_model.yaml, order must match here! */ -typedef enum signal_name_index { - SIGNAL_FOO, - SIGNAL_BAR, - __SIGNAL__COUNT__ -} signal_name_index; - - -/* Signal Vector definition (storage allocated during model_setup()). */ -static double* signal_value; - - -/* Model Function do_step() definition. Each Model Function has its own - do_step() and they are registered during model_setup(). */ -int MODEL_FUNCTION_DO_STEP(double* model_time, double stop_time) -{ - assert(signal_value); - - signal_value[SIGNAL_FOO] += 1.2; - signal_value[SIGNAL_BAR] += 4.2; - *model_time = stop_time; - - return 0; -} - - -/* Model Setup: a handle to this function will be dynamically loaded by the - Controller and it (this function) will be called to initialise the Model - and its Model Functions. */ -int MODEL_SETUP_FUNC(ModelInstanceSpec* model_instance) -{ - log_notice("Model function " MODEL_SETUP_FUNC_STR " called"); - assert(model_instance); - - int rc = model_function_register(model_instance, MODEL_FUNCTION_NAME, - MODEL_FUNCTION_STEP_SIZE, MODEL_FUNCTION_DO_STEP); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = MODEL_FUNCTION_CHANNEL, - .function_name = MODEL_FUNCTION_NAME, - }; - rc = model_configure_channel(model_instance, &channel_desc); - if (rc != 0) return rc; - assert(channel_desc.signal_count == __SIGNAL__COUNT__); - assert(channel_desc.vector_double); - signal_value = channel_desc.vector_double; - - return 0; -} - - -/* Model Exit: OPTIONAL! a handle to this function, if present, will be - dynamically loaded by the Controller and called before the Controller exits - the simulation (and sends the ModelExit message to the SimBus). */ -int MODEL_EXIT_FUNC(ModelInstanceSpec* model_instance) -{ - log_notice("Model function " MODEL_EXIT_FUNC_STR " called"); - if (model_instance) { - /* The parameter model_instance may, in some circumstances, not - be set. Therefore check before using. */ - } - - return 0; -} diff --git a/dse/modelc/examples/dynamic/signal_group.yaml b/dse/modelc/examples/dynamic/signal_group.yaml deleted file mode 100644 index 35525c3..0000000 --- a/dse/modelc/examples/dynamic/signal_group.yaml +++ /dev/null @@ -1,15 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: SignalGroup -metadata: - name: test_signals - labels: - channel: test - model: dynamic_model -spec: - signals: - - signal: foo - - signal: bar diff --git a/dse/modelc/examples/extended/CMakeLists.txt b/dse/modelc/examples/extended/CMakeLists.txt new file mode 100644 index 0000000..17cd3e2 --- /dev/null +++ b/dse/modelc/examples/extended/CMakeLists.txt @@ -0,0 +1,45 @@ +# Copyright 2024 Robert Bosch GmbH +# +# SPDX-License-Identifier: Apache-2.0 + +cmake_minimum_required(VERSION 3.21) + +project(Extended) +set(MODEL_PATH "examples/extended") + +include(FetchContent) +FetchContent_Declare(dse_clib + URL $ENV{DSE_CLIB_URL} + HTTP_USERNAME $ENV{GHE_USER} + HTTP_PASSWORD $ENV{GHE_TOKEN} +) +FetchContent_MakeAvailable(dse_clib) +set(DSE_CLIB_INCLUDE_DIR ${dse_clib_SOURCE_DIR}) + +add_library(extended SHARED + model.c +) +target_include_directories(extended + PRIVATE + ${DSE_CLIB_INCLUDE_DIR} + ../../../.. +) +target_link_libraries(extended + PRIVATE + $<$:${modelc_link_lib}> +) +install(TARGETS extended + LIBRARY DESTINATION + ${MODEL_PATH}/lib + COMPONENT + extended +) +install( + FILES + model.yaml + simulation.yaml + DESTINATION + ${MODEL_PATH}/data + COMPONENT + extended +) diff --git a/dse/modelc/examples/extended/model.c b/dse/modelc/examples/extended/model.c new file mode 100644 index 0000000..ebe14f0 --- /dev/null +++ b/dse/modelc/examples/extended/model.c @@ -0,0 +1,61 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 + +#include +#include +#include +#include +#include + + +typedef struct { + ModelDesc model; + /* Signal Pointers. */ + struct { + double* counter; + double* odd; + double* even; + } signals; +} ExtendedModelDesc; + + +static inline double* _index(ExtendedModelDesc* m, const char* v, const char* s) +{ + ModelSignalIndex idx = m->model.index((ModelDesc*)m, v, s); + if (idx.scalar == NULL) log_fatal("Signal not found (%s:%s)", v, s); + return idx.scalar; +} + + +ModelDesc* model_create(ModelDesc* model) +{ + /* Extend the ModelDesc object (using a shallow copy). */ + ExtendedModelDesc* m = calloc(1, sizeof(ExtendedModelDesc)); + memcpy(m, model, sizeof(ModelDesc)); + + /* Index the signals that are used by this model. */ + m->signals.counter = _index(m, "data", "counter"); + m->signals.odd = _index(m, "data", "odd"); + m->signals.even = _index(m, "data", "even"); + + /* Set initial values. */ + *(m->signals.counter) = 42; + *(m->signals.odd) = false; + *(m->signals.even) = true; + + /* Return the extended object. */ + return (ModelDesc*)m; +} + + +int model_step(ModelDesc* model, double* model_time, double stop_time) +{ + ExtendedModelDesc* m = (ExtendedModelDesc*)model; + *(m->signals.counter) += 1; + *(m->signals.odd) = (int)(*(m->signals.counter)) % 2 ? true : false; + *(m->signals.even) = (int)(*(m->signals.counter)) % 2 ? false : true; + + *model_time = stop_time; + return 0; +} diff --git a/dse/modelc/examples/dynamic/model.yaml b/dse/modelc/examples/extended/model.yaml similarity index 52% rename from dse/modelc/examples/dynamic/model.yaml rename to dse/modelc/examples/extended/model.yaml index b3125c3..8cd8caf 100644 --- a/dse/modelc/examples/dynamic/model.yaml +++ b/dse/modelc/examples/extended/model.yaml @@ -1,27 +1,27 @@ -# Copyright 2023 Robert Bosch GmbH +# Copyright 2024 Robert Bosch GmbH # # SPDX-License-Identifier: Apache-2.0 --- kind: Model metadata: - name: DynamicModel + name: Extended spec: runtime: dynlib: - os: linux arch: amd64 - path: lib/dynamic_model.so + path: lib/libextended.so - os: linux arch: x86 - path: lib/dynamic_model.so + path: lib/libextended.so - os: windows arch: x64 - path: bin/dynamic_model.dll + path: bin/extended.dll - os: windows arch: x86 - path: bin/dynamic_model.dll + path: bin/extended.dll channels: - - alias: model_channel + - alias: data selectors: - channel: test + side: data diff --git a/dse/modelc/examples/extended/simulation.yaml b/dse/modelc/examples/extended/simulation.yaml new file mode 100644 index 0000000..f96a5a2 --- /dev/null +++ b/dse/modelc/examples/extended/simulation.yaml @@ -0,0 +1,43 @@ +# Copyright 2024 Robert Bosch GmbH +# +# SPDX-License-Identifier: Apache-2.0 + +--- +kind: Stack +metadata: + name: extended_stack +spec: + connection: + transport: + redispubsub: + uri: redis://localhost:6379 + timeout: 60 + models: + - name: simbus + model: + name: simbus + channels: + - name: data_channel + expectedModelCount: 1 + - name: extended_inst + uid: 42 + model: + name: Extended + channels: + - name: data_channel + alias: data +--- +kind: Model +metadata: + name: simbus +--- +kind: SignalGroup +metadata: + name: data + labels: + side: data +spec: + signals: + - signal: counter + - signal: odd + - signal: even diff --git a/dse/modelc/examples/gateway/gateway.c b/dse/modelc/examples/gateway/gateway.c index 61b6e16..f081151 100644 --- a/dse/modelc/examples/gateway/gateway.c +++ b/dse/modelc/examples/gateway/gateway.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 #include +#include #include #include #include diff --git a/dse/modelc/examples/minimal/CMakeLists.txt b/dse/modelc/examples/minimal/CMakeLists.txt new file mode 100644 index 0000000..e07d159 --- /dev/null +++ b/dse/modelc/examples/minimal/CMakeLists.txt @@ -0,0 +1,35 @@ +# Copyright 2024 Robert Bosch GmbH +# +# SPDX-License-Identifier: Apache-2.0 + +cmake_minimum_required(VERSION 3.21) + +project(Minimal) +set(MODEL_PATH "examples/minimal") + +add_library(minimal SHARED + model.c +) +target_include_directories(minimal + PRIVATE + ../../../.. +) +target_link_libraries(minimal + PRIVATE + $<$:${modelc_link_lib}> +) +install(TARGETS minimal + LIBRARY DESTINATION + ${MODEL_PATH}/lib + COMPONENT + minimal +) +install( + FILES + model.yaml + simulation.yaml + DESTINATION + ${MODEL_PATH}/data + COMPONENT + minimal +) diff --git a/dse/modelc/examples/minimal/model.c b/dse/modelc/examples/minimal/model.c new file mode 100644 index 0000000..4ce7914 --- /dev/null +++ b/dse/modelc/examples/minimal/model.c @@ -0,0 +1,16 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 + +#include +#include +#include + +int model_step(ModelDesc* m, double* model_time, double stop_time) +{ + ModelSignalIndex counter = m->index(m, "data", "counter"); + if (counter.scalar == NULL) return -EINVAL; + *(counter.scalar) += 1; + *model_time = stop_time; + return 0; +} diff --git a/dse/modelc/examples/minimal/model.yaml b/dse/modelc/examples/minimal/model.yaml new file mode 100644 index 0000000..1718cfe --- /dev/null +++ b/dse/modelc/examples/minimal/model.yaml @@ -0,0 +1,27 @@ +# Copyright 2024 Robert Bosch GmbH +# +# SPDX-License-Identifier: Apache-2.0 + +--- +kind: Model +metadata: + name: Minimal +spec: + runtime: + dynlib: + - os: linux + arch: amd64 + path: lib/libminimal.so + - os: linux + arch: x86 + path: lib/libminimal.so + - os: windows + arch: x64 + path: bin/minimal.dll + - os: windows + arch: x86 + path: bin/minimal.dll + channels: + - alias: data + selectors: + side: data diff --git a/dse/modelc/examples/dynamic/stack.yaml b/dse/modelc/examples/minimal/simulation.yaml similarity index 55% rename from dse/modelc/examples/dynamic/stack.yaml rename to dse/modelc/examples/minimal/simulation.yaml index 9aa5a28..deb856c 100644 --- a/dse/modelc/examples/dynamic/stack.yaml +++ b/dse/modelc/examples/minimal/simulation.yaml @@ -1,11 +1,11 @@ -# Copyright 2023 Robert Bosch GmbH +# Copyright 2024 Robert Bosch GmbH # # SPDX-License-Identifier: Apache-2.0 --- kind: Stack metadata: - name: dynamic_model_stack + name: minimal_stack spec: connection: transport: @@ -17,16 +17,25 @@ spec: model: name: simbus channels: - - name: test + - name: data_channel expectedModelCount: 1 - - name: dynamic_model_instance + - name: minimal_inst uid: 42 model: - name: DynamicModel + name: Minimal channels: - - name: test - alias: model_channel + - name: data_channel + alias: data --- kind: Model metadata: name: simbus +--- +kind: SignalGroup +metadata: + name: data + labels: + side: data +spec: + signals: + - signal: counter diff --git a/dse/modelc/examples/stacked/CMakeLists.txt b/dse/modelc/examples/stacked/CMakeLists.txt deleted file mode 100644 index 2337df9..0000000 --- a/dse/modelc/examples/stacked/CMakeLists.txt +++ /dev/null @@ -1,102 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - -cmake_minimum_required(VERSION 3.21) - -project(StackedModel - DESCRIPTION "ModelC - Example Stacked Model." - HOMEPAGE_URL "${PROJECT_URL}" -) -set(PROJECT_VERSION ${VERSION}) - -include(FetchContent) - -set(MODEL_PATH "examples/stacked") -set(CMAKE_SHARED_LIBRARY_PREFIX "") - -set(CMAKE_C_STANDARD 99) -set(CMAKE_C_STANDARD_REQUIRED TRUE) -set(CMAKE_POSITION_INDEPENDENT_CODE ON) -set(CMAKE_C_FLAGS_DEBUG "-g -ggdb") -set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -O3") -list(APPEND C_CXX_WARNING_FLAGS - -Wall - -W - -Wwrite-strings - -Wno-missing-field-initializers - -Wno-misleading-indentation -) -add_compile_options(${C_CXX_WARNING_FLAGS}) - - -# External Project - DSE C Lib -# ---------------------------- -FetchContent_Declare(dse_clib - URL $ENV{DSE_CLIB_URL} - HTTP_USERNAME $ENV{GHE_USER} - HTTP_PASSWORD $ENV{GHE_TOKEN} -) -FetchContent_MakeAvailable(dse_clib) -set(DSE_CLIB_SOURCE_DIR ${dse_clib_SOURCE_DIR}/dse) -set(DSE_CLIB_SOURCE_FILES ) -set(DSE_CLIB_INCLUDE_DIR "${DSE_CLIB_SOURCE_DIR}/..") - - - -# Targets -# ======= - -# Target - stacked_one -# ---------------------- -add_library(stacked_one SHARED - stacked_model.c -) -target_include_directories(stacked_one - PRIVATE - ${DSE_CLIB_INCLUDE_DIR} - ../../../.. -) -target_link_libraries(stacked_one - PRIVATE - $<$:${modelc_link_lib}> -) -install(TARGETS stacked_one - LIBRARY DESTINATION - ${MODEL_PATH}/lib - COMPONENT - stacked_one -) - - -# Target - stacked_two -# ---------------------- -add_library(stacked_two SHARED - stacked_model.c -) -target_include_directories(stacked_two - PRIVATE - ${DSE_CLIB_INCLUDE_DIR} - ../../../.. -) -target_link_libraries(stacked_two - PRIVATE - $<$:${modelc_link_lib}> -) -install(TARGETS stacked_two - LIBRARY DESTINATION - ${MODEL_PATH}/lib - COMPONENT - stacked_two -) - - - -install( - FILES - model.yaml - stack.yaml - signal_group.yaml - DESTINATION - ${MODEL_PATH}/data -) diff --git a/dse/modelc/examples/stacked/README.md b/dse/modelc/examples/stacked/README.md deleted file mode 100644 index 78373f1..0000000 --- a/dse/modelc/examples/stacked/README.md +++ /dev/null @@ -1,7 +0,0 @@ - - -> Note: This model is used for testing stacked models in a single ModelC instance. diff --git a/dse/modelc/examples/stacked/model.yaml b/dse/modelc/examples/stacked/model.yaml deleted file mode 100644 index d4597a5..0000000 --- a/dse/modelc/examples/stacked/model.yaml +++ /dev/null @@ -1,50 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: Model -metadata: - name: StackedModelOne -spec: - runtime: - dynlib: - - os: linux - arch: amd64 - path: lib/stacked_one.so - - os: linux - arch: x86 - path: lib/stacked_one.so - - os: windows - arch: x64 - path: bin/stacked_one.dll - - os: windows - arch: x86 - path: bin/stacked_one.dll - channels: - - alias: model_channel - selectors: - channel: test ---- -kind: Model -metadata: - name: StackedModelTwo -spec: - runtime: - dynlib: - - os: linux - arch: amd64 - path: lib/stacked_two.so - - os: linux - arch: x86 - path: lib/stacked_two.so - - os: windows - arch: x64 - path: bin/stacked_two.dll - - os: windows - arch: x86 - path: bin/stacked_two.dll - channels: - - alias: model_channel - selectors: - channel: test diff --git a/dse/modelc/examples/stacked/signal_group.yaml b/dse/modelc/examples/stacked/signal_group.yaml deleted file mode 100644 index 7ae9e49..0000000 --- a/dse/modelc/examples/stacked/signal_group.yaml +++ /dev/null @@ -1,15 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: SignalGroup -metadata: - name: test_signals - labels: - channel: test - model: stacked_model -spec: - signals: - - signal: foo - - signal: bar diff --git a/dse/modelc/examples/stacked/stack.yaml b/dse/modelc/examples/stacked/stack.yaml deleted file mode 100644 index 34b8c85..0000000 --- a/dse/modelc/examples/stacked/stack.yaml +++ /dev/null @@ -1,39 +0,0 @@ -# Copyright 2023 Robert Bosch GmbH -# -# SPDX-License-Identifier: Apache-2.0 - ---- -kind: Stack -metadata: - name: stacked_model -spec: - connection: - transport: - redispubsub: - uri: redis://localhost:6379 - timeout: 60 - models: - - name: simbus - model: - name: simbus - channels: - - name: test - expectedModelCount: 2 - - name: stacked_one - uid: 43 - model: - name: StackedModelOne - channels: - - name: test - alias: model_channel - - name: stacked_two - uid: 44 - model: - name: StackedModelTwo - channels: - - name: test - alias: model_channel ---- -kind: Model -metadata: - name: simbus diff --git a/dse/modelc/examples/stacked/stacked_model.c b/dse/modelc/examples/stacked/stacked_model.c deleted file mode 100644 index 05ea63c..0000000 --- a/dse/modelc/examples/stacked/stacked_model.c +++ /dev/null @@ -1,69 +0,0 @@ -// Copyright 2023 Robert Bosch GmbH -// -// SPDX-License-Identifier: Apache-2.0 - -#include -#include -#include -#include - - -/* Model Function definitions. */ -#define MODEL_FUNCTION_NAME "example" -#define MODEL_FUNCTION_DO_STEP do_step -#define MODEL_FUNCTION_STEP_SIZE 0.005 -#define MODEL_FUNCTION_CHANNEL "model_channel" - - -/* Signals are defined in signal_group.yaml, order must match here! */ -typedef enum signal_name_index { - SIGNAL_FOO, - SIGNAL_BAR, - __SIGNAL__COUNT__ -} signal_name_index; - -static double* signal_value; - - -int MODEL_FUNCTION_DO_STEP(double* model_time, double stop_time) -{ - assert(signal_value); - - signal_value[SIGNAL_FOO] += 1.2; - signal_value[SIGNAL_BAR] += 4.2; - *model_time = stop_time; - - return 0; -} - - -int MODEL_SETUP_FUNC(ModelInstanceSpec* model_instance) -{ - log_notice("Model function " MODEL_SETUP_FUNC_STR " called"); - assert(model_instance); - - int rc = model_function_register(model_instance, MODEL_FUNCTION_NAME, - MODEL_FUNCTION_STEP_SIZE, MODEL_FUNCTION_DO_STEP); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = MODEL_FUNCTION_CHANNEL, - .function_name = MODEL_FUNCTION_NAME, - }; - rc = model_configure_channel(model_instance, &channel_desc); - if (rc != 0) return rc; - assert(channel_desc.signal_count == __SIGNAL__COUNT__); - assert(channel_desc.vector_double); - signal_value = channel_desc.vector_double; - - return 0; -} - - -int MODEL_EXIT_FUNC(ModelInstanceSpec* model_instance) -{ - assert(model_instance); - log_notice("Model function " MODEL_EXIT_FUNC_STR " called"); - return 0; -} diff --git a/dse/modelc/mcl.h b/dse/modelc/mcl.h index 30419bd..3755d46 100644 --- a/dse/modelc/mcl.h +++ b/dse/modelc/mcl.h @@ -5,9 +5,10 @@ #ifndef DSE_MODELC_MCL_H_ #define DSE_MODELC_MCL_H_ -#include #include +#include #include +#include /** @@ -84,7 +85,7 @@ typedef struct MclModelDesc MclModelDesc; /* MCL Strategy methods (generic model provided). */ -typedef int (*MclExecuteMethod)(MclStrategyAction action); +typedef int (*MclExecuteMethod)(ModelDesc* model, MclStrategyAction action); /* MCL Strategy handler functions. */ @@ -154,14 +155,11 @@ typedef struct MclModelDesc { typedef struct MclInstanceDesc { ModelInstanceSpec* model_instance; /* Instance Properties. */ - ModelChannelDesc* channel; + SignalVector* mcl_channel_sv; MclStrategyDesc* strategy; HashList models; /* MclModelDesc */ } MclInstanceDesc; -/* This portion of the API is implemented by an MCL. */ -DLL_PUBLIC int MCL_SETUP_FUNC(void); - /* mcl.c - Model Compatibility Library (MCL) interface.*/ DLL_PUBLIC void mcl_register_strategy(MclStrategyDesc* strategy); diff --git a/dse/modelc/model.h b/dse/modelc/model.h index 8f1a641..37ec038 100644 --- a/dse/modelc/model.h +++ b/dse/modelc/model.h @@ -1,4 +1,4 @@ -// Copyright 2023 Robert Bosch GmbH +// Copyright 2024 Robert Bosch GmbH // // SPDX-License-Identifier: Apache-2.0 @@ -7,111 +7,79 @@ #include #include -#include -#include -#include #ifndef DLL_PUBLIC -#define DLL_PUBLIC __attribute__((visibility("default"))) +#if defined _WIN32 || defined __CYGWIN__ +#ifdef DLL_BUILD +#define DLL_PUBLIC __declspec(dllexport) +#else +#define DLL_PUBLIC __declspec(dllimport) +#endif +#else +#define DLL_PUBLIC __attribute__((visibility("default"))) +#endif #endif #ifndef DLL_PRIVATE +#if defined _WIN32 || defined __CYGWIN__ +#define DLL_PRIVATE +#else #define DLL_PRIVATE __attribute__((visibility("hidden"))) #endif +#endif + + +#define __MODELC_ERROR_OFFSET (2000) +#define MODEL_DEFAULT_STEP_SIZE 0.0005 +#define MODEL_CREATE_FUNC_NAME "model_create" +#define MODEL_STEP_FUNC_NAME "model_step" +#define MODEL_DESTROY_FUNC_NAME "model_destroy" + + +typedef struct SimulationSpec SimulationSpec; +typedef struct ModelInstanceSpec ModelInstanceSpec; +typedef struct SignalVector SignalVector; +typedef struct ModelDesc ModelDesc; +typedef struct ModelSignalIndex ModelSignalIndex; /** Model API ========= -The Model API allows model developers and integrators to interface with a -Dynamic Simulation Environment via a connection with a Simulation Bus. -*/ +The Model API allows model developers and integrators to implement models which +can be connected to a Simulation Bus. +Models are able to exchange signals with other models via this connection to +a Simulation Bus. +A runtime environment, such as the ModelC Runtime/Importer, will load the +model and also manages the connection with the Simulation Bus. -#define __MODELC_ERROR_OFFSET (2000) - -#define MODEL_SETUP_FUNC model_setup -#define MODEL_SETUP_FUNC_STR "model_setup" - -#define MODEL_EXIT_FUNC model_exit -#define MODEL_EXIT_FUNC_STR "model_exit" - - -typedef struct ModelDefinitionSpec { - const char* name; - /* Path to the Model Package (i.e. where model.yaml is found). */ - const char* path; - /* Path to the Model Lib, relative to path (above). */ - const char* file; - /* Combined path and file for the Model Lib. */ - char* full_path; - /* Reference to parsed YAML Documents. */ - YamlNode* doc; - YamlNode* channels; -} ModelDefinitionSpec; - - -typedef struct ModelInstanceSpec { - uint32_t uid; - char* name; - ModelDefinitionSpec model_definition; - /* Reference to parsed YAML Documents. */ - YamlNode* spec; - YamlNode* propagators; - YamlDocList* yaml_doc_list; - /* Private data of the specific Model Instance. */ - void* private; -} ModelInstanceSpec; - - -typedef struct SimulationSpec { - /* Transport. */ - const char* transport; - char* uri; - uint32_t uid; - double timeout; - /* Simulation. */ - double step_size; - double end_time; - /* Model Instances, list, last entry all values set NULL (EOL detect). */ - ModelInstanceSpec* instance_list; -} SimulationSpec; - - -typedef struct ModelChannelDesc { - const char* name; - const char* function_name; - /* Reference to the parsed signal names. */ - const char** signal_names; - uint32_t signal_count; - /* Indicate if this Channel is connected to a Propagator Model. */ - bool propagator_source_channel; - bool propagator_target_channel; - /* Allocated vector table (one only depending on type). */ - double* vector_double; - void** vector_binary; - /* Additional vector tables supporting vector_binary. */ - uint32_t* vector_binary_size; /* Size of binary object. */ - uint32_t* vector_binary_buffer_size; /* Size of allocated buffer. */ -} ModelChannelDesc; - - -typedef struct ChannelSpec { - const char* name; - const char* alias; - /* Private data. */ - void* private; -} ChannelSpec; +The Model API provides two simple interfaces which facilitate the development +of models; the Model Interface which is concerned with the model lifecycle; and +the Signal Interface which facilitates signal exchange. + + +Model Interface +--------------- + +The Model Interface provides the necessary types, methods and objects required +for implementing a model. Such a model can easily participate in a simulation +by being connecting to a Simulation Bus (using the ModelC Importer) and then +exchanging signals with other models in that simulation by using the +provided SignalVector objects (which represent those signals). + +Additionally, model implementers may extend or modify the Model Interface +to support more complex integrations. -/** Signal Vector Interface -======================= +----------------------- Models exchange signals via the Simulation Bus using a Signal Vector. Signal -Vectors represent a logical grouping of signals (i.e. a collection of signals -belonging to an ECU interface or bus), they are defined by a `SignalGroup` -schema kind, and a Signal Vector can represent either scalar or binary values. +Vectors represent a logical grouping of signals (e.g. a collection of signals +belonging to an ECU interface or bus). They are defined by a `SignalGroup` +schema kind and may be configured to represent either either scalar +(double, int, bool) or binary values. Component Diagram @@ -119,23 +87,28 @@ Component Diagram -![](model-signal-vector.png) +![](model-interface.png) + + +Example (Model Interface) +------- + +{{< readfile file="../examples/model_interface.c" code="true" lang="c" >}} -Example +Example (Signal Vector Interface) ------- -{{< readfile file="../examples/model_signalvector.c" code="true" lang="c" >}} +{{< readfile file="../examples/signalvector_interface.c" code="true" lang="c" >}} */ -typedef struct SignalVector SignalVector; + +/* Model Interface. */ + +typedef ModelDesc* (*ModelCreate)(ModelDesc* m); +typedef int (*ModelStep)(ModelDesc* m, double* model_time, double stop_time); +typedef void (*ModelDestroy)(ModelDesc* m); +typedef ModelSignalIndex (*ModelIndex)(ModelDesc* m, const char* vname, + const char* sname); + + +typedef struct ModelVTable { + ModelCreate create; + ModelStep step; + ModelDestroy destroy; + ModelIndex index; +} ModelVTable; + + +typedef struct ModelDesc { + ModelVTable vtable; + ModelIndex index; + SimulationSpec* sim; + ModelInstanceSpec* mi; + SignalVector* sv; +} ModelDesc; + + +typedef struct ModelSignalIndex { + /* References, only set if index is valid. */ + SignalVector* sv; + double* scalar; + void** binary; + /* Indexes to the SignalVector object. */ + uint32_t vector; + uint32_t signal; +} ModelSignalIndex; + + +/* Implemented by Model. */ +DLL_PUBLIC ModelDesc* model_create(ModelDesc* m); +DLL_PUBLIC int model_step(ModelDesc* m, double* model_time, double stop_time); +DLL_PUBLIC void model_destroy(ModelDesc* m); + + +/* Provided by ModelC (via ModelIndex in ModelDesc and also ModelVTable). */ +DLL_PUBLIC ModelSignalIndex model_index_( + ModelDesc* model, const char* vname, const char* sname); + + +/* Signal Interface. */ typedef int (*BinarySignalAppendFunc)( SignalVector* sv, uint32_t index, void* data, uint32_t len); @@ -164,6 +192,7 @@ typedef void* (*BinarySignalCodecFunc)(SignalVector* sv, uint32_t index); typedef const char* (*SignalAnnotationGetFunc)( SignalVector* sv, uint32_t index, const char* name); + typedef struct SignalVectorVTable { BinarySignalAppendFunc append; BinarySignalResetFunc reset; @@ -172,6 +201,7 @@ typedef struct SignalVectorVTable { BinarySignalCodecFunc codec; } SignalVectorVTable; + typedef struct SignalVector { const char* name; const char* alias; @@ -203,136 +233,14 @@ typedef struct SignalVector { } SignalVector; -/** -Model Interface -=============== - -The Model Interface must be implemented by a Model. It includes the functions -necessary for a Model to be loaded and executed in the Dynamic Simulation -Environment. - - -Component Diagram ------------------ - - -![](model-interface.png) - - -Example -------- - -{{< readfile file="../examples/model_interface.c" code="true" lang="c" >}} - -*/ - -typedef int (*ModelSetupHandler)(ModelInstanceSpec* model_instance); -typedef int (*ModelDoStepHandler)(double* model_time, double stop_time); -typedef int (*ModelExitHandler)(ModelInstanceSpec* model_instance); - -typedef struct ModelInterfaceVTable { - ModelSetupHandler setup; - ModelDoStepHandler step; - ModelExitHandler exit; -} ModelInterfaceVTable; - -DLL_PUBLIC int MODEL_SETUP_FUNC(ModelInstanceSpec* model_instance); - - -/* Loader/Runner API Definition */ - -typedef struct ModelCArguments { - const char* transport; - char* uri; - const char* host; - uint32_t port; - double timeout; - uint8_t log_level; - double step_size; - double end_time; - uint32_t uid; - const char* name; - const char* file; - const char* path; - /* Parsed YAML Files. */ - YamlDocList* yaml_doc_list; - /* Allow detection of CLI provided arguments. */ - int timeout_set_by_cli; - int log_level_set_by_cli; - /* MStep "hidden" arguments. */ - uint32_t steps; -} ModelCArguments; - - -/* model.c - Model Interface. */ -DLL_PUBLIC int model_function_register(ModelInstanceSpec* model_instance, - const char* model_function_name, double step_size, - ModelDoStepHandler do_step_handler); -DLL_PUBLIC int model_configure_channel( - ModelInstanceSpec* model_instance, ModelChannelDesc* channel_desc); -DLL_PUBLIC ChannelSpec* model_build_channel_spec( - ModelInstanceSpec* model_instance, const char* channel_name); - - -/* signal.c - Signal Vector Interface. */ -DLL_PUBLIC SignalVector* model_sv_create(ModelInstanceSpec* mi); -DLL_PUBLIC void model_sv_destroy(SignalVector* sv); - -/* ncodec.c - Stream Interface (for NCodec). */ -DLL_PRIVATE void* model_sv_stream_create(SignalVector* sv, uint32_t idx); -DLL_PRIVATE void model_sv_stream_destroy(void* stream); - - -/* modelc.c - Runtime Interface (i.e. ModelC.exe). */ -DLL_PUBLIC int modelc_configure(ModelCArguments* args, SimulationSpec* sim); -DLL_PUBLIC ModelInstanceSpec* modelc_get_model_instance( - SimulationSpec* sim, const char* name); -DLL_PUBLIC int modelc_run(SimulationSpec* sim, bool run_async); -DLL_PUBLIC void modelc_exit(SimulationSpec* sim); -DLL_PUBLIC int modelc_sync(SimulationSpec* sim); -DLL_PUBLIC void modelc_shutdown(void); - - -/* modelc_debug.c - Debug Interface. */ -DLL_PUBLIC int modelc_step(ModelInstanceSpec* model_instance, double step_size); - - -/* modelc_args.c - CLI argument parsing routines. */ -DLL_PUBLIC void modelc_set_default_args(ModelCArguments* args, - const char* model_instance_name, double step_size, double end_time); -DLL_PUBLIC void modelc_parse_arguments( - ModelCArguments* args, int argc, char** argv, const char* doc_string); +/* Provided by ModelC (virtual methods of SignalVectorVTable). */ +DLL_PUBLIC int signal_append(SignalVector* sv, uint32_t index, + void* data, uint32_t len); +DLL_PUBLIC int signal_reset(SignalVector* sv, uint32_t index); +DLL_PUBLIC int signal_release(SignalVector* sv, uint32_t index); +DLL_PUBLIC void* signal_codec(SignalVector* sv, uint32_t index); +DLL_PUBLIC const char* signal_annotation(SignalVector* sv, uint32_t index, + const char* name); #endif // DSE_MODELC_MODEL_H_ diff --git a/dse/modelc/model/gateway.c b/dse/modelc/model/gateway.c index 13aef4e..64a6ec3 100644 --- a/dse/modelc/model/gateway.c +++ b/dse/modelc/model/gateway.c @@ -5,6 +5,7 @@ #include #include #include +#include #include #include #include @@ -15,81 +16,26 @@ /* Gateway Model Functions * These represent the Model Interface of the Gateway. */ - -static HashList __mcd_list; /* Storage for ModelChannelDesc objects. */ -static double __gw_step_size; - - -static void* channel_spec_generator(ModelInstanceSpec* mi, void* data) -{ - UNUSED(mi); - - const char* name = dse_yaml_get_scalar((YamlNode*)data, "name"); - const char* alias = dse_yaml_get_scalar((YamlNode*)data, "alias"); - if (name || alias) { - ChannelSpec* cs = calloc(1, sizeof(ChannelSpec)); - cs->name = name; - cs->alias = alias; - return cs; /* Caller to free. */ - } - return NULL; -} - - -DLL_PRIVATE int __model_gw_step__(double* model_time, double stop_time) +DLL_PRIVATE int __model_gw_step__(ModelDesc* model, double* model_time, double stop_time) { + UNUSED(model); *model_time = stop_time; - return 0; } -DLL_PRIVATE int __model_gw_setup__(ModelInstanceSpec* mi) +DLL_PRIVATE ModelDesc* __model_gw_create__(ModelDesc* model) { - int rc; - - hashlist_init(&__mcd_list, 10); - rc = model_function_register( - mi, mi->name, __gw_step_size, __model_gw_step__); - if (rc) log_fatal("Model registration failed!"); - - uint32_t index = 0; - do { - /* Enumerate over all channels of the Model Instance (not the Model). */ - SchemaObject object = { .doc = mi->spec }; - ChannelSpec* cs = schema_object_enumerator( - mi, &object, "channels", &index, channel_spec_generator); - if (cs == NULL) break; - - /* Register this channel. Priority to 'alias' over 'name' (for channel) - * as alias (if used) would match against a SignalGroup. Selector is - * defined on the Model Instance and will match to a Label on the - * signal group. */ - ModelChannelDesc* mcd = calloc(1, sizeof(ModelChannelDesc)); - mcd->name = cs->alias ? cs->alias : cs->name; - mcd->function_name = mi->name; - rc = model_configure_channel(mi, mcd); - hashlist_append(&__mcd_list, mcd); /* Keep the mcd object. */ - free(cs); - } while (1); - - return 0; + return model; } -DLL_PRIVATE int __model_gw_exit__(ModelInstanceSpec* mi) +DLL_PRIVATE void __model_gw_destroy__(ModelDesc* model) { - UNUSED(mi); - - for (uint32_t i = 0; i < hashlist_length(&__mcd_list); i++) { - void* o = hashlist_at(&__mcd_list, i); - free(o); - } - hashlist_destroy(&__mcd_list); - - return 0; + UNUSED(model); } + /** model_gw_setup ============== @@ -170,7 +116,6 @@ int model_gw_setup(ModelGatewayDesc* gw, const char* name, if (rc) log_fatal("Unable to configure Model C!"); /* Start the GW. */ - __gw_step_size = gw->sim->step_size; modelc_run(gw->sim, true); /* Calls __model_gw_setup__(). */ /* Complete the Gateway descriptor. */ diff --git a/dse/modelc/model/model.c b/dse/modelc/model/model.c index dc1af5b..2b350d2 100644 --- a/dse/modelc/model/model.c +++ b/dse/modelc/model/model.c @@ -65,70 +65,6 @@ void model_function_destroy(ModelFunction* model_function) } -/** -model_function_register -======================= - -Register a Model Function. A Model may register one or more Model Functions -with repeated calls to this function. - -Parameters ----------- -model_instance (ModelInstanceSpec*) -: The Model Instance object (provided via the `model_setup()` function of the - Model API). - -name (const char*) -: The name of the Model Function. - -step_size (double) -: The step size of the Model Function. - -do_step_handler (ModelDoStepHandler) -: The "do step" function of the Model Function. - -Returns -------- -0 -: The model function was registered. - -(errno) -: An error occurred during registration of the model function. The return - value is the `errno` which may indicate the reason for the failure. - -*/ -int model_function_register(ModelInstanceSpec* model_instance, const char* name, - double step_size, ModelDoStepHandler do_step_handler) -{ - int rc; - errno = 0; - - /* Create the Model Function object. */ - ModelFunction* mf = calloc(1, sizeof(ModelFunction)); - if (mf == NULL) { - log_error("ModelFunction malloc failed!"); - goto error_clean_up; - } - mf->name = name; - mf->step_size = step_size; - mf->do_step_handler = do_step_handler; - rc = hashmap_init(&mf->channels); - if (rc) { - log_error("Hashmap init failed for channels!"); - if (errno == 0) errno = rc; - goto error_clean_up; - } - /* Register the object with the Controller. */ - rc = controller_register_model_function(model_instance, mf); - if (rc && (errno != EEXIST)) goto error_clean_up; - return 0; - -error_clean_up: - model_function_destroy(mf); - return errno; -} - - static ModelFunctionChannel* _get_mfc(ModelInstanceSpec* model_instance, const char* model_function_name, const char* channel_name) { @@ -155,68 +91,6 @@ static ModelFunctionChannel* _get_mfc(ModelInstanceSpec* model_instance, } -static void _load_propagator_signal_names( - SimpleSet* set, YamlNode* signals_node, bool target_channel) -{ - if (signals_node == NULL) return; - - uint32_t _sig_count = hashlist_length(&signals_node->sequence); - for (uint32_t i = 0; i < _sig_count; i++) { - YamlNode* sig_node = hashlist_at(&signals_node->sequence, i); - YamlNode* n_node = NULL; - if (target_channel) { - n_node = dse_yaml_find_node(sig_node, "target"); - } else { - n_node = dse_yaml_find_node(sig_node, "source"); - } - /* Fallback the the common "signal" specifier. */ - if (n_node == NULL) { - n_node = dse_yaml_find_node(sig_node, "signal"); - } - set_add(set, n_node->scalar); - } -} - - -static void _find_signals_node_legacy(ModelInstanceSpec* model_instance, - const char* channel_name, YamlNode** channels_node, YamlNode** signals_node) -{ - *channels_node = NULL; - *signals_node = NULL; - const char* selectors[] = { "name" }; - const char* values[] = { channel_name }; - - /* Search in the Model definition. */ - if (model_instance->model_definition.doc) { - YamlNode* c_node = - dse_yaml_find_node_in_seq(model_instance->model_definition.doc, - "spec/channels", selectors, values, 1); - YamlNode* s_node = dse_yaml_find_node(c_node, "signals"); - if (c_node && s_node) { - log_info("Signals for channel[%s] selected from Model Definition.", - channel_name); - *channels_node = c_node; - *signals_node = s_node; - return; - } - } - - /* Search in the Model Instance. */ - if (model_instance->spec) { - YamlNode* c_node = dse_yaml_find_node_in_seq( - model_instance->spec, "channels", selectors, values, 1); - YamlNode* s_node = dse_yaml_find_node(c_node, "signals"); - if (c_node && s_node) { - log_info("Signals for channel[%s] selected from Model Instance.", - channel_name); - *channels_node = c_node; - *signals_node = s_node; - return; - } - } -} - - static HashList __handler_signal_list; static ModelChannelType __handler_signal_vector_type; @@ -319,78 +193,7 @@ void _load_signals(ModelInstanceSpec* model_instance, ChannelSpec* channel_spec, } -void _load_propagator_signals(ModelInstanceSpec* model_instance, - ModelChannelDesc* channel_desc, __signal_list_t* signal_list) -{ - YamlNode* propagators_node = model_instance->propagators; - /* Propagators are loaded dynamically from one or more YAML Docs - which means duplicate signal name might exist. Therefore parse the - signal names into a Set to avoid duplicates. - - NOTE: The order of signals for a propagator does not matter as the - propagator model will dynamically load the related configuration direct - from the YAML Documents. Therefore a Set is viable intermediate storage - container. - */ - SimpleSet signal_name_set; - set_init(&signal_name_set); - /* Parse out the signals from all listed propagators. */ - uint32_t _prop_count = hashlist_length(&propagators_node->sequence); - for (uint32_t i = 0; i < _prop_count; i++) { - YamlNode* prop_node = hashlist_at(&propagators_node->sequence, i); - YamlNode* prop_name_node = dse_yaml_find_node(prop_node, "name"); - // Find the propagator. - const char* selector[] = { "metadata/name" }; - const char* value[] = { prop_name_node->scalar }; - YamlNode* p_doc = dse_yaml_find_doc_in_doclist( - model_instance->yaml_doc_list, "Propagator", selector, value, 1); - assert(p_doc); - // Find the signals node. - YamlNode* signals_node = dse_yaml_find_node(p_doc, "spec/signals"); - _load_propagator_signal_names(&signal_name_set, signals_node, - channel_desc->propagator_target_channel); - } - /* Get the list of signals. */ - uint64_t _sig_count64 = 0; - signal_list->names = - (const char**)set_to_array(&signal_name_set, &_sig_count64); - signal_list->length = _sig_count64; - set_destroy(&signal_name_set); -} - - -void _load_signals_legacy(ModelInstanceSpec* model_instance, - const char* channel_name, __signal_list_t* signal_list) -{ - /* Models are expected to define and reference signals according to the - index of the signal in the Model Definition. - - IMPORTANT: signals must be parsed in the order which they occur in the - Model Definition YAML Doc. - */ - YamlNode* c_node = NULL; - YamlNode* signals_node = NULL; - _find_signals_node_legacy( - model_instance, channel_name, &c_node, &signals_node); - if (c_node == NULL) { - log_fatal("Channel (%s) not found in model definition", channel_name); - } - if (signals_node == NULL) { - log_fatal("Signals node not found in model definition "); - } - assert(c_node); - assert(signals_node); - signal_list->length = hashlist_length(&signals_node->sequence); - signal_list->names = calloc(signal_list->length, sizeof(const char*)); - for (uint32_t i = 0; i < signal_list->length; i++) { - YamlNode* sig_node = hashlist_at(&signals_node->sequence, i); - YamlNode* n_node = dse_yaml_find_node(sig_node, "signal"); - signal_list->names[i] = n_node->scalar; - } -} - - -/** +/* model_configure_channel ======================= @@ -459,25 +262,11 @@ int model_configure_channel( return 0; } - /* Find the signal definitions: - If the Model Instance specifies a Propagator, then the signals will be - defined under Propagator:spec/signals. - Otherwise, the signals will be under Model:spec/channels[name]/signals. - */ + /* Load signals via SignalGroups. */ __signal_list_t signal_list = { NULL, 0 }; ModelChannelType vector_type = MODEL_VECTOR_DOUBLE; assert(model_instance->spec); - if (model_instance->propagators) { - _load_propagator_signals(model_instance, channel_desc, &signal_list); - } else { - /* Load signals via SignalGroups. */ - _load_signals(model_instance, channel_spec, &signal_list, &vector_type); - if (signal_list.length == 0) { - /* Fallback to legacy method. */ - _load_signals_legacy( - model_instance, channel_spec->name, &signal_list); - } - } + _load_signals(model_instance, channel_spec, &signal_list, &vector_type); log_notice(" Unique signals identified: %u", signal_list.length); /* Init the channel and register signals. */ @@ -519,3 +308,168 @@ int model_configure_channel( /* Brutal, eh? */ return 0; } + + +DLL_PRIVATE ModelSignalIndex __model_index__(ModelDesc* m, const char* vname, const char* sname) +{ + ModelSignalIndex index = {}; + if (m == NULL || m->sv == NULL) return index; + + SignalVector* sv = m->sv; + uint32_t v_idx = 0; + while (sv && sv->name) { + if (strcmp(sv->alias, vname) == 0) { + for (uint32_t s_idx = 0; s_idx < sv->count; s_idx++) { + if (strcmp(sv->signal[s_idx], sname) == 0) { + /* Match! */ + index.sv = sv; + index.vector = v_idx; + index.signal = s_idx; + if (sv->is_binary) { + index.binary = &(sv->binary[s_idx]); + } else { + index.scalar = &(sv->scalar[s_idx]); + } + return index; + } + } + } + sv++; + v_idx++; + } + return index; +} + + +/** +model_create +============ + +> Optional method of `ModelVTable` interface. + +Called by the Model Runtime to create a new instance of this model. + +The `model_create()` method may extend or mutilate the provided Model +Descriptor. When extending the Model Descriptor _and_ allocating additional +resources then the `model_destroy()` method should also be implemented. + +Fault conditions can be communicated to the caller by setting variable +`errno` to a non-zero value. Additionally, `log_fatal()` can be used to +immediately halt execution of a model. + +Parameters +---------- +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +Returns +------- +NULL +: The Channel was configured. + +(ModelDesc*) +: Pointer to a new, or mutilated, version of the Model Descriptor object. The + original Model Descriptor object will be released by the Model Runtime (i.e. + don't call `free()`). + +errno <> 0 (indirect) +: Indicates an error condition. + +Example +------- + +{{< readfile file="../examples/model_create.c" code="true" lang="c" >}} + + +*/ +extern ModelDesc* model_create(ModelDesc* m); + + +/** +model_step +========== + +> Mandatory method of `ModelVTable` interface. Alternatively, Model implementers + may specify the `ModelVTable.step` method dynamically by mutilating the + Model Descriptor in the `model_create()` method, or even at runtime. + +Called by the Model Runtime to step the model for a time interval. + +Parameters +---------- +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +model_time (double*) +: (in/out) Specifies the model time for this step of the model. + +stop_time (double) +: Specifies the stop time for this step of the model. The model step should not + exceed this time. + +Returns +------- +0 +: The step completed without error. + +<>0 +: An error occurred at some point during the step execution. + +model_time (via parameter) +: The final model time reached for this step. This value may be less than + `stop_time` if a step decides to return early. +*/ +extern int model_step(ModelDesc* model, double* model_time, double stop_time); + + +/** +model_destroy +============= + +> Optional method of `ModelVTable` interface. + +Called by the Model Runtime at the end of a simulation, the `model_destroy()` +function may be implemented by a Model Integrator to perform any custom +cleanup operations (e.g. releasing instance related resources, such as open +files or allocated memory). + +Parameters +---------- +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +*/ +extern void model_destroy(ModelDesc* model); + + +/** +model_index_ +============ + +> Provided method (by the Runtime). Model implementers may specify + a different index method by mutilating the Model Descriptor in the + `model_create()` method, or even at runtime. + +A model may use this method to index a signal that is contained within the +Signal Vectors of the Model Descriptor. + +Parameters +---------- +model (ModelDesc*) +: The Model Descriptor object representing an instance of this model. + +vname (const char*) +: The name (alias) of the Signal Vector. + +sname (const char*) +: The name of the signal within the Signal Vector. + +Returns +------- +ModelSignalIndex +: An index. When valid, either the `scalar` or `binary` fields will be set to + a valid pointer (i.e. not NULL). + +*/ +extern ModelSignalIndex model_index_(ModelDesc* model, const char* vname, + const char* sname); diff --git a/dse/modelc/model/signal.c b/dse/modelc/model/signal.c index bc0ef88..08c9593 100644 --- a/dse/modelc/model/signal.c +++ b/dse/modelc/model/signal.c @@ -7,8 +7,9 @@ #include #include #include +#include #include -#include +#include #include #include @@ -255,7 +256,7 @@ static int _count_sv(void* _mf, void* _number) } -/** +/* model_sv_create =============== @@ -272,11 +273,6 @@ SignalVector (pointer to NULL terminated list) : A list of SignalVector objects representing the signals assigned to a model. The list is NULL terminated (sv->name == NULL). Caller to free. -Example -------- - -{{< readfile file="../examples/model_sv_create.c" code="true" lang="c" >}} - */ SignalVector* model_sv_create(ModelInstanceSpec* mi) { @@ -321,7 +317,7 @@ SignalVector* model_sv_create(ModelInstanceSpec* mi) } -/** +/* model_sv_destroy ================ @@ -360,3 +356,188 @@ void model_sv_destroy(SignalVector* sv) free(sv_save); } + + + +/** +signal_append +============= + +Append data to the end of the specified binary signal. The append method will +resize the buffers of the binary signal as required. + +Parameters +---------- +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +data (void*) +: Address/pointer to the data which should be appended to the binary signal. + +len (uint32_t) +: Length of the provided data buffer being appended. + +Returns +------- +0 +: The operation completed without error. + +<>0 +: Indicates an error condition. Inspect `errno` for additional information. +*/ +extern int signal_append(SignalVector* sv, uint32_t index, + void* data, uint32_t len); + + +/** +signal_reset +============ + +Reset a binary signal (e.g. sets its buffer length to 0). The buffers of the +binary signal are not released (see `signal_release()`). + +Parameters +---------- +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +Returns +------- +0 +: The operation completed without error. + +<>0 +: Indicates an error condition. Inspect `errno` for additional information. +*/ +extern int signal_reset(SignalVector* sv, uint32_t index); + + +/** +signal_release +============== + +Release the resources allocated to a binary signal (e.g. free the buffer). + +Parameters +---------- +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +Returns +------- +0 +: The operation completed without error. + +<>0 +: Indicates an error condition. Inspect `errno` for additional information. +*/ +extern int signal_release(SignalVector* sv, uint32_t index); + + +/** +signal_codec +============ + +Return a pointer to the Codec object associated with a binary signal. + +Codec objects are created when a binary signal is specified with a `mime_type` +annotation. + +Parameters +---------- +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +Returns +------- +void* +: The Codec object associated with the binary signal. + +NULL +: The binary signal does not have an associated Codec object. + +Example (Codec Specification) +------- + +```yaml +kind: SignalGroup +metadata: + name: network + labels: + channel: network_vector + annotations: + vector_type: binary +spec: + signals: + - signal: can_bus + annotations: + mime_type: application/x-automotive-bus; interface=stream; type=frame; bus=can; schema=fbs; bus_id=1; node_id=2; interface_id=3 +``` + +Reference +--------- + +[Network Codec API](https://github.com/boschglobal/dse.standards/tree/main/dse/ncodec) + +*/ +extern void* signal_codec(SignalVector* sv, uint32_t index); + + +/** +signal_annotation +================= + +Get an annotation from a signal definition. + +Parameters +---------- +sv (SignalVector*) +: The Signal Vector object containing the signal. + +index (uint32_t) +: Index of the signal in the Signal Vector object. + +name (const char*) +: The name of the annotation. + +Returns +------- +const char* +: The annotation value. + +Example (Annotation Specification) +------- + +```yaml +kind: SignalGroup +metadata: + name: data +spec: + signals: + - signal: counter + annotations: + initial_value: 10 +``` + +Example (Code Usage) +------- + +{{< readfile file="../examples/signalvector_annotation.c" code="true" lang="c" >}} + + +NULL +: The requested annotation was not found. +*/ +extern const char* signal_annotation(SignalVector* sv, uint32_t index, const char* name); diff --git a/dse/modelc/runtime.h b/dse/modelc/runtime.h new file mode 100644 index 0000000..df15676 --- /dev/null +++ b/dse/modelc/runtime.h @@ -0,0 +1,150 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 + +#ifndef DSE_MODELC_RUNTIME_H_ +#define DSE_MODELC_RUNTIME_H_ + +#include + + +/** +Runtime API +=========== + +The Runtime API provides methods for implementing a model Runtime/Importer +which can be used to load, configure and execute a model. + +*/ + +typedef struct ChannelSpec { + const char* name; + const char* alias; + /* Private data. */ + void* private; +} ChannelSpec; + + +typedef struct ModelChannelDesc { + const char* name; + const char* function_name; + /* Reference to the parsed signal names. */ + const char** signal_names; + uint32_t signal_count; + /* Indicate if this Channel is connected to a Propagator Model. */ + bool propagator_source_channel; + bool propagator_target_channel; + /* Allocated vector table (one only depending on type). */ + double* vector_double; + void** vector_binary; + /* Additional vector tables supporting vector_binary. */ + uint32_t* vector_binary_size; /* Size of binary object. */ + uint32_t* vector_binary_buffer_size; /* Size of allocated buffer. */ +} ModelChannelDesc; + + +typedef struct ModelDefinitionSpec { + const char* name; + const char* path; + const char* file; + char* full_path; /* path + file */ + + /* Reference to parsed YAML Documents. */ + void* doc; + void* channels; +} ModelDefinitionSpec; + + +typedef struct ModelInstanceSpec { + uint32_t uid; + char* name; + ModelDesc* model_desc; + ModelDefinitionSpec model_definition; + + /* Reference to parsed YAML Documents. */ + void* spec; + void* yaml_doc_list; + /* Private data of the specific Model Instance. */ + void* private; +} ModelInstanceSpec; + + +typedef struct SimulationSpec { + /* Transport. */ + const char* transport; + char* uri; + uint32_t uid; + double timeout; + /* Simulation. */ + double step_size; + double end_time; + /* Model Instances, list, last entry all values set NULL (EOL detect). */ + ModelInstanceSpec* instance_list; +} SimulationSpec; + + +/* Loader/Runner API Definition */ +typedef struct ModelCArguments { + const char* transport; + char* uri; + const char* host; + uint32_t port; + double timeout; + uint8_t log_level; + double step_size; + double end_time; + uint32_t uid; + const char* name; + const char* file; + const char* path; + /* Parsed YAML Files. */ + void* yaml_doc_list; + /* Allow detection of CLI provided arguments. */ + int timeout_set_by_cli; + int log_level_set_by_cli; + /* MStep "hidden" arguments. */ + uint32_t steps; +} ModelCArguments; + + +/* modelc.c - Runtime Interface (i.e. ModelC.exe). */ +DLL_PUBLIC int modelc_configure(ModelCArguments* args, SimulationSpec* sim); +DLL_PUBLIC ModelInstanceSpec* modelc_get_model_instance( + SimulationSpec* sim, const char* name); +DLL_PUBLIC int modelc_run(SimulationSpec* sim, bool run_async); +DLL_PUBLIC void modelc_exit(SimulationSpec* sim); +DLL_PUBLIC int modelc_sync(SimulationSpec* sim); +DLL_PUBLIC void modelc_shutdown(void); +DLL_PUBLIC int modelc_model_create( + SimulationSpec* sim, ModelInstanceSpec* mi, ModelVTable* model_vtable); + + +/* modelc_debug.c - Debug Interface. */ +DLL_PUBLIC int modelc_step(ModelInstanceSpec* model_instance, double step_size); + + +/* modelc_args.c - CLI argument parsing routines. */ +DLL_PUBLIC void modelc_set_default_args(ModelCArguments* args, + const char* model_instance_name, double step_size, double end_time); +DLL_PUBLIC void modelc_parse_arguments( + ModelCArguments* args, int argc, char** argv, const char* doc_string); + + +/* signal.c - Signal Vector Interface. */ +DLL_PUBLIC SignalVector* model_sv_create(ModelInstanceSpec* mi); +DLL_PUBLIC void model_sv_destroy(SignalVector* sv); + + +/* ncodec.c - Stream Interface (for NCodec). */ +DLL_PRIVATE void* model_sv_stream_create(SignalVector* sv, uint32_t idx); +DLL_PRIVATE void model_sv_stream_destroy(void* stream); + + +/* model.c - Model Interface. */ +DLL_PRIVATE ChannelSpec* model_build_channel_spec( + ModelInstanceSpec* model_instance, const char* channel_name); +DLL_PRIVATE int model_configure_channel( + ModelInstanceSpec* model_instance, ModelChannelDesc* channel_desc); + + +#endif // DSE_MODELC_RUNTIME_H_ diff --git a/dse/modelc/schema.h b/dse/modelc/schema.h index 48babe3..5a67936 100644 --- a/dse/modelc/schema.h +++ b/dse/modelc/schema.h @@ -7,7 +7,7 @@ #include #include -#include +#include #ifndef DLL_PUBLIC diff --git a/dse/modelc/tools/mcl_model/model.c b/dse/modelc/tools/mcl_model/model.c index b18fb9d..f5101b5 100644 --- a/dse/modelc/tools/mcl_model/model.c +++ b/dse/modelc/tools/mcl_model/model.c @@ -13,155 +13,124 @@ #include -#define ARRAY_LENGTH(array) (sizeof((array)) / sizeof((array)[0])) +#define MCL_CHANNEL "mcl_channel" -/* Model Function definitions. */ -#define MODEL_FUNCTION_NAME "mcl_model" -#define MODEL_FUNCTION_DO_STEP do_step -#define MODEL_FUNCTION_STEP_SIZE 0.005 -#define MODEL_FUNCTION_CHANNEL "mcl_channel" +typedef struct { + ModelDesc model; + /* MCL Properties. */ + MclInstanceDesc* mcl_instance; +} MclExtendedModelDesc; -/* Copy of MCL Instance for DO_STEP(). */ -static MclInstanceDesc* __mcl_instance; - -static int __mcl_execute(MclStrategyAction action) -{ - assert(__mcl_instance); - assert(__mcl_instance->strategy); - assert(__mcl_instance->strategy->execute_func); - - int rc = __mcl_instance->strategy->execute_func(__mcl_instance, action); - return rc; -} - - -/* Model Function do_step() definition. */ -int MODEL_FUNCTION_DO_STEP(double* model_time, double stop_time) +static int __mcl_execute(ModelDesc* model, MclStrategyAction action) { - log_debug("Model function do_step() called: model_time=%f, stop_time=%f", - *model_time, stop_time); - MclInstanceDesc* mcl_instance = __mcl_instance; - assert(mcl_instance); - assert(mcl_instance->channel); + MclExtendedModelDesc* m = (MclExtendedModelDesc*)model; + MclInstanceDesc* mcl_instance = m->mcl_instance; assert(mcl_instance->strategy); - assert(mcl_instance->strategy->mcl_instance); - - /* Marshall data from Channel to MCL Models. */ - mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_MARSHALL_OUT); - /* Execute the step strategy (i.e. call step_func() on all MCL Models). */ - log_debug("Call MCL Strategy: step_func ..."); - mcl_instance->strategy->model_time = *model_time; - mcl_instance->strategy->stop_time = stop_time; - int rc = mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_STEP); - if (rc != 0) return rc; - /* Marshall data from Channel to MCL Models. */ - mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_MARSHALL_IN); - /* Set the model_time to the *finalised* stop_time of the strategy. */ - *model_time = mcl_instance->strategy->stop_time; + assert(mcl_instance->strategy->execute_func); + int rc = mcl_instance->strategy->execute_func(mcl_instance, action); return rc; } -/* Model Setup: a handle to this function will be dynamically loaded by the - Controller and it (this function) will be called to initialise the Model - and its Model Functions. */ -int MODEL_SETUP_FUNC(ModelInstanceSpec* model_instance) +ModelDesc* model_create(ModelDesc* model) { - log_notice("Model function " MODEL_SETUP_FUNC_STR " called"); - assert(model_instance); - int rc = 0; - - /* Register this Model Function with the SimBus/ModelC. */ - rc = model_function_register(model_instance, MODEL_FUNCTION_NAME, - MODEL_FUNCTION_STEP_SIZE, MODEL_FUNCTION_DO_STEP); - if (rc != 0) return rc; - - /* Register channels (and get storage). */ - static ModelChannelDesc channel_desc = { - .name = MODEL_FUNCTION_CHANNEL, - .function_name = MODEL_FUNCTION_NAME, - }; - rc = model_configure_channel(model_instance, &channel_desc); - if (rc != 0) return rc; - log_debug("%p", channel_desc.vector_double); - assert(channel_desc.vector_double); + /* Extend the ModelDesc object (using a shallow copy). */ + MclExtendedModelDesc* m = calloc(1, sizeof(MclExtendedModelDesc)); + memcpy(m, model, sizeof(ModelDesc)); + ModelInstanceSpec* mi = m->model.mi; - /* Load the MCL dll's. */ + /* Load the MCL dll's. */ log_debug("Load the MCL(s) ..."); errno = 0; - rc = mcl_load(model_instance); + int rc = mcl_load(mi); if (rc) { if (errno == 0) errno = ECANCELED; - log_error("Failed to load the configured MCL(s)!"); - return rc; + log_fatal("Failed to load the configured MCL(s)!"); } /* Create the MCL Instance. */ log_debug("Create the MCL Instance ..."); errno = 0; - rc = mcl_create(model_instance); + rc = mcl_create(mi); if (rc) { if (errno == 0) errno = ECANCELED; - log_error("Failed to create the MCL Instance!"); - return rc; + log_fatal("Failed to create the MCL Instance!"); } /* Save a reference to MCL Instance for the do_step(). */ - assert(model_instance->private); - ModelInstancePrivate* mip = model_instance->private; + assert(mi->private); + ModelInstancePrivate* mip = mi->private; assert(mip->mcl_instance); - __mcl_instance = mip->mcl_instance; /* Save for use by do_step(). */ - MclInstanceDesc* mcl_instance = __mcl_instance; - /* Set the MCL Instance Channel to the Channel created here. */ - mcl_instance->channel = &channel_desc; + MclInstanceDesc* mcl_instance = m->mcl_instance = mip->mcl_instance; + + /* Set the MCL Channel (should be the first/only SV). */ + assert(strcmp(m->model.sv->name, MCL_CHANNEL) == 0); + mcl_instance->mcl_channel_sv = m->model.sv; /* Set the MCL Strategy execute function. */ mcl_instance->strategy->execute = __mcl_execute; /* Execute the load strategy (i.e. call load_func() on all MCL Models). */ log_debug("Call MCL Strategy: load_func ..."); - rc = mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_LOAD); + rc = mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_LOAD); if (rc) { if (errno == 0) errno = ECANCELED; - log_error("MCL strategy ACTION_LOAD failed!"); - return rc; + log_fatal("MCL strategy ACTION_LOAD failed!"); } /* Execute the Init strategy (i.e. call load_func() on all MCL Models). */ log_debug("Call MCL Strategy: init_func ..."); - rc = mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_INIT); + rc = mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_INIT); if (rc) { if (errno == 0) errno = ECANCELED; - log_error("MCL strategy ACTION_INIT failed!"); - return rc; + log_fatal("MCL strategy ACTION_INIT failed!"); } - return 0; + /* Return the extended object. */ + return (ModelDesc*)m; } -/* Model Exit: called before the Controller exits the simulation (and sends - the SyncExit message to the SimBus). */ -int MODEL_EXIT_FUNC(ModelInstanceSpec* model_instance) +int model_step(ModelDesc* model, double* model_time, double stop_time) { - log_notice("Model function " MODEL_EXIT_FUNC_STR " called"); - assert(model_instance); - assert(model_instance->private); - ModelInstancePrivate* mip = model_instance->private; - assert(mip->mcl_instance); - MclInstanceDesc* mcl_instance = mip->mcl_instance; + MclExtendedModelDesc* m = (MclExtendedModelDesc*)model; + MclInstanceDesc* mcl_instance = m->mcl_instance; + assert(mcl_instance); + //assert(mcl_instance->channel); assert(mcl_instance->strategy); assert(mcl_instance->strategy->mcl_instance); - /* Execute the unload strategy (i.e. call unload_func() on all MCL Models). - */ - log_debug("Call MCL Strategy: unload_func ..."); - int rc = mcl_instance->strategy->execute(MCL_STRATEGY_ACTION_UNLOAD); - mcl_destroy(model_instance); - /* Set the local copy of mcl_instance to NULL. */ - __mcl_instance = NULL; + /* Marshall data from Channel to MCL Models. */ + mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_MARSHALL_OUT); + /* Execute the step strategy (i.e. call step_func() on all MCL Models). */ + log_debug("Call MCL Strategy: step_func ..."); + mcl_instance->strategy->model_time = *model_time; + mcl_instance->strategy->stop_time = stop_time; + int rc = mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_STEP); + if (rc != 0) return rc; + /* Marshall data from Channel to MCL Models. */ + mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_MARSHALL_IN); + /* Set the model_time to the *finalised* stop_time of the strategy. */ + *model_time = mcl_instance->strategy->stop_time; return rc; } + + +void model_destroy(ModelDesc* model) +{ + MclExtendedModelDesc* m = (MclExtendedModelDesc*)model; + ModelInstanceSpec* mi = m->model.mi; + MclInstanceDesc* mcl_instance = m->mcl_instance; + assert(mcl_instance->strategy); + assert(mcl_instance->strategy->mcl_instance); + + log_notice("Model function " MODEL_DESTROY_FUNC_NAME " called"); + + /* Execute the unload strategy. */ + log_debug("Call MCL Strategy: unload_func ..."); + mcl_instance->strategy->execute(model, MCL_STRATEGY_ACTION_UNLOAD); + mcl_destroy(mi); +} diff --git a/dse/modelc/tools/modelc/modelc.c b/dse/modelc/tools/modelc/modelc.c index c042475..66b6048 100644 --- a/dse/modelc/tools/modelc/modelc.c +++ b/dse/modelc/tools/modelc/modelc.c @@ -11,7 +11,7 @@ #include #include #include -#include +#include #include #include #include diff --git a/dse/modelc/tools/mstep/mstep.c b/dse/modelc/tools/mstep/mstep.c index 2509dba..c6c956e 100644 --- a/dse/modelc/tools/mstep/mstep.c +++ b/dse/modelc/tools/mstep/mstep.c @@ -10,11 +10,11 @@ #include #include #include -#include +#include #include -#define STEP_SIZE 0.005 +#define STEP_SIZE MODEL_DEFAULT_STEP_SIZE #define END_TIME 3600 #define STEPS 10 @@ -46,10 +46,11 @@ static void print_signal_vectors(SignalVector* sv); * - sample: 0.0005 * values: * - D_BrakePedal_0_1_f: 0.0 # 0..1 - * - D_ActMode_1_3_f: 3 # 1 = MC Press, 2 = Pedal Force, 3 = Rod Stroke + * - D_ActMode_1_3_f: 3 # 1 = MC Press, 2 = Pedal Force, 3 = Rod + * Stroke * - sample: 0.0010 * values: - * - D_BrakePedal_0_1_f: 0.3 + * - D_BrakePedal_0_1_f: 0.3 */ typedef struct ValueObject { char* signal; @@ -119,20 +120,25 @@ int main(int argc, char** argv) log_error("ERROR: dlopen call failed: %s", dlerror()); log_fatal("Model library not loaded!"); } - ModelSetupHandler model_setup_func = dlsym(handle, MODEL_SETUP_FUNC_STR); - ModelExitHandler model_exit_func = dlsym(handle, MODEL_EXIT_FUNC_STR); - if (model_setup_func == NULL) log_fatal("model_setup_func not found!"); - + ModelVTable vtable; + vtable.create = dlsym(handle, MODEL_CREATE_FUNC_NAME); + vtable.step = dlsym(handle, MODEL_STEP_FUNC_NAME); + vtable.destroy = dlsym(handle, MODEL_DESTROY_FUNC_NAME); + if (vtable.create == NULL && vtable.step == NULL) { + log_fatal("vtable not complete!"); + } - /* Call the setup function of the Model. */ - rc = model_setup_func(mi); - if (rc) log_fatal("Call: model_setup_func failed! (rc=%d)!", rc); - SignalVector* sv = model_sv_create(mi); + /* Call the create function of the Model. */ + rc = modelc_model_create(&sim, mi, &vtable); + if (rc) { + log_fatal("Call: model_setup_func failed! (rc=%d)!", rc); + } + memcpy(&vtable, &mi->model_desc->vtable, sizeof(ModelVTable)); + SignalVector* sv = mi->model_desc->sv; print_signal_vectors(sv); - /* Load any input data. */ - Enumerator samples = { .sv = sv }; + Enumerator samples = { .sv = sv }; SchemaObjectSelector selector = { .kind = "Input", .name = args.name, @@ -192,9 +198,8 @@ int main(int argc, char** argv) /* Call the exit function of the Model. */ - if (model_exit_func) { - rc = model_exit_func(mi); - if (rc) log_fatal("Executing model exit failed! (rc=%d)!", rc); + if (vtable.destroy) { + vtable.destroy(mi->model_desc); } exit(0); @@ -252,15 +257,16 @@ static void* value_object_generator(ModelInstanceSpec* mi, void* data) YamlNode* node = data; - int len = hashmap_number_keys(node->mapping); + int len = hashmap_number_keys(node->mapping); if (len == 0) return NULL; - char** keys = hashmap_keys(&node->mapping); + char** keys = hashmap_keys(&node->mapping); ValueObject* vo = calloc(1, sizeof(ValueObject)); vo->signal = keys[0]; dse_yaml_get_double(node, keys[0], &vo->value); /* Free the keys, except for the first entry ... which is returned. */ - for (int i = 1; i < len; i++) free(keys[i]); + for (int i = 1; i < len; i++) + free(keys[i]); free(keys); return vo; /* Caller to free, also vo->signal. */ diff --git a/dse/modelc/tools/simbus/simbus.c b/dse/modelc/tools/simbus/simbus.c index 99571d7..f1d8b19 100644 --- a/dse/modelc/tools/simbus/simbus.c +++ b/dse/modelc/tools/simbus/simbus.c @@ -11,7 +11,7 @@ #include #include #include -#include +#include #include diff --git a/tests/cmocka/CMakeLists.txt b/tests/cmocka/CMakeLists.txt index 06fc4ac..694c970 100644 --- a/tests/cmocka/CMakeLists.txt +++ b/tests/cmocka/CMakeLists.txt @@ -93,6 +93,7 @@ set(DSE_NCODEC_INCLUDE_DIR "${DSE_NCODEC_SOURCE_DIR}") # Set the project paths # ===================== set(DSE_MODELC_SOURCE_DIR ../../dse/modelc) +set(DSE_MOCKS_SOURCE_DIR ../../dse/mocks) set(DSE_MODELC_SOURCE_FILES ${DSE_MODELC_SOURCE_DIR}/model/gateway.c ${DSE_MODELC_SOURCE_DIR}/model/model.c @@ -103,9 +104,11 @@ set(DSE_MODELC_SOURCE_FILES ${DSE_MODELC_SOURCE_DIR}/controller/loader.c ${DSE_MODELC_SOURCE_DIR}/controller/model_function.c ${DSE_MODELC_SOURCE_DIR}/controller/modelc.c + ${DSE_MODELC_SOURCE_DIR}/controller/modelc_debug.c ${DSE_MODELC_SOURCE_DIR}/controller/modelc_args.c ${DSE_MODELC_SOURCE_DIR}/controller/step.c + ${DSE_MOCKS_SOURCE_DIR}/simmock.c ) set(DSE_MODELC_INCLUDE_DIR "${DSE_MODELC_SOURCE_DIR}/../..") @@ -138,7 +141,7 @@ target_include_directories(test_model ) target_compile_definitions(test_model PUBLIC -# UNIT_TESTING # Using valgrind instead. + CMOCKA_TESTING PRIVATE PLATFORM_OS="${CDEF_PLATFORM_OS}" PLATFORM_ARCH="${CDEF_PLATFORM_ARCH}" @@ -160,4 +163,41 @@ install( model/signal.yaml DESTINATION resources/model -) \ No newline at end of file +) + + +# Target - Model Interface +# ------------------------ +add_executable(test_model_interface + model/interface/__test__.c + model/interface/test_model_interface.c + model/stub_controller.c + ${DSE_MODELC_SOURCE_FILES} + ${DSE_NCODEC_SOURCE_FILES} + ${DSE_CLIB_SOURCE_FILES} +) +target_include_directories(test_model_interface + PRIVATE + ${DSE_MODELC_INCLUDE_DIR} + ${DSE_NCODEC_INCLUDE_DIR} + ${DSE_CLIB_INCLUDE_DIR} + ${YAML_SOURCE_DIR}/include + ./ +) +target_compile_definitions(test_model_interface + PUBLIC + CMOCKA_TESTING + PRIVATE + PLATFORM_OS="${CDEF_PLATFORM_OS}" + PLATFORM_ARCH="${CDEF_PLATFORM_ARCH}" +) +target_link_libraries(test_model_interface + PRIVATE + dse_ncodec + cmocka + yaml + dl + m + # -Wl,--wrap=strdup # Wrapping strdup does not work with libyaml. +) +install(TARGETS test_model_interface) diff --git a/tests/cmocka/Makefile b/tests/cmocka/Makefile index 2d2815c..2f3bde5 100644 --- a/tests/cmocka/Makefile +++ b/tests/cmocka/Makefile @@ -24,6 +24,7 @@ build: run: cd build/_out; $(GDB_CMD) bin/test_model + cd build/_out; $(GDB_CMD) bin/test_model_interface clean: rm -rf build diff --git a/tests/cmocka/model/interface/__test__.c b/tests/cmocka/model/interface/__test__.c new file mode 100644 index 0000000..6d8403f --- /dev/null +++ b/tests/cmocka/model/interface/__test__.c @@ -0,0 +1,17 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 +// SPDX-License-Identifier: BIOS-4.0 + +#include + + +extern int run_model_interface_tests(void); + + +int main() +{ + int rc = 0; + rc |= run_model_interface_tests(); + return rc; +} diff --git a/tests/cmocka/model/interface/test_model_interface.c b/tests/cmocka/model/interface/test_model_interface.c new file mode 100644 index 0000000..1f7cd1c --- /dev/null +++ b/tests/cmocka/model/interface/test_model_interface.c @@ -0,0 +1,204 @@ +// Copyright 2024 Robert Bosch GmbH +// +// SPDX-License-Identifier: Apache-2.0 +// SPDX-License-Identifier: BIOS-4.0 + +#include +#include +#include +#include +#include + + +static char __entry_path__[200]; + + +static int test_setup(void** state) +{ + UNUSED(state); + return 0; +} + + +static int test_teardown(void** state) +{ + SimMock* mock = *state; + + simmock_exit(mock); + simmock_free(mock); + + chdir(__entry_path__); + + return 0; +} + + +#define MINIMAL_INST_NAME "minimal_inst" +#define MINIMAL_SIGNAL_COUNTER 0 + +void test_model__minimal(void** state) +{ + chdir("../../../../dse/modelc/build/_out/examples/minimal"); + + const char* inst_names[] = { + MINIMAL_INST_NAME, + }; + char* argv[] = { + (char*)"test_model_interface", + (char*)"--name=" MINIMAL_INST_NAME, + (char*)"--logger=5", // 1=debug, 5=QUIET (commit with 5!) + (char*)"data/simulation.yaml", + (char*)"data/model.yaml", + }; + SimMock* mock = *state = simmock_alloc(inst_names, ARRAY_SIZE(inst_names)); + simmock_configure(mock, argv, ARRAY_SIZE(argv), ARRAY_SIZE(inst_names)); + ModelMock* model = simmock_find_model(mock, MINIMAL_INST_NAME); + simmock_load(mock); + simmock_load_model_check(model, false, true, false); + simmock_setup(mock, "data_channel", NULL); + + /* Initial value. */ + double counter = 0.0; + simmock_print_scalar_signals(mock, LOG_DEBUG); + /* T0 ... Tn */ + for (uint32_t i = 0; i < 5; i++) { + /* Do the check. */ + SignalCheck checks[] = { + { .index = MINIMAL_SIGNAL_COUNTER, .value = counter }, + }; + simmock_signal_check( + mock, MINIMAL_INST_NAME, checks, ARRAY_SIZE(checks), NULL); + /* Step the model. */ + assert_int_equal(simmock_step(mock, true), 0); + simmock_print_scalar_signals(mock, LOG_DEBUG); + counter += 1.0; + } +} + + +#define EXTENDED_INST_NAME "extended_inst" +#define EXTENDED_SIGNAL_COUNTER 0 +#define EXTENDED_SIGNAL_ODD 1 +#define EXTENDED_SIGNAL_EVEN 2 + +void test_model__extended(void** state) +{ + chdir("../../../../dse/modelc/build/_out/examples/extended"); + + const char* inst_names[] = { + EXTENDED_INST_NAME, + }; + char* argv[] = { + (char*)"test_model_interface", + (char*)"--name=" EXTENDED_INST_NAME, + (char*)"--logger=5", // 1=debug, 5=QUIET (commit with 5!) + (char*)"data/simulation.yaml", + (char*)"data/model.yaml", + }; + SimMock* mock = *state = simmock_alloc(inst_names, ARRAY_SIZE(inst_names)); + simmock_configure(mock, argv, ARRAY_SIZE(argv), ARRAY_SIZE(inst_names)); + ModelMock* model = simmock_find_model(mock, EXTENDED_INST_NAME); + simmock_load(mock); + simmock_load_model_check(model, true, true, false); + simmock_setup(mock, "data_channel", NULL); + + /* Initial value. */ + double counter = 42.0; + double odd = false; + double even = true; + simmock_print_scalar_signals(mock, LOG_DEBUG); + /* T0 ... Tn */ + for (uint32_t i = 0; i < 5; i++) { + /* Do the check. */ + SignalCheck checks[] = { + { .index = EXTENDED_SIGNAL_COUNTER, .value = counter }, + { .index = EXTENDED_SIGNAL_ODD, .value = odd }, + { .index = EXTENDED_SIGNAL_EVEN, .value = even }, + }; + simmock_signal_check( + mock, EXTENDED_INST_NAME, checks, ARRAY_SIZE(checks), NULL); + /* Step the model. */ + assert_int_equal(simmock_step(mock, true), 0); + simmock_print_scalar_signals(mock, LOG_DEBUG); + /* Set check conditions (for next step). */ + counter += 1.0; + odd = (bool)odd ? false : true; + even = (bool)even ? false : true; + } +} + + +#define BINARY_INST_NAME "binary_inst" +#define BINARY_SIGNAL_COUNTER 0 +#define BINARY_SIGNAL_MESSAGE 0 + +void test_model__binary(void** state) +{ + chdir("../../../../dse/modelc/build/_out/examples/binary"); + + const char* inst_names[] = { + BINARY_INST_NAME, + }; + char* argv[] = { + (char*)"test_model_interface", + (char*)"--name=" BINARY_INST_NAME, + (char*)"--logger=5", // 1=debug, 5=QUIET (commit with 5!) + (char*)"data/simulation.yaml", + (char*)"data/model.yaml", + }; + SimMock* mock = *state = simmock_alloc(inst_names, ARRAY_SIZE(inst_names)); + simmock_configure(mock, argv, ARRAY_SIZE(argv), ARRAY_SIZE(inst_names)); + ModelMock* model = simmock_find_model(mock, BINARY_INST_NAME); + simmock_load(mock); + simmock_load_model_check(model, true, true, true); + simmock_setup(mock, "scalar_channel", "binary_channel"); + + /* Initial value. */ + double counter = 42.0; + char buffer[100] = ""; + uint32_t len = 0; + + simmock_print_scalar_signals(mock, LOG_DEBUG); + simmock_print_binary_signals(mock, LOG_DEBUG); + /* T0 ... Tn */ + for (uint32_t i = 0; i < 2; i++) { + /* Do the check. */ + SignalCheck checks[] = { + { .index = BINARY_SIGNAL_COUNTER, .value = counter }, + }; + BinaryCheck b_checks[] = { + { .index = BINARY_SIGNAL_MESSAGE, + .buffer = (uint8_t*)buffer, + .len = len }, + }; + simmock_signal_check( + mock, BINARY_INST_NAME, checks, ARRAY_SIZE(checks), NULL); + simmock_binary_check( + mock, BINARY_INST_NAME, b_checks, ARRAY_SIZE(b_checks), NULL); + /* Step the model. */ + assert_int_equal(simmock_step(mock, true), 0); + simmock_print_scalar_signals(mock, LOG_DEBUG); + simmock_print_binary_signals(mock, LOG_DEBUG); + /* Set check conditions (for next step). */ + counter += 1.0; + snprintf(buffer, sizeof(buffer), "count is %d", (int)counter); + len = strlen(buffer) + 1; + } +} + + +int run_model_interface_tests(void) +{ + void* s = test_setup; + void* t = test_teardown; + + getcwd(__entry_path__, 200); + + const struct CMUnitTest tests[] = { + cmocka_unit_test_setup_teardown(test_model__minimal, s, t), + cmocka_unit_test_setup_teardown(test_model__extended, s, t), + cmocka_unit_test_setup_teardown(test_model__binary, s, t), + }; + + return cmocka_run_group_tests_name("MODEL / INTERFACE", tests, NULL, NULL); +} diff --git a/tests/cmocka/model/stub_controller.c b/tests/cmocka/model/stub_controller.c index 2bd8b77..145d552 100644 --- a/tests/cmocka/model/stub_controller.c +++ b/tests/cmocka/model/stub_controller.c @@ -77,17 +77,13 @@ void controller_exit(SimulationSpec* sim) { ModelInstanceSpec* _instptr = sim->instance_list; while (_instptr && _instptr->name) { - ModelInstancePrivate* mip = _instptr->private; - ControllerModel* cm = mip->controller_model; - if (cm->model_exit_func == NULL) goto exit_next; + if (_instptr->model_desc->vtable.destroy == NULL) goto exit_next; - log_notice("Call symbol: %s ...", MODEL_EXIT_FUNC_STR); + log_notice("Call symbol: %s ...", MODEL_DESTROY_FUNC_NAME); errno = 0; - int rc = cm->model_exit_func(_instptr); - if (rc) { - if (errno == 0) errno = rc; - log_error("model_exit_func() failed"); - } + _instptr->model_desc->vtable.destroy(_instptr->model_desc); + if (errno) log_error(MODEL_DESTROY_FUNC_NAME "() failed"); + exit_next: /* Next instance? */ _instptr++; diff --git a/tests/cmocka/model/test_ncodec.c b/tests/cmocka/model/test_ncodec.c index f772fde..8fb4d3c 100644 --- a/tests/cmocka/model/test_ncodec.c +++ b/tests/cmocka/model/test_ncodec.c @@ -2,16 +2,12 @@ // // SPDX-License-Identifier: Apache-2.0 -#include -#include -#include -#include -#include -#include -#include +#include #include #include +#include #include +#include #include @@ -39,8 +35,9 @@ static uint _sv_count(SignalVector* sv) } -static int _sv_nop(double* model_time, double stop_time) +static int _sv_nop(ModelDesc* model, double* model_time, double stop_time) { + UNUSED(model); UNUSED(model_time); UNUSED(stop_time); return 0; @@ -67,15 +64,8 @@ static int test_setup(void** state) assert_int_equal(rc, 0); mock->mi = modelc_get_model_instance(&mock->sim, args.name); assert_non_null(mock->mi); - rc = model_function_register(mock->mi, "NOP", 0.005, _sv_nop); - assert_int_equal(rc, 0); - - /* Binary channel. */ - static ModelChannelDesc binary_channel_desc = { - .name = "binary_vector", - .function_name = "NOP", - }; - rc = model_configure_channel(mock->mi, &binary_channel_desc); + ModelVTable vtable = { .step = _sv_nop }; + rc = modelc_model_create(&mock->sim, mock->mi, &vtable); assert_int_equal(rc, 0); /* Return the mock. */ @@ -104,7 +94,7 @@ void test_ncodec__ncodec_open(void** state) { ModelCMock* mock = *state; - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 1); /* Use the "binary" signal vector. */ @@ -123,8 +113,6 @@ void test_ncodec__ncodec_open(void** state) assert_null(sv->ncodec[1]); assert_non_null(sv->ncodec[2]); assert_ptr_equal(sv->codec(sv, 2), sv->ncodec[2]); - - model_sv_destroy(sv_save); } @@ -135,7 +123,7 @@ void test_ncodec__read_empty(void** state) NCodecMessage msg; /* Use the "binary" signal vector. */ - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 1); SignalVector* sv = sv_save; while (sv && sv->name) { @@ -168,8 +156,6 @@ void test_ncodec__read_empty(void** state) assert_int_equal(len, -ENOMSG); assert_int_equal(msg.len, 0); assert_null(msg.buffer); - - model_sv_destroy(sv_save); } @@ -179,7 +165,7 @@ void test_ncodec__network_stream(void** state) int rc; /* Use the "binary" signal vector. */ - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 1); SignalVector* sv = sv_save; while (sv && sv->name) { @@ -223,8 +209,6 @@ void test_ncodec__network_stream(void** state) assert_int_equal(msg.len, strlen(greeting)); assert_non_null(msg.buffer); assert_memory_equal(msg.buffer, greeting, strlen(greeting)); - - model_sv_destroy(sv_save); } @@ -233,7 +217,7 @@ void test_ncodec__config(void** state) ModelCMock* mock = *state; /* Use the "binary" signal vector. */ - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 1); SignalVector* sv = sv_save; while (sv && sv->name) { @@ -259,8 +243,6 @@ void test_ncodec__config(void** state) assert_int_equal(len, 0x66); assert_int_equal(len, sv->length[2]); assert_int_equal(((uint8_t*)(sv->binary[2]))[54], 8); - - model_sv_destroy(sv_save); } diff --git a/tests/cmocka/model/test_signal.c b/tests/cmocka/model/test_signal.c index 78e7b7a..57d3824 100644 --- a/tests/cmocka/model/test_signal.c +++ b/tests/cmocka/model/test_signal.c @@ -2,16 +2,12 @@ // // SPDX-License-Identifier: Apache-2.0 -#include -#include -#include -#include -#include -#include -#include +#include #include #include +#include #include +#include #define UNUSED(x) ((void)x) @@ -38,8 +34,9 @@ static uint _sv_count(SignalVector* sv) } -static int _sv_nop(double* model_time, double stop_time) +static int _sv_nop(ModelDesc* model, double* model_time, double stop_time) { + UNUSED(model); UNUSED(model_time); UNUSED(stop_time); return 0; @@ -66,23 +63,8 @@ static int test_setup(void** state) assert_int_equal(rc, 0); mock->mi = modelc_get_model_instance(&mock->sim, args.name); assert_non_null(mock->mi); - rc = model_function_register(mock->mi, "NOP", 0.005, _sv_nop); - assert_int_equal(rc, 0); - - /* Scalar channel. */ - static ModelChannelDesc scalar_channel_desc = { - .name = "scalar_vector", - .function_name = "NOP", - }; - rc = model_configure_channel(mock->mi, &scalar_channel_desc); - assert_int_equal(rc, 0); - - /* Binary channel. */ - static ModelChannelDesc binary_channel_desc = { - .name = "binary_vector", - .function_name = "NOP", - }; - rc = model_configure_channel(mock->mi, &binary_channel_desc); + ModelVTable vtable = { .step = _sv_nop }; + rc = modelc_model_create(&mock->sim, mock->mi, &vtable); assert_int_equal(rc, 0); /* Return the mock. */ @@ -111,7 +93,7 @@ void test_signal__scalar(void** state) { ModelCMock* mock = *state; - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 2); /* Use the "scalar" signal vector. */ @@ -125,7 +107,7 @@ void test_signal__scalar(void** state) /* General properties. */ assert_string_equal(sv->name, "scalar"); assert_string_equal(sv->alias, "scalar_vector"); - assert_string_equal(sv->function_name, "NOP"); + assert_string_equal(sv->function_name, "model_step"); assert_int_equal(sv->count, 2); assert_int_equal(sv->is_binary, false); assert_non_null(sv->signal); @@ -151,8 +133,6 @@ void test_signal__scalar(void** state) sv->append(sv, i, (void*)"1234", 4); /* NOP */ assert_double_equal(sv->scalar[i], test_val, DBL_EPSILON); } - - model_sv_destroy(sv_save); } @@ -160,7 +140,7 @@ void test_signal__binary(void** state) { ModelCMock* mock = *state; - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 2); /* Use the "binary" signal vector. */ @@ -174,7 +154,7 @@ void test_signal__binary(void** state) /* General properties. */ assert_string_equal(sv->name, "binary"); assert_string_equal(sv->alias, "binary_vector"); - assert_string_equal(sv->function_name, "NOP"); + assert_string_equal(sv->function_name, "model_step"); assert_int_equal(sv->count, 2); assert_int_equal(sv->is_binary, true); assert_non_null(sv->signal); @@ -230,8 +210,6 @@ void test_signal__binary(void** state) /* Cleanup. */ free(test_val); } - - model_sv_destroy(sv_save); } @@ -239,7 +217,7 @@ void test_signal__annotations(void** state) { ModelCMock* mock = *state; - SignalVector* sv_save = model_sv_create(mock->mi); + SignalVector* sv_save = mock->mi->model_desc->sv; assert_int_equal(_sv_count(sv_save), 2); /* Use the "binary" signal vector. */ @@ -262,8 +240,6 @@ void test_signal__annotations(void** state) assert_string_equal(value, "binary_bar"); value = sv->annotation(sv, 1, "mime_type"); assert_string_equal(value, "application/custom-stream"); - - model_sv_destroy(sv_save); } @@ -278,5 +254,5 @@ int run_signal_tests(void) cmocka_unit_test_setup_teardown(test_signal__annotations, s, t), }; - return cmocka_run_group_tests_name("NCODEC", tests, NULL, NULL); + return cmocka_run_group_tests_name("SIGNAL", tests, NULL, NULL); } diff --git a/tests/pytest/modelc/test_modelc.py b/tests/pytest/modelc/test_modelc.py index c773c76..8e6fe98 100644 --- a/tests/pytest/modelc/test_modelc.py +++ b/tests/pytest/modelc/test_modelc.py @@ -22,8 +22,7 @@ # Sandbox for the Model (common only, specific in test cases). MODEL_YAML = 'data/model.yaml' -SIGNAL_GROUP_YAML = 'data/signal_group.yaml' -STACK_YAML = 'data/stack.yaml' +STACK_YAML = 'data/simulation.yaml' async def run(dir, cmd): @@ -81,21 +80,21 @@ async def main(params, checks): def test_modelc_double(): params = { - 'MODEL_SANDBOX_DIR': os.getenv('MODELC_SANDBOX_DIR')+'/examples/dynamic', + 'MODEL_SANDBOX_DIR': os.getenv('MODELC_SANDBOX_DIR')+'/examples/minimal', 'models': [ { - 'MODEL_INST': 'dynamic_model_instance', - 'MODEL_YAML_FILES': f'{MODEL_YAML} ' + f'{STACK_YAML} ' + f'{SIGNAL_GROUP_YAML}', + 'MODEL_INST': 'minimal_inst', + 'MODEL_YAML_FILES': f'{MODEL_YAML} ' + f'{STACK_YAML} ', }, ] } checks = [ - 'SignalValue: 2851307223 = 4.800000 [name=foo]', - 'SignalValue: 1991736602 = 16.800000 [name=bar]', + 'SignalValue: 2628574755 = 4.000000 [name=counter]', ] asyncio.run(main(params, checks)) +@pytest.mark.skip(reason="needs refactor") @pytest.mark.skipif(os.environ["PACKAGE_ARCH"] != "linux-amd64", reason="test only runs on linux-amd64") def test_modelc_binary(): # Two models write data at same cycle (ModelReady). SimBus should append @@ -106,11 +105,11 @@ def test_modelc_binary(): 'models': [ { 'MODEL_INST': 'binary_model_instance', - 'MODEL_YAML_FILES' : f'{MODEL_YAML} ' + f'{STACK_YAML} ' + f'{SIGNAL_GROUP_YAML}', + 'MODEL_YAML_FILES' : f'{MODEL_YAML} ' + f'{STACK_YAML} ', }, { 'MODEL_INST': 'second_binary_model_instance', - 'MODEL_YAML_FILES' : f'{MODEL_YAML} ' + f'{STACK_YAML} ' + f'{SIGNAL_GROUP_YAML}', + 'MODEL_YAML_FILES' : f'{MODEL_YAML} ' + f'{STACK_YAML} ', }, ] } diff --git a/tests/pytest/modelc/test_mstep.py b/tests/pytest/modelc/test_mstep.py index 4ac1c47..f1774ee 100644 --- a/tests/pytest/modelc/test_mstep.py +++ b/tests/pytest/modelc/test_mstep.py @@ -19,13 +19,12 @@ MODELC_SANDBOX_DIR = os.getenv('MODELC_SANDBOX_DIR') MSTEP_EXE = MODELC_SANDBOX_DIR+'/bin/mstep' -# Sandbox for the Dynamic Model (test basis). -MODEL_SANDBOX_DIR = os.getenv('MODELC_SANDBOX_DIR')+'/examples/dynamic' -DYNAMIC_MODEL_INST = 'dynamic_model_instance' +# Sandbox for the Minimal Model (test basis). +MODEL_SANDBOX_DIR = os.getenv('MODELC_SANDBOX_DIR')+'/examples/minimal' +DYNAMIC_MODEL_INST = 'minimal_inst' DYNAMIC_MODEL_YAML = 'data/model.yaml' -DYNAMIC_MODEL_LIB = 'lib/dynamic_model.so' # Not used, see model.yaml. -SIGNAL_GROUP_YAML = 'data/signal_group.yaml' -STACK_YAML = 'data/stack.yaml' +DYNAMIC_MODEL_LIB = 'lib/libminimal.so' # Not used, see model.yaml. +STACK_YAML = 'data/simulation.yaml' async def run(dir, cmd): @@ -46,7 +45,7 @@ async def main(): result_list = await asyncio.gather( asyncio.wait_for( run( - MODEL_SANDBOX_DIR, f'{MSTEP_EXE} --logger 2 --name {DYNAMIC_MODEL_INST} ' + f'{DYNAMIC_MODEL_YAML} ' + f'{STACK_YAML} ' + f'{SIGNAL_GROUP_YAML} ' + ''), timeout=TIMEOUT), + MODEL_SANDBOX_DIR, f'{MSTEP_EXE} --logger 2 --name {DYNAMIC_MODEL_INST} ' + f'{DYNAMIC_MODEL_YAML} ' + f'{STACK_YAML} ' + ''), timeout=TIMEOUT), ) for result in result_list: print('************************************************************') @@ -64,11 +63,9 @@ async def main(): for result in result_list: assert result['rc'] == 0 - assert 'value: 12' in result['stdout'] - assert 'value: 42' in result['stdout'] + assert 'value: 10' in result['stdout'] assert 'Starting Simulation (for 10 steps) ...' in result['stdout'] - assert 'Model Function: example' in result['stdout'] - assert 'Model function model_exit called' in result['stdout'] + assert 'Model Function: model_step' in result['stdout'] if __name__ == '__main__': diff --git a/tests/pytest/modelc/test_stacked_modelc.py b/tests/pytest/modelc/test_stacked_modelc.py index aad8461..b0e0bae 100644 --- a/tests/pytest/modelc/test_stacked_modelc.py +++ b/tests/pytest/modelc/test_stacked_modelc.py @@ -77,6 +77,7 @@ async def main(params, checks): assert found, check +@pytest.mark.skip(reason="Needs refactor") def test_modelc_stack(): # Two models executed by the same instance of ModelC. params = {