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

34 visualizaciones (últimos 30 días)
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 comentario
Friedrich
Friedrich el 24 de Oct. de 2024 a las 10:44
Editada: Friedrich el 24 de Oct. de 2024 a las 11:42
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) {}
};

Iniciar sesión para comentar.

Respuesta aceptada

埃博拉酱
埃博拉酱 el 25 de Oct. de 2024 a las 15:43
Editada: 埃博拉酱 el 25 de Oct. de 2024 a las 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 comentario
Walter Roberson
Walter Roberson el 25 de Oct. de 2024 a las 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.)

Iniciar sesión para comentar.

Más respuestas (1)

Friedrich
Friedrich el 24 de Oct. de 2024 a las 11:41
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".

Etiquetas

Productos


Versión

R2021b

Community Treasure Hunt

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

Start Hunting!

Translated by