Mex programming for a beginner

8 views (last 30 days)
Mandeguz
Mandeguz on 21 Jun 2017
Commented: Mandeguz on 10 Jul 2017
I'm having trouble learning how to use MEX files, and all of the documentation (even this one: https://www.mathworks.com/help/matlab/matlab_external/standalone-example.html#zmw57dd0e18781, which seemed simple at first but completely lost me after a couple paragraphs) is going WAY over my head.
Right now, I'm trying to use code from a .dll and I don't have the original source code, but I do have a header file with function prototypes of all of the functions that I need, all within extern "C".
So instead of changing these functions or really doing anything all that complicated, I'm just trying to use the .dll functions in my MATLAB code. Here's an example of one of my functions (I have twenty):
#include "mex.h"
#include "AllFunctions.h"
void mexFunction(int nlhs, mxArray *plhs[],
int nrhs, const mxArray *prhs[]){
if(nrhs != 0) {
mexErrMsgIdAndTxt("Error: myfun.c",
"Cannot have any inputs.");
}
if(nlhs != 1) {
mexErrMsgIdAndTxt("Error: myfun.c",
"One output required.");
}
double *output; //output
/* create the output matrix */
plhs[0] = mxCreateDoubleMatrix(1,1,mxREAL);
/* get a pointer to the real data in the output matrix */
output = mxGetPr(plhs[0]);
/* call the computational routine */
myfun;
}
It has no inputs and one output. Most of my functions have a few simple inputs and one output, and a few functions pass pointers. Loadlibrary completely fails for me, and the solutions given on multiple threads do not pertain to me, so I'm trying to use MEX's instead.
I'd like to know if this is the proper approach and maybe what I should include to make this actually work before I make twenty functions that don't work. They all work in tandem with each other, so it is very difficult to test one function and then move on from there. Thanks for helping me out.
PS: Please don't tell me I need to go study MEXs more, as I am doing so as you type up your response. If there is another thing I should be looking at, I'll be very grateful. Any help is appreciated, but please don't tell me to spend my time researching (unless its something specific) before I ask beginner questions because I did a lot of research before posting this question, and I am continuing to do so afterwards.
  5 Comments
Mandeguz
Mandeguz on 22 Jun 2017
"Are you somehow expecting that myfun call in your mex routine to call a function by that name in your dll just because you included the header file? (It won't)." - James Tursa
How do you call a specific function in the dll then, if not my name? Doesn't it work that way in C?
"When you are compiling your mex routine, are you trying to link in that dll?"
Not sure what you mean. How do you link a dll to your mex routine? Do you have to include it like the header file?
Hopefully, this will clarify things. Thanks!
Mandeguz
Mandeguz on 5 Jul 2017
Alright, after checking in on some of the responses and researching the material as well as I could, I'm left with the following error:
Error using mex
C:\Users\guzmanjj\AppData\Local\Temp\mex_510323448540256_3152\myfun.obj:myfun.cpp:(.text+0xf2):
undefined reference to `__imp_myfun'
collect2.exe: error: ld returned 1 exit status
whenever I run this command:
mex -setup C++;
mex myfun.cpp
I have the correct C/C++ compiler and permissions to use it. The correct path is set to grab this function. I thought my code was written well based off of MATLAB's own examples. Here is my C++ file:
#include "mex.h"
#include "myheader.h"
void mexFunction(int nlhs, mxArray *plhs[],
int nrhs, const mxArray *prhs[]){
/* check for proper number of arguments */
if(nrhs != 2)
mexErrMsgIdAndTxt("Error: myfun.cpp","Two inputs required.");
if(nlhs != 1)
mexErrMsgIdAndTxt("Error: myfun.cpp","One output required.");
/* variables */
double in_one; //input 1
double in_two; //input 2
double out; //output
/* make sure that both input arguments are type double */
if( !mxIsDouble(prhs[0]) || mxIsComplex(prhs[0]) || mxGetNumberOfElements(prhs[0])!=1 )
mexErrMsgIdAndTxt("Error: myfun.cpp","Input 1 must be type double.");
if( !mxIsDouble(prhs[1]) || mxIsComplex(prhs[1]) || mxGetNumberOfElements(prhs[1])!=1 )
mexErrMsgIdAndTxt("Error: myfun.cpp","Input 2 must be type double.");
/* get the values of the scalar inputs */
in_one = mxGetScalar(prhs[0]);
in_two = mxGetScalar(prhs[1]);
/* create the output scalar */
out = mxGetScalar(plhs[0]);
/* call the computational routine */
out = myfun(in_one,in_two);
}
And here's my header file:
#pragma once
#include <windows.h>
#include "WinDef.h" //include BOOL definition
#ifdef MYFUNPACKAGE_EXPORTS
#define MYFUN_PACKAGE_API __declspec(dllexport)
#else
#define MYFUN_PACKAGE_API __declspec(dllimport)
#endif
#ifdef __cplusplus
extern "C" {
#endif
extern MY_DLL_API BOOL myfun(unsigned int in_one, unsigned int in_two);
#ifdef __cplusplus
}
#endif
My dll file cannot be read. I cannot get the source code unless I decompile the dll, which I would like help with if that is absoluetly necessary; I'd rather avoid that if I could.
I believe the problem is that I do not know how to directly connect my dll to my cpp/mex file. I'm looking more into loadlibrary currently, but I still have not had any success. If anyone could clear up this mex problem, that would be very helpful. Thanks

Sign in to comment.

Accepted Answer

James Tursa
James Tursa on 5 Jul 2017
Edited: James Tursa on 5 Jul 2017
For the mex routine approach, assuming you can link with the library properly per Philip's comments, I have the following comments:
if(nlhs != 1)
mexErrMsgIdAndTxt("Error: myfun.cpp","One output required.");
The above line requires the user to specify an output variable when calling your code. This is usually too strict a requirement to place on a user. A more friendly approach is to require "at most 1" output" so that the result can simply go into ans. Note that even if nlhs==0, MATLAB always has room for at least one output in the plhs array. So this is advised:
if(nlhs > 1)
mexErrMsgIdAndTxt("Error: myfun.cpp","At most one output required.");
Then there are these lines:
/* create the output scalar */
out = mxGetScalar(plhs[0]);
/* call the computational routine */
out = myfun(in_one,in_two);
The above lines will crash MATLAB. The plhs[0] variable does not exist when you enter the mex routine ... that is something that you, the programmer, must create before you can use it. Thus the mxGetScalar(plhs[0]) call accesses invalid memory and crashes MATLAB. And even if you had created plhs[0] previously, the above lines of code would not get the next out result from myfun into plhs[0]. So these lines should be coded this way instead:
/* call the computational routine */
out = myfun(in_one,in_two);
/* create the output scalar */
plhs[0] = mxCreateDoubleScalar(out);
  8 Comments
Mandeguz
Mandeguz on 7 Jul 2017
I forgot something in the header file, it really looks like this:
#pragma once
#include <windows.h>
#include "WinDef.h" //include BOOL definition
#ifdef MYFUNPACKAGE_EXPORTS
#define MYFUN_PACKAGE_API __declspec(dllexport)
#else
#define MYFUN_PACKAGE_API __declspec(dllimport)
#endif
#ifdef __cplusplus
extern "C" {
#endif
extern MY_DLL_API BOOL myfun(unsigned int in_one, unsigned int & in_two);
#ifdef __cplusplus
}
#endif
I believe that's a C++ reference (not a pointer but similar to it) but if I'm wrong, then I need to figure out why that's there and what to do about it in the mex (I did not write this header file, I only added the #ifdef __cplusplus lines).
James Tursa
James Tursa on 7 Jul 2017
The prototype for the in_two argument using & means that yes the argument is being passed by reference, and any changes made to in_two inside of myfun will be made to the original variable.

Sign in to comment.

More Answers (1)

Philip Borghesani
Philip Borghesani on 5 Jul 2017
Edited: Philip Borghesani on 5 Jul 2017
I suggest a two pronged approach.
  1. Get Loadlibrary working if at all possible. There is only one real error in your loadlibrary output (the last line) the rest are warnings caused by parsing windows.h and can safely be ignored. The error is the last line and is due to a missing #ifdef __cplusplus I think you fixed it in the header you are using for myfun for mex. Using loadlibrary will save you from writing a large number of simple mex files or a large mex file with a gateway.
  2. You may find that some functions are better called from a mex file. As alluded to by James Tursa, to call a dll from a mex file you must link with its corresponding library. The mex command gives information on how to link with an extra lib file. On Linux this is the same shared library that loadlibrary would use but on Windows this must be a .lib file. If you have mex configured for mingGW you may need to generate a lib file for that compiler. You were probably supplied with a lib file for Microsoft compilers.
I suggest new questions that are specific to current problems you are having with loadlibrary or mex. Solving multiple problems in a single answer causes confusion... Learning how a function is called from mex (or c) helps quite a bit in how to structure code that calls that function with loadlibrary

Categories

Find more on Write C Functions Callable from MATLAB (MEX Files) in Help Center and File Exchange

Tags

Community Treasure Hunt

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

Start Hunting!