How to use mxCreateNumericArray

14 views (last 30 days)
I am building a mex function, but am having difficulty creating a vector of integers. My code throws about 10 errors, all of which around the declaration of 'gn' and plhs[0]. I'm just trying to make a xynum long vector of class int16. Why is it so difficult?
void findnegs(double *phi, int gnum, int xynum, int *gn, double *adj, double *newphi, double *dphidtSum)
{
long i,g;
double u,s;
for (i = 0; i<xynum; i++)
{
s = 0;
for (g = 0; g<gnum; g++)
{
u = dphidtSum[g+i*gnum]/phi[g+i*gnum];
if (newphi[g+i*gnum]<0 & u<s)
{
s = u;
gn[i] = g;
adj[i] = -1/u;
}
}
}
}
/* The gateway function that replaces the "main".
*plhs[] - the array of output values
*prhs[] - the array of input values
*/
void mexFunction( int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
{
/* DECLARATIONS*/
double *phi, *newphi, *dphidtSum, *adj;
int gnum, xynum, *gn;
/* INPUTS */
gnum = mxGetScalar(prhs[0]);
xynum = mxGetScalar(prhs[1]);
phi = mxGetPr(prhs[2]);
newphi = mxGetPr(prhs[3]);
dphidtSum = mxGetPr(prhs[4]);
/* OUTPUTS */
plhs[0] = mxCreateNumericArray(xynum,1,mxINT16_CLASS); /*Creates a matrix*/
plhs[1] = mxCreateDoubleMatrix(xynum,1,mxREAL); /*Creates a matrix*/
gn = mxGetInt16s(plhs[0]); /*Set to be the first of the output values*/
adj = mxGetPr(plhs[1]); /*Set to be the first of the output values*/
/* CALL ROUTINE */
findnegs(phi,gnum,xynum,gn,adj,newphi,dphidtSum);
}
  1 Comment
Jan
Jan on 13 May 2019
Edited: Jan on 13 May 2019
How do you compile the code? Which API version are you using? mxGetInt16s belongs to the R2018a API, but and mxGetPr to R12017b. Decide for one of them.
Which error message do you get? Please share them with the readers. This is more efficient thabn letting us guess, what you see.

Sign in to comment.

Accepted Answer

James Tursa
James Tursa on 13 May 2019
Edited: James Tursa on 14 May 2019
This line is incorrect (wrong function and signature):
plhs[0] = mxCreateNumericArray(xynum,1,mxINT16_CLASS); /*Creates a matrix*/
It should be this instead:
plhs[0] = mxCreateNumericMatrix(xynum,1,mxINT16_CLASS,mxREAL); /*Creates a matrix*/
Also, these lines will not work:
int gnum, xynum, *gn;
:
gn = mxGetInt16s(plhs[0]); /*Set to be the first of the output values*/
An int on your system is almost surely a 32-bit integer, not a 16-bit integer. You need to make sure your pointer definition matches exactly what is in memory. E.g., you can probably use a short int for this:
int gnum, xynum;
short *gn;
:
gn = mxGetInt16s(plhs[0]); /*Set to be the first of the output values*/
That will mean you need to change this signature:
void findnegs(double *phi, int gnum, int xynum, int *gn, double *adj, double *newphi, double *dphidtSum)
to this instead also:
void findnegs(double *phi, int gnum, int xynum, short *gn, double *adj, double *newphi, double *dphidtSum)
But this begs the question, are integer calculations in the findnegs function now going to overflow your 16-bit integers in gn? You may need to adjust your code to make sure this isn't a problem.
  6 Comments
Christopher Grose
Christopher Grose on 14 May 2019
I was looking at
edit([matlabroot '/extern/examples/refbook/matrixDivide.c']);
and noticed that the syntax for mxCreateNumericArray is not the same as mxCreateDoubleMatrix. I changed the code to:
void mexFunction( int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
{
/* DECLARATIONS*/
double *phi, *newphi, *dphidtSum, *adj;
int xynum;
short gnum, *gn;
mwSignedIndex dims[2];
/* INPUTS */
gnum = mxGetScalar(prhs[0]);
xynum = mxGetScalar(prhs[1]);
phi = mxGetPr(prhs[2]);
newphi = mxGetPr(prhs[3]);
dphidtSum = mxGetPr(prhs[4]);
/* OUTPUTS */
dims[0] = xynum;
dims[1] = 1;
plhs[0] = mxCreateNumericArray(2,dims,mxINT16_CLASS,mxREAL); /*Creates a matrix*/
plhs[1] = mxCreateDoubleMatrix(xynum,1,mxREAL); /*Creates a matrix*/
// gn = mxGetInt16s(plhs[0]); /*Set to be the first of the output values*/
gn = (short *) mxGetData(plhs[0]); /*Set to be the first of the output values*/
adj = mxGetPr(plhs[1]); /*Set to be the first of the output values*/
/* CALL ROUTINE */
findnegs(phi,gnum,xynum,gn,adj,newphi,dphidtSum);
}
which works.
But is there a way to do this without declaring and assigning each level of dims? Can I just put the values in the mxCreateNumericArray line to simplify/save some overhead?
James Tursa
James Tursa on 14 May 2019
Edited: James Tursa on 14 May 2019
You didn't read my initial Answer closely enough. I told you to replace your mxCreateNumericArray call with a mxCreateNumericMatrix call for exactly the reason you specify ... it avoids the dims stuff if you only want a 2D result.
The reason you were still crashing is because you had the wrong arguments for mxCreateNumericArray.

Sign in to comment.

More Answers (0)

Categories

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

Community Treasure Hunt

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

Start Hunting!