How to get multiple (reference) models using the same c-code (RAM instance) variables during simulation.

3 次查看(过去 30 天)
I am struggling to setup a model that uses a reference model such that when they call c-code, they are accessing the same RAM (variable instances). The hierarchy is as follows:
In both "main_model" and "function_model" the Simulation Target is set to use "interface.[ch]", which very simply contains an array of 4 structures to allow both global and accessor functions for access. The header is as follows:
typedef struct
{
bool enabled;
uint32_t value;
} counter_t;
counter_t counter_list[4];
#define COUNTER_0 ((counter_t *)&(counter_list[0]))
#define COUNTER_1 ((counter_t *)&(counter_list[1]))
#define COUNTER_2 ((counter_t *)&(counter_list[2]))
#define COUNTER_3 ((counter_t *)&(counter_list[3]))
void initialize(void);
void counter_enable(counter_t *ptr);
void counter_disable(counter_t *ptr);
uint32_t counter_get_current(counter_t *ptr);
void counter_reset(counter_t *ptr);
I do understand in regards to C-code, choose either the global access or accessor functions, however, this scenerio is setup to mimic that of a hardware and firmware relationship special function register (SFR) access to something like an input compare using vendor supplied packages (such as NXP or STM) and maintain the call footprint for code generation.
The problem is when this setup is run, the "main_model" and the "function_model" each have their own internal values for "counter_list" rather than sharing a single instance of them. The "main_model" will initialize the counter value to 100 and the "function_model" increments by 1 per function call (occuring every 0.1 seconds). Looking at the values during simulation from both the "main_model" and "function_model" the following is observed:
The expectation would be that they both have the same value, starting at 100 and incrementing up to 200 over the 10 seconds. Is this possible? If so, how what am I missing? or help point me to the proper documentation (I haven't been able to find this in the user guide... yet). I've also attached the project and all files (2019b). Appreciate any guidance, thanks!
  7 个评论
James Tursa
James Tursa 2020-6-8
编辑:James Tursa 2020-6-8
So main_model and function_model are two different s-functions? That is, you have two different sets of source code that include interface.h, and these two different sets of source code inlude interface.h? If that is the case, then yes the s-functions do not share anything. You would have to set up communication between them at the model level (e.g., have main_model feed function_model the starting value).
Jeremy Quandt
Jeremy Quandt 2020-6-8
编辑:Jeremy Quandt 2020-6-8
Also, worth noting, the chart Action Language is set to C. Further, if I modify the model by moving the contents of the reference model into the main model and execute the results are as follows,
(no longer a reference)

请先登录,再进行评论。

采纳的回答

Jeremy Quandt
Jeremy Quandt 2020-6-9
编辑:Jeremy Quandt 2020-6-9
I was able to get this working. It wasn't explicitly in the reference manual, but I was able to draw the resolve from some lines of an example. In a nutshell, all the variables are moved internal to the code module (i.e. into the source) and declared statically. The source module then becomes as follows:
interface.h
#ifdef __cplusplus
extern "C"
{
#endif
#ifndef INTERFACE_H
#define INTERFACE_H
#include <stdint.h>
#include <stdbool.h>
typedef struct
{
bool enabled;
uint32_t value;
} counter_t;
#define COUNTER_0 counter_object(0)
#define COUNTER_1 counter_object(1)
#define COUNTER_2 counter_object(2)
#define COUNTER_3 counter_object(3)
extern void initialize(void);
extern void counter_enable(counter_t *ptr);
extern void counter_disable(counter_t *ptr);
extern uint32_t counter_get_current(counter_t *ptr);
extern void counter_reset(counter_t *ptr);
extern counter_t *counter_object(uint8_t id);
#endif // INTERFACE_H
#ifdef __cplusplus
}
#endif
interface.c
#include <string.h>
#include "interface.h"
static counter_t counter_list[4];
void initialize(void)
{
memset(&counter_list, 0, sizeof(counter_list));
}
void counter_enable(counter_t *ptr)
{
if (NULL != ptr)
{
ptr->enabled = true;
}
}
void counter_disable(counter_t *ptr)
{
if (NULL != ptr)
{
ptr->enabled = false;
}
}
uint32_t counter_get_current(counter_t *ptr)
{
if (NULL != ptr)
{
return (ptr->value);
}
else
{
return (0);
}
}
void counter_reset(counter_t *ptr)
{
if (NULL != ptr)
{
ptr->value = 0;
}
}
counter_t *counter_object(uint8_t id)
{
if (id < 4)
{
return (&(counter_list[id]));
}
else
{
return (NULL);
}
}
All the syntax in the state charts then operate identical.

更多回答(0 个)

类别

Help CenterFile Exchange 中查找有关 Simulink Functions 的更多信息

产品


版本

R2019b

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!

Translated by