How to convert a matlab::me​x::Argumen​tList to a custom C++-Type and vice versa

9 Ansichten (letzte 30 Tage)
I have some C++ source code that is accessable in Simulink with the help of the Legacy Code Tool. I want the same C++ code also be callable from Matlab. My research led me to the "C++ MEX API" with which I am struggeling. If there is another simpler approach, please let me know.
My problem is that I need to wrap the arguments of the operator method
void operator()(matlab::mex::ArgumentList outputs, matlab::mex::ArgumentList inputs)
into some user-defined C++ structs such that I can call my custom code
void myCode(userStruct1 &in, userStruct2 &out)
For simple inputs of operator() I managed to loop through "inputs" and the containing matlab::data::Array. I then assigned these values into my userStruct1 object. I find this procedure tedious but I can live with that. If there's an easier way to convert from matlab::data::ArgumentList to userStruct1 please let me know.
My problem, however, is the other way around. Assume myCode is called and the results are written in the userStruct2 object. How do I pass these results to "outputs"?
  1 Kommentar
Friedrich
Friedrich am 24 Okt. 2024
Bearbeitet: Friedrich am 24 Okt. 2024
Let's assume userStruct2 is:
struct userStruct2 {
std::vector<double> a;
std::vector<double> b;
double c;
bool d;
char e[10];
userStruct2() : a(3), b(5) {}
};

Melden Sie sich an, um zu kommentieren.

Akzeptierte Antwort

埃博拉酱
埃博拉酱 am 25 Okt. 2024 um 15:43
Bearbeitet: 埃博拉酱 am 25 Okt. 2024 um 15:56
Obviously we can't think of any simpler way to convert to userStruct1 unless you tell us how you defined it.
For userStruct2, your code is risky because MATLAB chars are UTF16 encoded while C++ chars are usually UTF8 encoded. MATLAB does not do encoding conversions, so if your string contains non-ASCII characters it will be garbled. On Windows you can consider:
#include <Windows.h>
import std;
using namespace matlab::data;
ArrayFactory factory;
template<typename T>
TypedArray<T> VectorToArray(const std::vector<T>& vec)
{
TypedArray<T> arr = factory.createArray<T>({ vec.size() });
std::copy(vec.cbegin(), vec.cend(), arr.begin());
return arr;
}
CharArray CppCharToMatlab(const char* Input, size_t Length)
{
//This function does encoding conversion. You can implement it with any other libraries if you don't have Windows SDK.
buffer_ptr_t<char16_t> Buffer = factory.createBuffer<char16_t>(Length + 1);
Length = MultiByteToWideChar(CP_UTF8, 0, Input, Length, (wchar_t*)Buffer.get(), Length + 1);
return factory.createArrayFromBuffer({ 1,Length }, std::move(Buffer));
}
void Example(matlab::mex::ArgumentList outputs, const userStruct2& input)
{
StructArray out = factory.createStructArray({ 1 }, { "a", "b", "c", "d", "e" });
out[0]["a"] = VectorToArray(input.a);
out[0]["b"] = VectorToArray(input.b);
out[0]["c"] = factory.createScalar(input.c);
out[0]["d"] = factory.createScalar(input.d);
out[0]["e"] = CppCharToMatlab(input.e, std::extent_v<decltype(input.e)>);
outputs[0] = std::move(out);
}
  1 Kommentar
Walter Roberson
Walter Roberson am 25 Okt. 2024 um 18:59
your code is risky because MATLAB chars are UTF16 encoded while C++ chars are usually UTF8 encoded.
The discussion is further complicated because Simulink does not support char signals, but does support what it calls string signals. string signals are explicitly said to be ISO/IEC 8859-1 (that is, 8 bit with no surrogate code points.)

Melden Sie sich an, um zu kommentieren.

Weitere Antworten (1)

Friedrich
Friedrich am 24 Okt. 2024
Nevermind, I found the solution right after I posted my question. For those who have the same problem, here is an example
using namespace matlab::data;
class MexFunction : public matlab::mex::Function {
public:
void operator()(matlab::mex::ArgumentList outputs, matlab::mex::ArgumentList inputs) {
ArrayFactory factory;
StructArray out = factory.createStructArray({ 1 }, { "a", "b", "c", "d", "e"});
out[0]["a"] = factory.createArray<double>({ 3, 1 }, { 1, 2, 3 });
out[0]["b"] = factory.createArray<double>({ 5, 1 }, { 1, 2, 3, 4, 5 });
out[0]["c"] = factory.createArray<double>({ 1, 1 }, { 0 });
out[0]["d"] = factory.createArray<bool>({ 1, 1 }, { true });
out[0]["e"] = factory.createCharArray("0123456789");
outputs[0] = std::move(out);
}
}
The only thing I don't know yet is how to easily pass long arrays into the fields of "a" or "b".

Tags

Produkte


Version

R2021b

Community Treasure Hunt

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

Start Hunting!

Translated by