Mex файл дает ошибку сегментации только при втором использовании - PullRequest
1 голос
/ 20 апреля 2011

Следующий код успешно компилируется и возвращает правильные результаты при первом вызове. Во время повторного вызова я получаю ошибку ошибки сегментации.

//% function TF = InWindow(Date,WindowStartDates,WindowEndDates,EndHint)
//% INWINDOW returns true for window that contains Date. All inputs must be 
//% uint32 and WindowEndDates must be sorted.

//% EndHint is an optional input that specifies the row number to start
//% searching from.

#include "mex.h"
#include "matrix.h"
#include "math.h"

void CalculationRoutine(mxLogical *ismember, uint32_T *Date, uint32_T *WindowStartDates, uint32_T *WindowEndDates, unsigned int *StartIndex, int NumObs) {

mwIndex Counter;

// Find the first window that ends on or after the date. 
for (Counter = (mwIndex) *StartIndex; Counter < NumObs; Counter++) {  
    if (*Date <= *(WindowEndDates+Counter)) {
        break;
    }
}
*StartIndex = (unsigned int) Counter;

// Now flag every observation within the window. Remember that WindowStartDates
// is not necessarily sorted (but WindowEndDates is).
for (Counter = (mwIndex) *StartIndex; Counter < NumObs; Counter++) { 
    if (*Date >= *(WindowStartDates+Counter)) {
        *(ismember+Counter) = true;
    }
}

}

void mexFunction( int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[] ) {

    mxArray *ismember;
    unsigned int *StartIndex;

    //Input Checking.

    if (nrhs == 3) {
        // Default Hint to first entry.        
        mexPrintf("SI Starts OK.\n");
        *StartIndex = 1;
        mexPrintf("SI Ends OK.\n");
    } else if (nrhs == 4) {
        if (!mxIsUint32(prhs[3])) {
            mexErrMsgTxt("EndHint must be uint32.");
        } 
        StartIndex = mxGetData(prhs[3]);
    } else {
        mexErrMsgTxt("Must provide three or four input arguments.");
    }
    // Convert the hint to base-zero indexing.
    *StartIndex = *StartIndex - 1;

    // Check the inputs for the window range.
    if (!mxIsUint32(prhs[0])) {
        mexErrMsgTxt("DatesList must be uint32.");
    }
    if (!mxIsUint32(prhs[1])) {
        mexErrMsgTxt("WindowStartsDates must be uint32.");
    }
    if (!mxIsUint32(prhs[2])) {
        mexErrMsgTxt("WindowEndsDates must be uint32.");
    }

    if (mxGetM(prhs[1]) != mxGetM(prhs[2])) {
        mexErrMsgTxt("WindowStartDates must be the same length as WindowEndDates.");
    }    

    // Prepare the output array.
    ismember = mxCreateLogicalArray(mxGetNumberOfDimensions(prhs[1]), mxGetDimensions(prhs[1]));

    CalculationRoutine(mxGetLogicals(ismember),mxGetData(prhs[0]), 
        mxGetData(prhs[1]), mxGetData(prhs[2]), StartIndex, (int) mxGetM(prhs[1]));

    plhs[0] = ismember;
}

Я звоню с:

>>InWindow(uint32(5),uint32((1:6)'),uint32((3:8)'))

Код достигает линии между двумя вызовами mexPrintf до ошибки сегментации (т. Е. Печатается первый вызов, но не второй).

Я на Matlab 2007a (да, я знаю), Win7 64bit и VS 2008.

1 Ответ

4 голосов
/ 20 апреля 2011

Вам нужно инициализировать указатель StartIndex - вам повезло, что он работает с первого раза, так как он не указывает на определенную область памяти. Почему бы не сделать что-то более похожее на:

unsigned int StartIndex;
// and either:
StartIndex = 1;
// or:
StartIndex = * (static_cast< unsigned int * >( mxGetData(prhs[3]) ) );
...