Je veux libérer le GIL afin de paralléliser la boucle dans Cython, où différentes tranches de visions de mémoire sont passées à une certaine fonction à l'intérieur de la boucle. Le code ressemble à ceci:cython memoryviews tranches sans GIL
cpdef void do_sth_in_parallel(bint[:,:] input, bint[:] output, int D):
for d in prange(D, schedule=dynamic, nogil=True):
ouput[d] = some_function_not_requiring_gil(x[d,:])
Ce n'est pas possible, car la sélection de la tranche x [d ,:], semble exiger GIL. En cours d'exécution cython -a, et en utilisant une boucle normale, je reçois le code ci-dessous. Comment cela peut-il être fait en C pur?
__pyx_t_5.data = __pyx_v_x.data;
__pyx_t_5.memview = __pyx_v_x.memview;
__PYX_INC_MEMVIEW(&__pyx_t_5, 0);
{
Py_ssize_t __pyx_tmp_idx = __pyx_v_d;
Py_ssize_t __pyx_tmp_shape = __pyx_v_x.shape[0];
Py_ssize_t __pyx_tmp_stride = __pyx_v_x.strides[0];
if (0 && (__pyx_tmp_idx < 0))
__pyx_tmp_idx += __pyx_tmp_shape;
if (0 && (__pyx_tmp_idx < 0 || __pyx_tmp_idx >= __pyx_tmp_shape)) {
PyErr_SetString(PyExc_IndexError, "Index out of bounds (axis 0)");
__PYX_ERR(0, 130, __pyx_L1_error)
}
__pyx_t_5.data += __pyx_tmp_idx * __pyx_tmp_stride;
}
__pyx_t_5.shape[0] = __pyx_v_x.shape[1];
__pyx_t_5.strides[0] = __pyx_v_x.strides[1];
__pyx_t_5.suboffsets[0] = -1;
__pyx_t_6.data = __pyx_v_u.data;
__pyx_t_6.memview = __pyx_v_u.memview;
__PYX_INC_MEMVIEW(&__pyx_t_6, 0);
__pyx_t_6.shape[0] = __pyx_v_u.shape[0];
__pyx_t_6.strides[0] = __pyx_v_u.strides[0];
__pyx_t_6.suboffsets[0] = -1;
Il existe des exemples de 'nogil' dans les codes' numpy' et 'scipy' compilés. Je suggère de regarder pour voir s'il en existe dans les fichiers '.pyx'. – hpaulj
numpy/numpy/random/mtrand/mtrand.pyx – hpaulj
Je n'arrive toujours pas à comprendre pourquoi l'opération d'appel d'une tranche à partir de la vue mémoire, c'est-à-dire x [d ,:] nécessite une interaction python? – tammo