-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathstDetectMex.cpp
84 lines (73 loc) · 3.2 KB
/
stDetectMex.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
/*******************************************************************************
* Sketch Token Toolbox V0.95
* Copyright 2013 Joseph Lim [[email protected]]
* Please email me if you find bugs, or have suggestions or questions!
* Licensed under the Simplified BSD License [see bsd.txt]
*******************************************************************************/
#include "mex.h"
#include <math.h>
#include <omp.h>
typedef unsigned int uint32;
void mexFunction( int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[] )
{
// get inputs
float *chns = (float*) mxGetData(prhs[0]);
float *chnsSs = (float*) mxGetData(prhs[1]);
float *thrs = (float*) mxGetData(prhs[2]);
uint32 *fids = (uint32*) mxGetData(prhs[3]);
uint32 *child = (uint32*) mxGetData(prhs[4]);
float *distr = (float*) mxGetData(prhs[5]);
uint32 *cids1 = (uint32*) mxGetData(prhs[6]);
uint32 *cids2 = (uint32*) mxGetData(prhs[7]);
const int stride = (int) mxGetScalar(prhs[8]);
const int rad = (int) mxGetScalar(prhs[9]);
const int nChnFtrs = (int) mxGetScalar(prhs[10]);
// get dimensions and constants
const mwSize *chnsSize = mxGetDimensions(prhs[0]);
const int height = (int) chnsSize[0];
const int width = (int) chnsSize[1];
const mwSize *distrSize = mxGetDimensions(prhs[5]);
const int nTokens = (int) distrSize[0];
const int nTreeNodes = (int) distrSize[1];
const int nTrees = (int) distrSize[2];
const int heightOut = (int) ceil((height-rad*2.0)/stride);
const int widthOut = (int) ceil((width-rad*2.0)/stride);
// create output
const int outDims[3]={nTokens,heightOut,widthOut};
float *S = (float*) mxCalloc(nTokens*heightOut*widthOut,sizeof(float));
plhs[0] = mxCreateNumericMatrix(0,0,mxSINGLE_CLASS,mxREAL);
mxSetData(plhs[0],S);
mxSetDimensions(plhs[0],outDims,3);
// apply forest to each patch
#pragma omp parallel for
for( int c=0; c<widthOut; c++ ) {
for( int r=0; r<heightOut; r++ ) {
// classify a single patch using all trees
float *chns1 = chns + (r*stride) + (c*stride)*height;
float *chnsSs1 = chnsSs + (r*stride) + (c*stride)*height;
for( int t = 0; t < nTrees; t++ ) {
uint32 k = t*nTreeNodes, res;
while( child[k] ) {
// compute feature (either lookup in channel or self-similarity feature)
uint32 f = fids[k], cid1 = cids1[f], cid2 = cids2[f];
float ftr;
if( f<nChnFtrs )
ftr = chns1[cid1];
else
ftr = chnsSs1[cid1] - chnsSs1[cid2];
// compare ftr to threshold and move left or right accordingly
if( ftr < thrs[k] )
k = child[k]-1;
else
k = child[k];
res = k;
k += t*nTreeNodes;
}
// lookup probability and store results
for( int i = 0; i < nTokens; i++ ) {
S[r*nTokens + c*heightOut*nTokens + i] += distr[t*nTreeNodes*nTokens + res*nTokens + i];
}
}
}
}
}