/* * $Id: spm_get_lm.c 6534 2015-08-24 16:02:56Z guillaume $ * Jesper Andersson */ /**************************************************************** ** ** Routine that identifies which voxels in a list of coordinates ** that are local maxima, and returns a list of indices into ** the coordinate list for those maxima. ** ***************************************************************/ #include #include #include #include "mex.h" /* Macros */ #ifndef MIN #define MIN(A,B) ((A) > (B) ? (B) : (A)) #endif #ifndef MAX #define MAX(A,B) ((A) > (B) ? (A) : (B)) #endif /* get_index */ mwSignedIndex get_index(mwIndex x, mwIndex y, mwIndex z, mwSize dim[3]) { if (x < 1 || x > dim[0] || y < 1 || y > dim[1] || z < 1 || z > dim[2]) { return(-1); } else { return((z-1)*dim[0]*dim[1]+(y-1)*dim[0]+x-1); } } /* is_maxima */ int is_maxima(double *v, mwSize dim[3], mwIndex x, mwIndex y, mwIndex z, unsigned int cc) { mwSignedIndex ii = 0, i = 0; double cv = 0.0; if ((ii=get_index(x,y,z,dim))<0) {return(0);} cv = v[ii]; if (cc >= 6) { if ((i=get_index(x+1,y,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y+1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y-1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y,z-1,dim))>0 && v[i] > cv) {return(0);} } if (cc >= 18) { if ((i=get_index(x+1,y+1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y-1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y+1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y-1,z,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y+1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y+1,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y-1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x,y-1,z-1,dim))>0 && v[i] > cv) {return(0);} } if (cc == 26) { if ((i=get_index(x+1,y+1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y+1,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y-1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x+1,y-1,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y+1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y+1,z-1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y-1,z+1,dim))>0 && v[i] > cv) {return(0);} if ((i=get_index(x-1,y-1,z-1,dim))>0 && v[i] > cv) {return(0);} } return(1); } /* get_maxima */ unsigned int get_maxima(double *vol, mwSize vdim[3], double *list, mwSize nlist, unsigned int cc, mwIndex **ldx) { mwIndex i = 0, j = 0; mwIndex ix = 0, iy = 0, iz = 0; mwSize ldx_sz = 0; mwIndex ldx_n = 0; *ldx = (mwIndex *) mxCalloc((ldx_sz = 1024),sizeof(mwIndex)); for (i=0, j=0; i= 0) { if (is_maxima(vol,vdim,ix,iy,iz,cc)) { if (ldx_n >= ldx_sz) { *ldx = (mwIndex *) mxRealloc(*ldx,(ldx_sz += 1024)*sizeof(mwIndex)); } (*ldx)[ldx_n] = i+1; ldx_n++; } } } return(ldx_n); } /* Gateway function */ void mexFunction(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[]) { mwIndex i = 0, j = 0, k = 0, m = 0, n = 0; int tmpint = 0; const mwSize *pdim = NULL; mwSize ndim = 0; mwSize vdim[3]; mwSize ln = 0, lm = 0; mwSize n_lindex = 0; mwIndex *lindex = NULL; unsigned int conn; double *vol = NULL; double *lp = NULL; double *list = NULL; double *plindex = NULL; if (nrhs < 1) mexErrMsgTxt("Not enough input arguments."); if (nrhs > 3) mexErrMsgTxt("Too many input arguments."); if (nlhs > 1) mexErrMsgTxt("Too many output arguments."); /* Get binary map. */ if (!mxIsNumeric(prhs[0]) || mxIsComplex(prhs[0]) || mxIsSparse(prhs[0]) || !mxIsDouble(prhs[0])) { mexErrMsgTxt("Input 'vol' must be numeric, real, full and double."); } ndim = mxGetNumberOfDimensions(prhs[0]); if (ndim != 3 && ndim != 2) { mexErrMsgTxt("Input 'vol' must 2- or 3-dimensional."); } pdim = mxGetDimensions(prhs[0]); vdim[0] = pdim[0]; vdim[1] = pdim[1]; if (ndim == 2) { vdim[2] = 1; } else { vdim[2] = pdim[2]; } vol = mxGetPr(prhs[0]); /* Get list of coordinates */ if (nrhs < 2) { lm = 3; ln = vdim[0] * vdim[1] * vdim[2]; list = (double *) mxCalloc(lm*ln,sizeof(double)); for (k=0,m=0,n=0; k