OpenCV: утечка памяти с интерфейсом Python, но не в версии C - PullRequest
7 голосов
/ 14 ноября 2010

Я спрашиваю здесь, потому что я до сих пор не получил никакой помощи от разработчиков OpenCV. Я свел проблему к очень простому тестовому случаю, так что, возможно, любой, кто имеет некоторый опыт работы с CPython, может помочь здесь.

Этот код C не протекает:

int main() {
    while(true) {
        int hist_size[] = {40};
        float range[] = {0.0f,255.0f};
        float* ranges[] = {range};
        CvHistogram* hist = cvCreateHist(1, hist_size, CV_HIST_ARRAY, ranges, 1);
        cvReleaseHist(&hist);
    }
}

Этот код Python действительно пропускает:

while True: cv.CreateHist([40], cv.CV_HIST_ARRAY, [[0,255]], 1)

Я искал код CPython (из текущего кода магистрали SVN OpenCV) и нашел это:

struct cvhistogram_t {
  PyObject_HEAD
  CvHistogram h;
  PyObject *bins;
};

...

/* cvhistogram */

static void cvhistogram_dealloc(PyObject *self)
{
  cvhistogram_t *cvh = (cvhistogram_t*)self;
  Py_DECREF(cvh->bins);
  PyObject_Del(self);
}

static PyTypeObject cvhistogram_Type = {
  PyObject_HEAD_INIT(&PyType_Type)
  0,                                      /*size*/
  MODULESTR".cvhistogram",                /*name*/
  sizeof(cvhistogram_t),                  /*basicsize*/
};

static PyObject *cvhistogram_getbins(cvhistogram_t *cvh)
{
  Py_INCREF(cvh->bins);
  return cvh->bins;
}

static PyGetSetDef cvhistogram_getseters[] = {
  {(char*)"bins", (getter)cvhistogram_getbins, (setter)NULL, (char*)"bins", NULL},
  {NULL}  /* Sentinel */
};

static void cvhistogram_specials(void)
{
  cvhistogram_Type.tp_dealloc = cvhistogram_dealloc;
  cvhistogram_Type.tp_getset = cvhistogram_getseters;
}

...

static PyObject *pycvCreateHist(PyObject *self, PyObject *args, PyObject *kw)
{
  const char *keywords[] = { "dims", "type", "ranges", "uniform", NULL };
  PyObject *dims;
  int type;
  float **ranges = NULL;
  int uniform = 1;

  if (!PyArg_ParseTupleAndKeywords(args, kw, "Oi|O&i", (char**)keywords, &dims, &type, convert_to_floatPTRPTR, (void*)&ranges, &uniform)) {
    return NULL;
  }
  cvhistogram_t *h = PyObject_NEW(cvhistogram_t, &cvhistogram_Type);
  args = Py_BuildValue("Oi", dims, CV_32FC1);
  h->bins = pycvCreateMatND(self, args);
  Py_DECREF(args);
  if (h->bins == NULL) {
    return NULL;
  }
  h->h.type = CV_HIST_MAGIC_VAL;
  if (!convert_to_CvArr(h->bins, &(h->h.bins), "bins"))
    return NULL;

  ERRWRAP(cvSetHistBinRanges(&(h->h), ranges, uniform));

  return (PyObject*)h;
}

А из заголовков OpenCV C:

typedef struct CvHistogram
{
    int     type;
    CvArr*  bins;
    float   thresh[CV_MAX_DIM][2];  /* For uniform histograms.                      */
    float** thresh2;                /* For non-uniform histograms.                  */
    CvMatND mat;                    /* Embedded matrix header for array histograms. */
}
CvHistogram;

Я не совсем все понимаю, потому что раньше я никогда не работал с C-интерфейсом для Python. Но, вероятно, ошибка, которую я ищу, находится где-то в этом коде.

Я прав? Или где искать ошибку? Как бы это исправить?

(Примечание для людей, которые видели более раннюю версию этого вопроса: я посмотрел на неправильный код. Их интерфейс SWIG устарел и больше не используется (но код все еще был в SVN, поэтому я запутал его. Так что не смотрите на interfaces/swig, этот код старый и не используется. Текущий код живет в modules/python.)


Отчет об ошибке в восходящем потоке: memleak в OpenCV Python CreateHist

Ответы [ 2 ]

2 голосов
/ 09 февраля 2011

Исправлено.

Изменено 3 недели назад на jamesb

  • статус изменен с принятого на закрытый
  • фиксированное разрешение

Исправлено в r4526

Параметры диапазонов не были освобождены, а итератор для диапазонов не был DECREF. Регрессии теперь проходят, а оригинальный цикл не течет.

0 голосов
/ 22 января 2011

Я думаю, что у вас есть проблема со сборкой мусора, в которой вы никогда не выходите из цикла.

Работает ли это больше, чем ожидалось?

while True: 
    cv.CreateHist([40], cv.CV_HIST_ARRAY, [[0,255]], 1)
    cv = None
...