| #define FORTRANOBJECT_C
|
| #include "fortranobject.h"
|
|
|
| #ifdef __cplusplus
|
| extern "C" {
|
| #endif
|
|
|
| #include <stdarg.h>
|
| #include <stdlib.h>
|
| #include <string.h>
|
|
|
| |
| |
| |
| |
| |
| |
|
|
|
|
| int
|
| F2PyDict_SetItemString(PyObject *dict, char *name, PyObject *obj)
|
| {
|
| if (obj == NULL) {
|
| fprintf(stderr, "Error loading %s\n", name);
|
| if (PyErr_Occurred()) {
|
| PyErr_Print();
|
| PyErr_Clear();
|
| }
|
| return -1;
|
| }
|
| return PyDict_SetItemString(dict, name, obj);
|
| }
|
|
|
| |
| |
|
|
| void *
|
| F2PySwapThreadLocalCallbackPtr(char *key, void *ptr)
|
| {
|
| PyObject *local_dict, *value;
|
| void *prev;
|
|
|
| local_dict = PyThreadState_GetDict();
|
| if (local_dict == NULL) {
|
| Py_FatalError(
|
| "F2PySwapThreadLocalCallbackPtr: PyThreadState_GetDict "
|
| "failed");
|
| }
|
|
|
| value = PyDict_GetItemString(local_dict, key);
|
| if (value != NULL) {
|
| prev = PyLong_AsVoidPtr(value);
|
| if (PyErr_Occurred()) {
|
| Py_FatalError(
|
| "F2PySwapThreadLocalCallbackPtr: PyLong_AsVoidPtr failed");
|
| }
|
| }
|
| else {
|
| prev = NULL;
|
| }
|
|
|
| value = PyLong_FromVoidPtr((void *)ptr);
|
| if (value == NULL) {
|
| Py_FatalError(
|
| "F2PySwapThreadLocalCallbackPtr: PyLong_FromVoidPtr failed");
|
| }
|
|
|
| if (PyDict_SetItemString(local_dict, key, value) != 0) {
|
| Py_FatalError(
|
| "F2PySwapThreadLocalCallbackPtr: PyDict_SetItemString failed");
|
| }
|
|
|
| Py_DECREF(value);
|
|
|
| return prev;
|
| }
|
|
|
| void *
|
| F2PyGetThreadLocalCallbackPtr(char *key)
|
| {
|
| PyObject *local_dict, *value;
|
| void *prev;
|
|
|
| local_dict = PyThreadState_GetDict();
|
| if (local_dict == NULL) {
|
| Py_FatalError(
|
| "F2PyGetThreadLocalCallbackPtr: PyThreadState_GetDict failed");
|
| }
|
|
|
| value = PyDict_GetItemString(local_dict, key);
|
| if (value != NULL) {
|
| prev = PyLong_AsVoidPtr(value);
|
| if (PyErr_Occurred()) {
|
| Py_FatalError(
|
| "F2PyGetThreadLocalCallbackPtr: PyLong_AsVoidPtr failed");
|
| }
|
| }
|
| else {
|
| prev = NULL;
|
| }
|
|
|
| return prev;
|
| }
|
|
|
| static PyArray_Descr *
|
| get_descr_from_type_and_elsize(const int type_num, const int elsize) {
|
| PyArray_Descr * descr = PyArray_DescrFromType(type_num);
|
| if (type_num == NPY_STRING) {
|
|
|
| PyArray_DESCR_REPLACE(descr);
|
| if (descr == NULL) {
|
| return NULL;
|
| }
|
| PyDataType_SET_ELSIZE(descr, elsize);
|
| }
|
| return descr;
|
| }
|
|
|
|
|
|
|
| typedef PyObject *(*fortranfunc)(PyObject *, PyObject *, PyObject *, void *);
|
|
|
| PyObject *
|
| PyFortranObject_New(FortranDataDef *defs, f2py_void_func init)
|
| {
|
| int i;
|
| PyFortranObject *fp = NULL;
|
| PyObject *v = NULL;
|
| if (init != NULL) {
|
| (*(init))();
|
| }
|
| fp = PyObject_New(PyFortranObject, &PyFortran_Type);
|
| if (fp == NULL) {
|
| return NULL;
|
| }
|
| if ((fp->dict = PyDict_New()) == NULL) {
|
| Py_DECREF(fp);
|
| return NULL;
|
| }
|
| fp->len = 0;
|
| while (defs[fp->len].name != NULL) {
|
| fp->len++;
|
| }
|
| if (fp->len == 0) {
|
| goto fail;
|
| }
|
| fp->defs = defs;
|
| for (i = 0; i < fp->len; i++) {
|
| if (fp->defs[i].rank == -1) {
|
| v = PyFortranObject_NewAsAttr(&(fp->defs[i]));
|
| if (v == NULL) {
|
| goto fail;
|
| }
|
| PyDict_SetItemString(fp->dict, fp->defs[i].name, v);
|
| Py_XDECREF(v);
|
| }
|
| else if ((fp->defs[i].data) !=
|
| NULL) {
|
| PyArray_Descr *
|
| descr = get_descr_from_type_and_elsize(fp->defs[i].type,
|
| fp->defs[i].elsize);
|
| if (descr == NULL) {
|
| goto fail;
|
| }
|
| v = PyArray_NewFromDescr(&PyArray_Type, descr, fp->defs[i].rank,
|
| fp->defs[i].dims.d, NULL, fp->defs[i].data,
|
| NPY_ARRAY_FARRAY, NULL);
|
| if (v == NULL) {
|
| Py_DECREF(descr);
|
| goto fail;
|
| }
|
| PyDict_SetItemString(fp->dict, fp->defs[i].name, v);
|
| Py_XDECREF(v);
|
| }
|
| }
|
| return (PyObject *)fp;
|
| fail:
|
| Py_XDECREF(fp);
|
| return NULL;
|
| }
|
|
|
| PyObject *
|
| PyFortranObject_NewAsAttr(FortranDataDef *defs)
|
| {
|
| PyFortranObject *fp = NULL;
|
| fp = PyObject_New(PyFortranObject, &PyFortran_Type);
|
| if (fp == NULL)
|
| return NULL;
|
| if ((fp->dict = PyDict_New()) == NULL) {
|
| PyObject_Del(fp);
|
| return NULL;
|
| }
|
| fp->len = 1;
|
| fp->defs = defs;
|
| if (defs->rank == -1) {
|
| PyDict_SetItemString(fp->dict, "__name__", PyUnicode_FromFormat("function %s", defs->name));
|
| } else if (defs->rank == 0) {
|
| PyDict_SetItemString(fp->dict, "__name__", PyUnicode_FromFormat("scalar %s", defs->name));
|
| } else {
|
| PyDict_SetItemString(fp->dict, "__name__", PyUnicode_FromFormat("array %s", defs->name));
|
| }
|
| return (PyObject *)fp;
|
| }
|
|
|
|
|
|
|
| static void
|
| fortran_dealloc(PyFortranObject *fp)
|
| {
|
| Py_XDECREF(fp->dict);
|
| PyObject_Del(fp);
|
| }
|
|
|
|
|
| static Py_ssize_t
|
| format_def(char *buf, Py_ssize_t size, FortranDataDef def)
|
| {
|
| char *p = buf;
|
| int i;
|
| npy_intp n;
|
|
|
| n = PyOS_snprintf(p, size, "array(%" NPY_INTP_FMT, def.dims.d[0]);
|
| if (n < 0 || n >= size) {
|
| return -1;
|
| }
|
| p += n;
|
| size -= n;
|
|
|
| for (i = 1; i < def.rank; i++) {
|
| n = PyOS_snprintf(p, size, ",%" NPY_INTP_FMT, def.dims.d[i]);
|
| if (n < 0 || n >= size) {
|
| return -1;
|
| }
|
| p += n;
|
| size -= n;
|
| }
|
|
|
| if (size <= 0) {
|
| return -1;
|
| }
|
|
|
| *p++ = ')';
|
| size--;
|
|
|
| if (def.data == NULL) {
|
| static const char notalloc[] = ", not allocated";
|
| if ((size_t)size < sizeof(notalloc)) {
|
| return -1;
|
| }
|
| memcpy(p, notalloc, sizeof(notalloc));
|
| p += sizeof(notalloc);
|
| size -= sizeof(notalloc);
|
| }
|
|
|
| return p - buf;
|
| }
|
|
|
| static PyObject *
|
| fortran_doc(FortranDataDef def)
|
| {
|
| char *buf, *p;
|
| PyObject *s = NULL;
|
| Py_ssize_t n, origsize, size = 100;
|
|
|
| if (def.doc != NULL) {
|
| size += strlen(def.doc);
|
| }
|
| origsize = size;
|
| buf = p = (char *)PyMem_Malloc(size);
|
| if (buf == NULL) {
|
| return PyErr_NoMemory();
|
| }
|
|
|
| if (def.rank == -1) {
|
| if (def.doc) {
|
| n = strlen(def.doc);
|
| if (n > size) {
|
| goto fail;
|
| }
|
| memcpy(p, def.doc, n);
|
| p += n;
|
| size -= n;
|
| }
|
| else {
|
| n = PyOS_snprintf(p, size, "%s - no docs available", def.name);
|
| if (n < 0 || n >= size) {
|
| goto fail;
|
| }
|
| p += n;
|
| size -= n;
|
| }
|
| }
|
| else {
|
| PyArray_Descr *d = PyArray_DescrFromType(def.type);
|
| n = PyOS_snprintf(p, size, "%s : '%c'-", def.name, d->type);
|
| Py_DECREF(d);
|
| if (n < 0 || n >= size) {
|
| goto fail;
|
| }
|
| p += n;
|
| size -= n;
|
|
|
| if (def.data == NULL) {
|
| n = format_def(p, size, def);
|
| if (n < 0) {
|
| goto fail;
|
| }
|
| p += n;
|
| size -= n;
|
| }
|
| else if (def.rank > 0) {
|
| n = format_def(p, size, def);
|
| if (n < 0) {
|
| goto fail;
|
| }
|
| p += n;
|
| size -= n;
|
| }
|
| else {
|
| n = strlen("scalar");
|
| if (size < n) {
|
| goto fail;
|
| }
|
| memcpy(p, "scalar", n);
|
| p += n;
|
| size -= n;
|
| }
|
| }
|
| if (size <= 1) {
|
| goto fail;
|
| }
|
| *p++ = '\n';
|
| size--;
|
|
|
|
|
| s = PyUnicode_FromStringAndSize(buf, p - buf);
|
|
|
| PyMem_Free(buf);
|
| return s;
|
|
|
| fail:
|
| fprintf(stderr,
|
| "fortranobject.c: fortran_doc: len(p)=%zd>%zd=size:"
|
| " too long docstring required, increase size\n",
|
| p - buf, origsize);
|
| PyMem_Free(buf);
|
| return NULL;
|
| }
|
|
|
| static FortranDataDef *save_def;
|
| static void
|
| set_data(char *d, npy_intp *f)
|
| {
|
| if (*f)
|
| save_def->data = d;
|
| else
|
| save_def->data = NULL;
|
|
|
| }
|
|
|
| static PyObject *
|
| fortran_getattr(PyFortranObject *fp, char *name)
|
| {
|
| int i, j, k, flag;
|
| if (fp->dict != NULL) {
|
| PyObject *v = _PyDict_GetItemStringWithError(fp->dict, name);
|
| if (v == NULL && PyErr_Occurred()) {
|
| return NULL;
|
| }
|
| else if (v != NULL) {
|
| Py_INCREF(v);
|
| return v;
|
| }
|
| }
|
| for (i = 0, j = 1; i < fp->len && (j = strcmp(name, fp->defs[i].name));
|
| i++)
|
| ;
|
| if (j == 0)
|
| if (fp->defs[i].rank != -1) {
|
| if (fp->defs[i].func == NULL)
|
| return NULL;
|
| for (k = 0; k < fp->defs[i].rank; ++k) fp->defs[i].dims.d[k] = -1;
|
| save_def = &fp->defs[i];
|
| (*(fp->defs[i].func))(&fp->defs[i].rank, fp->defs[i].dims.d,
|
| set_data, &flag);
|
| if (flag == 2)
|
| k = fp->defs[i].rank + 1;
|
| else
|
| k = fp->defs[i].rank;
|
| if (fp->defs[i].data != NULL) {
|
| PyObject *v = PyArray_New(
|
| &PyArray_Type, k, fp->defs[i].dims.d, fp->defs[i].type,
|
| NULL, fp->defs[i].data, 0, NPY_ARRAY_FARRAY, NULL);
|
| if (v == NULL)
|
| return NULL;
|
|
|
| return v;
|
| }
|
| else {
|
| Py_RETURN_NONE;
|
| }
|
| }
|
| if (strcmp(name, "__dict__") == 0) {
|
| Py_INCREF(fp->dict);
|
| return fp->dict;
|
| }
|
| if (strcmp(name, "__doc__") == 0) {
|
| PyObject *s = PyUnicode_FromString(""), *s2, *s3;
|
| for (i = 0; i < fp->len; i++) {
|
| s2 = fortran_doc(fp->defs[i]);
|
| s3 = PyUnicode_Concat(s, s2);
|
| Py_DECREF(s2);
|
| Py_DECREF(s);
|
| s = s3;
|
| }
|
| if (PyDict_SetItemString(fp->dict, name, s))
|
| return NULL;
|
| return s;
|
| }
|
| if ((strcmp(name, "_cpointer") == 0) && (fp->len == 1)) {
|
| PyObject *cobj =
|
| F2PyCapsule_FromVoidPtr((void *)(fp->defs[0].data), NULL);
|
| if (PyDict_SetItemString(fp->dict, name, cobj))
|
| return NULL;
|
| return cobj;
|
| }
|
| PyObject *str, *ret;
|
| str = PyUnicode_FromString(name);
|
| ret = PyObject_GenericGetAttr((PyObject *)fp, str);
|
| Py_DECREF(str);
|
| return ret;
|
| }
|
|
|
| static int
|
| fortran_setattr(PyFortranObject *fp, char *name, PyObject *v)
|
| {
|
| int i, j, flag;
|
| PyArrayObject *arr = NULL;
|
| for (i = 0, j = 1; i < fp->len && (j = strcmp(name, fp->defs[i].name));
|
| i++)
|
| ;
|
| if (j == 0) {
|
| if (fp->defs[i].rank == -1) {
|
| PyErr_SetString(PyExc_AttributeError,
|
| "over-writing fortran routine");
|
| return -1;
|
| }
|
| if (fp->defs[i].func != NULL) {
|
| npy_intp dims[F2PY_MAX_DIMS];
|
| int k;
|
| save_def = &fp->defs[i];
|
| if (v != Py_None) { |
| |
|
|
| for (k = 0; k < fp->defs[i].rank; k++) dims[k] = -1;
|
| if ((arr = array_from_pyobj(fp->defs[i].type, dims,
|
| fp->defs[i].rank, F2PY_INTENT_IN,
|
| v)) == NULL)
|
| return -1;
|
| (*(fp->defs[i].func))(&fp->defs[i].rank, PyArray_DIMS(arr),
|
| set_data, &flag);
|
| }
|
| else {
|
| for (k = 0; k < fp->defs[i].rank; k++) dims[k] = 0;
|
| (*(fp->defs[i].func))(&fp->defs[i].rank, dims, set_data,
|
| &flag);
|
| for (k = 0; k < fp->defs[i].rank; k++) dims[k] = -1;
|
| }
|
| memcpy(fp->defs[i].dims.d, dims,
|
| fp->defs[i].rank * sizeof(npy_intp));
|
| }
|
| else {
|
| if ((arr = array_from_pyobj(fp->defs[i].type, fp->defs[i].dims.d,
|
| fp->defs[i].rank, F2PY_INTENT_IN,
|
| v)) == NULL)
|
| return -1;
|
| }
|
| if (fp->defs[i].data !=
|
| NULL) {
|
| npy_intp s = PyArray_MultiplyList(fp->defs[i].dims.d,
|
| PyArray_NDIM(arr));
|
| if (s == -1)
|
| s = PyArray_MultiplyList(PyArray_DIMS(arr), PyArray_NDIM(arr));
|
| if (s < 0 || (memcpy(fp->defs[i].data, PyArray_DATA(arr),
|
| s * PyArray_ITEMSIZE(arr))) == NULL) {
|
| if ((PyObject *)arr != v) {
|
| Py_DECREF(arr);
|
| }
|
| return -1;
|
| }
|
| if ((PyObject *)arr != v) {
|
| Py_DECREF(arr);
|
| }
|
| }
|
| else
|
| return (fp->defs[i].func == NULL ? -1 : 0);
|
| return 0;
|
| }
|
| if (fp->dict == NULL) {
|
| fp->dict = PyDict_New();
|
| if (fp->dict == NULL)
|
| return -1;
|
| }
|
| if (v == NULL) {
|
| int rv = PyDict_DelItemString(fp->dict, name);
|
| if (rv < 0)
|
| PyErr_SetString(PyExc_AttributeError,
|
| "delete non-existing fortran attribute");
|
| return rv;
|
| }
|
| else
|
| return PyDict_SetItemString(fp->dict, name, v);
|
| }
|
|
|
| static PyObject *
|
| fortran_call(PyFortranObject *fp, PyObject *arg, PyObject *kw)
|
| {
|
| int i = 0;
|
| |
| |
|
|
| if (fp->defs[i].rank == -1) {
|
| if (fp->defs[i].func == NULL) {
|
| PyErr_Format(PyExc_RuntimeError, "no function to call");
|
| return NULL;
|
| }
|
| else if (fp->defs[i].data == NULL)
|
|
|
| return (*((fortranfunc)(fp->defs[i].func)))((PyObject *)fp, arg,
|
| kw, NULL);
|
| else
|
| return (*((fortranfunc)(fp->defs[i].func)))(
|
| (PyObject *)fp, arg, kw, (void *)fp->defs[i].data);
|
| }
|
| PyErr_Format(PyExc_TypeError, "this fortran object is not callable");
|
| return NULL;
|
| }
|
|
|
| static PyObject *
|
| fortran_repr(PyFortranObject *fp)
|
| {
|
| PyObject *name = NULL, *repr = NULL;
|
| name = PyObject_GetAttrString((PyObject *)fp, "__name__");
|
| PyErr_Clear();
|
| if (name != NULL && PyUnicode_Check(name)) {
|
| repr = PyUnicode_FromFormat("<fortran %U>", name);
|
| }
|
| else {
|
| repr = PyUnicode_FromString("<fortran object>");
|
| }
|
| Py_XDECREF(name);
|
| return repr;
|
| }
|
|
|
| PyTypeObject PyFortran_Type = {
|
| PyVarObject_HEAD_INIT(NULL, 0).tp_name = "fortran",
|
| .tp_basicsize = sizeof(PyFortranObject),
|
| .tp_dealloc = (destructor)fortran_dealloc,
|
| .tp_getattr = (getattrfunc)fortran_getattr,
|
| .tp_setattr = (setattrfunc)fortran_setattr,
|
| .tp_repr = (reprfunc)fortran_repr,
|
| .tp_call = (ternaryfunc)fortran_call,
|
| };
|
|
|
|
|
|
|
| #ifdef F2PY_REPORT_ATEXIT
|
| static int passed_time = 0;
|
| static int passed_counter = 0;
|
| static int passed_call_time = 0;
|
| static struct timeb start_time;
|
| static struct timeb stop_time;
|
| static struct timeb start_call_time;
|
| static struct timeb stop_call_time;
|
| static int cb_passed_time = 0;
|
| static int cb_passed_counter = 0;
|
| static int cb_passed_call_time = 0;
|
| static struct timeb cb_start_time;
|
| static struct timeb cb_stop_time;
|
| static struct timeb cb_start_call_time;
|
| static struct timeb cb_stop_call_time;
|
|
|
| extern void
|
| f2py_start_clock(void)
|
| {
|
| ftime(&start_time);
|
| }
|
| extern void
|
| f2py_start_call_clock(void)
|
| {
|
| f2py_stop_clock();
|
| ftime(&start_call_time);
|
| }
|
| extern void
|
| f2py_stop_clock(void)
|
| {
|
| ftime(&stop_time);
|
| passed_time += 1000 * (stop_time.time - start_time.time);
|
| passed_time += stop_time.millitm - start_time.millitm;
|
| }
|
| extern void
|
| f2py_stop_call_clock(void)
|
| {
|
| ftime(&stop_call_time);
|
| passed_call_time += 1000 * (stop_call_time.time - start_call_time.time);
|
| passed_call_time += stop_call_time.millitm - start_call_time.millitm;
|
| passed_counter += 1;
|
| f2py_start_clock();
|
| }
|
|
|
| extern void
|
| f2py_cb_start_clock(void)
|
| {
|
| ftime(&cb_start_time);
|
| }
|
| extern void
|
| f2py_cb_start_call_clock(void)
|
| {
|
| f2py_cb_stop_clock();
|
| ftime(&cb_start_call_time);
|
| }
|
| extern void
|
| f2py_cb_stop_clock(void)
|
| {
|
| ftime(&cb_stop_time);
|
| cb_passed_time += 1000 * (cb_stop_time.time - cb_start_time.time);
|
| cb_passed_time += cb_stop_time.millitm - cb_start_time.millitm;
|
| }
|
| extern void
|
| f2py_cb_stop_call_clock(void)
|
| {
|
| ftime(&cb_stop_call_time);
|
| cb_passed_call_time +=
|
| 1000 * (cb_stop_call_time.time - cb_start_call_time.time);
|
| cb_passed_call_time +=
|
| cb_stop_call_time.millitm - cb_start_call_time.millitm;
|
| cb_passed_counter += 1;
|
| f2py_cb_start_clock();
|
| }
|
|
|
| static int f2py_report_on_exit_been_here = 0;
|
| extern void
|
| f2py_report_on_exit(int exit_flag, void *name)
|
| {
|
| if (f2py_report_on_exit_been_here) {
|
| fprintf(stderr, " %s\n", (char *)name);
|
| return;
|
| }
|
| f2py_report_on_exit_been_here = 1;
|
| fprintf(stderr, " /-----------------------\\\n");
|
| fprintf(stderr, " < F2PY performance report >\n");
|
| fprintf(stderr, " \\-----------------------/\n");
|
| fprintf(stderr, "Overall time spent in ...\n");
|
| fprintf(stderr, "(a) wrapped (Fortran/C) functions : %8d msec\n",
|
| passed_call_time);
|
| fprintf(stderr, "(b) f2py interface, %6d calls : %8d msec\n",
|
| passed_counter, passed_time);
|
| fprintf(stderr, "(c) call-back (Python) functions : %8d msec\n",
|
| cb_passed_call_time);
|
| fprintf(stderr, "(d) f2py call-back interface, %6d calls : %8d msec\n",
|
| cb_passed_counter, cb_passed_time);
|
|
|
| fprintf(stderr,
|
| "(e) wrapped (Fortran/C) functions (actual) : %8d msec\n\n",
|
| passed_call_time - cb_passed_call_time - cb_passed_time);
|
| fprintf(stderr,
|
| "Use -DF2PY_REPORT_ATEXIT_DISABLE to disable this message.\n");
|
| fprintf(stderr, "Exit status: %d\n", exit_flag);
|
| fprintf(stderr, "Modules : %s\n", (char *)name);
|
| }
|
| #endif
|
|
|
|
|
|
|
| #ifdef F2PY_REPORT_ON_ARRAY_COPY
|
| static void
|
| f2py_report_on_array_copy(PyArrayObject *arr)
|
| {
|
| const npy_intp arr_size = PyArray_Size((PyObject *)arr);
|
| if (arr_size > F2PY_REPORT_ON_ARRAY_COPY) {
|
| fprintf(stderr,
|
| "copied an array: size=%ld, elsize=%" NPY_INTP_FMT "\n",
|
| arr_size, (npy_intp)PyArray_ITEMSIZE(arr));
|
| }
|
| }
|
| static void
|
| f2py_report_on_array_copy_fromany(void)
|
| {
|
| fprintf(stderr, "created an array from object\n");
|
| }
|
|
|
| #define F2PY_REPORT_ON_ARRAY_COPY_FROMARR \
|
| f2py_report_on_array_copy((PyArrayObject *)arr)
|
| #define F2PY_REPORT_ON_ARRAY_COPY_FROMANY f2py_report_on_array_copy_fromany()
|
| #else
|
| #define F2PY_REPORT_ON_ARRAY_COPY_FROMARR
|
| #define F2PY_REPORT_ON_ARRAY_COPY_FROMANY
|
| #endif
|
|
|
|
|
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
|
|
| static int check_and_fix_dimensions(const PyArrayObject* arr,
|
| const int rank,
|
| npy_intp *dims,
|
| const char *errmess);
|
|
|
| static int
|
| find_first_negative_dimension(const int rank, const npy_intp *dims)
|
| {
|
| int i;
|
| for (i = 0; i < rank; ++i) {
|
| if (dims[i] < 0) {
|
| return i;
|
| }
|
| }
|
| return -1;
|
| }
|
|
|
| #ifdef DEBUG_COPY_ND_ARRAY
|
| void
|
| dump_dims(int rank, npy_intp const *dims)
|
| {
|
| int i;
|
| printf("[");
|
| for (i = 0; i < rank; ++i) {
|
| printf("%3" NPY_INTP_FMT, dims[i]);
|
| }
|
| printf("]\n");
|
| }
|
| void
|
| dump_attrs(const PyArrayObject *obj)
|
| {
|
| const PyArrayObject_fields *arr = (const PyArrayObject_fields *)obj;
|
| int rank = PyArray_NDIM(arr);
|
| npy_intp size = PyArray_Size((PyObject *)arr);
|
| printf("\trank = %d, flags = %d, size = %" NPY_INTP_FMT "\n", rank,
|
| arr->flags, size);
|
| printf("\tstrides = ");
|
| dump_dims(rank, arr->strides);
|
| printf("\tdimensions = ");
|
| dump_dims(rank, arr->dimensions);
|
| }
|
| #endif
|
|
|
| #define SWAPTYPE(a, b, t) \
|
| { \
|
| t c; \
|
| c = (a); \
|
| (a) = (b); \
|
| (b) = c; \
|
| }
|
|
|
| static int
|
| swap_arrays(PyArrayObject *obj1, PyArrayObject *obj2)
|
| {
|
| PyArrayObject_fields *arr1 = (PyArrayObject_fields *)obj1,
|
| *arr2 = (PyArrayObject_fields *)obj2;
|
| SWAPTYPE(arr1->data, arr2->data, char *);
|
| SWAPTYPE(arr1->nd, arr2->nd, int);
|
| SWAPTYPE(arr1->dimensions, arr2->dimensions, npy_intp *);
|
| SWAPTYPE(arr1->strides, arr2->strides, npy_intp *);
|
| SWAPTYPE(arr1->base, arr2->base, PyObject *);
|
| SWAPTYPE(arr1->descr, arr2->descr, PyArray_Descr *);
|
| SWAPTYPE(arr1->flags, arr2->flags, int);
|
|
|
| return 0;
|
| }
|
|
|
| #define ARRAY_ISCOMPATIBLE(arr,type_num) \
|
| ((PyArray_ISINTEGER(arr) && PyTypeNum_ISINTEGER(type_num)) || \
|
| (PyArray_ISFLOAT(arr) && PyTypeNum_ISFLOAT(type_num)) || \
|
| (PyArray_ISCOMPLEX(arr) && PyTypeNum_ISCOMPLEX(type_num)) || \
|
| (PyArray_ISBOOL(arr) && PyTypeNum_ISBOOL(type_num)) || \
|
| (PyArray_ISSTRING(arr) && PyTypeNum_ISSTRING(type_num)))
|
|
|
| static int
|
| get_elsize(PyObject *obj) {
|
| |
| |
| |
| |
| |
| |
|
|
|
|
| if (PyArray_Check(obj)) {
|
| return PyArray_ITEMSIZE((PyArrayObject *)obj);
|
| } else if (PyBytes_Check(obj)) {
|
| return PyBytes_GET_SIZE(obj);
|
| } else if (PyUnicode_Check(obj)) {
|
| return PyUnicode_GET_LENGTH(obj);
|
| } else if (PySequence_Check(obj)) {
|
| PyObject* fast = PySequence_Fast(obj, "f2py:fortranobject.c:get_elsize");
|
| if (fast != NULL) {
|
| Py_ssize_t i, n = PySequence_Fast_GET_SIZE(fast);
|
| int sz, elsize = 0;
|
| for (i=0; i<n; i++) {
|
| sz = get_elsize(PySequence_Fast_GET_ITEM(fast, i) );
|
| if (sz > elsize) {
|
| elsize = sz;
|
| }
|
| }
|
| Py_DECREF(fast);
|
| return elsize;
|
| }
|
| }
|
| return -1;
|
| }
|
|
|
| extern PyArrayObject *
|
| ndarray_from_pyobj(const int type_num,
|
| const int elsize_,
|
| npy_intp *dims,
|
| const int rank,
|
| const int intent,
|
| PyObject *obj,
|
| const char *errmess) {
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| char mess[F2PY_MESSAGE_BUFFER_SIZE];
|
| PyArrayObject *arr = NULL;
|
| int elsize = (elsize_ < 0 ? get_elsize(obj) : elsize_);
|
| if (elsize < 0) {
|
| if (errmess != NULL) {
|
| strcpy(mess, errmess);
|
| }
|
| sprintf(mess + strlen(mess),
|
| " -- failed to determine element size from %s",
|
| Py_TYPE(obj)->tp_name);
|
| PyErr_SetString(PyExc_SystemError, mess);
|
| return NULL;
|
| }
|
| PyArray_Descr * descr = get_descr_from_type_and_elsize(type_num, elsize);
|
| if (descr == NULL) {
|
| return NULL;
|
| }
|
| elsize = PyDataType_ELSIZE(descr);
|
| if ((intent & F2PY_INTENT_HIDE)
|
| || ((intent & F2PY_INTENT_CACHE) && (obj == Py_None))
|
| || ((intent & F2PY_OPTIONAL) && (obj == Py_None))
|
| ) {
|
|
|
| int ineg = find_first_negative_dimension(rank, dims);
|
| if (ineg >= 0) {
|
| int i;
|
| strcpy(mess, "failed to create intent(cache|hide)|optional array"
|
| "-- must have defined dimensions but got (");
|
| for(i = 0; i < rank; ++i)
|
| sprintf(mess + strlen(mess), "%" NPY_INTP_FMT ",", dims[i]);
|
| strcat(mess, ")");
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| arr = (PyArrayObject *) \
|
| PyArray_NewFromDescr(&PyArray_Type, descr, rank, dims,
|
| NULL, NULL, !(intent & F2PY_INTENT_C), NULL);
|
| if (arr == NULL) {
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| if (PyArray_ITEMSIZE(arr) != elsize) {
|
| strcpy(mess, "failed to create intent(cache|hide)|optional array");
|
| sprintf(mess+strlen(mess)," -- expected elsize=%d got %" NPY_INTP_FMT, elsize, (npy_intp)PyArray_ITEMSIZE(arr));
|
| PyErr_SetString(PyExc_ValueError,mess);
|
| Py_DECREF(arr);
|
| return NULL;
|
| }
|
| if (!(intent & F2PY_INTENT_CACHE)) {
|
| PyArray_FILLWBYTE(arr, 0);
|
| }
|
| return arr;
|
| }
|
|
|
| if (PyArray_Check(obj)) {
|
| arr = (PyArrayObject *)obj;
|
| if (intent & F2PY_INTENT_CACHE) {
|
|
|
| if (PyArray_ISONESEGMENT(arr)
|
| && PyArray_ITEMSIZE(arr) >= elsize) {
|
| if (check_and_fix_dimensions(arr, rank, dims, errmess)) {
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| if (intent & F2PY_INTENT_OUT)
|
| Py_INCREF(arr);
|
| Py_DECREF(descr);
|
| return arr;
|
| }
|
| strcpy(mess, "failed to initialize intent(cache) array");
|
| if (!PyArray_ISONESEGMENT(arr))
|
| strcat(mess, " -- input must be in one segment");
|
| if (PyArray_ITEMSIZE(arr) < elsize)
|
| sprintf(mess + strlen(mess),
|
| " -- expected at least elsize=%d but got "
|
| "%" NPY_INTP_FMT,
|
| elsize, (npy_intp)PyArray_ITEMSIZE(arr));
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
|
|
| |
|
|
|
|
| if (check_and_fix_dimensions(arr, rank, dims, errmess)) {
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| |
| |
| |
| |
| |
| |
|
|
| if ((! (intent & F2PY_INTENT_COPY)) &&
|
| PyArray_ITEMSIZE(arr) == elsize &&
|
| ARRAY_ISCOMPATIBLE(arr,type_num) &&
|
| F2PY_CHECK_ALIGNMENT(arr, intent)) {
|
| if ((intent & F2PY_INTENT_INOUT || intent & F2PY_INTENT_INPLACE)
|
| ? ((intent & F2PY_INTENT_C) ? PyArray_ISCARRAY(arr) : PyArray_ISFARRAY(arr))
|
| : ((intent & F2PY_INTENT_C) ? PyArray_ISCARRAY_RO(arr) : PyArray_ISFARRAY_RO(arr))) {
|
| if ((intent & F2PY_INTENT_OUT)) {
|
| Py_INCREF(arr);
|
| }
|
|
|
| Py_DECREF(descr);
|
| return arr;
|
| }
|
| }
|
| if (intent & F2PY_INTENT_INOUT) {
|
| strcpy(mess, "failed to initialize intent(inout) array");
|
| |
|
|
| if ((intent & F2PY_INTENT_C) && !PyArray_ISCARRAY(arr))
|
| strcat(mess, " -- input not contiguous");
|
| if (!(intent & F2PY_INTENT_C) && !PyArray_ISFARRAY(arr))
|
| strcat(mess, " -- input not fortran contiguous");
|
| if (PyArray_ITEMSIZE(arr) != elsize)
|
| sprintf(mess + strlen(mess),
|
| " -- expected elsize=%d but got %" NPY_INTP_FMT,
|
| elsize,
|
| (npy_intp)PyArray_ITEMSIZE(arr)
|
| );
|
| if (!(ARRAY_ISCOMPATIBLE(arr, type_num))) {
|
| sprintf(mess + strlen(mess),
|
| " -- input '%c' not compatible to '%c'",
|
| PyArray_DESCR(arr)->type, descr->type);
|
| }
|
| if (!(F2PY_CHECK_ALIGNMENT(arr, intent)))
|
| sprintf(mess + strlen(mess), " -- input not %d-aligned",
|
| F2PY_GET_ALIGNMENT(intent));
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
|
|
|
|
|
|
| {
|
| PyArrayObject * retarr = (PyArrayObject *) \
|
| PyArray_NewFromDescr(&PyArray_Type, descr, PyArray_NDIM(arr), PyArray_DIMS(arr),
|
| NULL, NULL, !(intent & F2PY_INTENT_C), NULL);
|
| if (retarr==NULL) {
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| F2PY_REPORT_ON_ARRAY_COPY_FROMARR;
|
| if (PyArray_CopyInto(retarr, arr)) {
|
| Py_DECREF(retarr);
|
| return NULL;
|
| }
|
| if (intent & F2PY_INTENT_INPLACE) {
|
| if (swap_arrays(arr,retarr)) {
|
| Py_DECREF(retarr);
|
| return NULL;
|
| }
|
| Py_XDECREF(retarr);
|
| if (intent & F2PY_INTENT_OUT)
|
| Py_INCREF(arr);
|
| } else {
|
| arr = retarr;
|
| }
|
| }
|
| return arr;
|
| }
|
|
|
| if ((intent & F2PY_INTENT_INOUT) || (intent & F2PY_INTENT_INPLACE) ||
|
| (intent & F2PY_INTENT_CACHE)) {
|
| PyErr_Format(PyExc_TypeError,
|
| "failed to initialize intent(inout|inplace|cache) "
|
| "array, input '%s' object is not an array",
|
| Py_TYPE(obj)->tp_name);
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
|
|
| {
|
| F2PY_REPORT_ON_ARRAY_COPY_FROMANY;
|
| arr = (PyArrayObject *)PyArray_FromAny(
|
| obj, descr, 0, 0,
|
| ((intent & F2PY_INTENT_C) ? NPY_ARRAY_CARRAY
|
| : NPY_ARRAY_FARRAY) |
|
| NPY_ARRAY_FORCECAST,
|
| NULL);
|
|
|
|
|
| if (arr == NULL) {
|
| Py_DECREF(descr);
|
| return NULL;
|
| }
|
| if (type_num != NPY_STRING && PyArray_ITEMSIZE(arr) != elsize) {
|
|
|
|
|
| strcpy(mess, "failed to initialize intent(in) array");
|
| sprintf(mess + strlen(mess),
|
| " -- expected elsize=%d got %" NPY_INTP_FMT, elsize,
|
| (npy_intp)PyArray_ITEMSIZE(arr));
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| Py_DECREF(arr);
|
| return NULL;
|
| }
|
| if (check_and_fix_dimensions(arr, rank, dims, errmess)) {
|
| Py_DECREF(arr);
|
| return NULL;
|
| }
|
| return arr;
|
| }
|
| }
|
|
|
| extern PyArrayObject *
|
| array_from_pyobj(const int type_num,
|
| npy_intp *dims,
|
| const int rank,
|
| const int intent,
|
| PyObject *obj) {
|
| |
| |
| |
|
|
| PyArray_Descr* descr = PyArray_DescrFromType(type_num);
|
| int elsize = PyDataType_ELSIZE(descr);
|
| Py_DECREF(descr);
|
| return ndarray_from_pyobj(type_num, elsize, dims, rank, intent, obj, NULL);
|
| }
|
|
|
|
|
|
|
|
|
|
|
| static int
|
| check_and_fix_dimensions(const PyArrayObject* arr, const int rank,
|
| npy_intp *dims, const char *errmess)
|
| {
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| char mess[F2PY_MESSAGE_BUFFER_SIZE];
|
| const npy_intp arr_size =
|
| (PyArray_NDIM(arr)) ? PyArray_Size((PyObject *)arr) : 1;
|
| #ifdef DEBUG_COPY_ND_ARRAY
|
| dump_attrs(arr);
|
| printf("check_and_fix_dimensions:init: dims=");
|
| dump_dims(rank, dims);
|
| #endif
|
| if (rank > PyArray_NDIM(arr)) {
|
| npy_intp new_size = 1;
|
| int free_axe = -1;
|
| int i;
|
| npy_intp d;
|
|
|
| for (i = 0; i < PyArray_NDIM(arr); ++i) {
|
| d = PyArray_DIM(arr, i);
|
| if (dims[i] >= 0) {
|
| if (d > 1 && dims[i] != d) {
|
| PyErr_Format(
|
| PyExc_ValueError,
|
| "%d-th dimension must be fixed to %" NPY_INTP_FMT
|
| " but got %" NPY_INTP_FMT "\n",
|
| i, dims[i], d);
|
| return 1;
|
| }
|
| if (!dims[i])
|
| dims[i] = 1;
|
| }
|
| else {
|
| dims[i] = d ? d : 1;
|
| }
|
| new_size *= dims[i];
|
| }
|
| for (i = PyArray_NDIM(arr); i < rank; ++i)
|
| if (dims[i] > 1) {
|
| PyErr_Format(PyExc_ValueError,
|
| "%d-th dimension must be %" NPY_INTP_FMT
|
| " but got 0 (not defined).\n",
|
| i, dims[i]);
|
| return 1;
|
| }
|
| else if (free_axe < 0)
|
| free_axe = i;
|
| else
|
| dims[i] = 1;
|
| if (free_axe >= 0) {
|
| dims[free_axe] = arr_size / new_size;
|
| new_size *= dims[free_axe];
|
| }
|
| if (new_size != arr_size) {
|
| PyErr_Format(PyExc_ValueError,
|
| "unexpected array size: new_size=%" NPY_INTP_FMT
|
| ", got array with arr_size=%" NPY_INTP_FMT
|
| " (maybe too many free indices)\n",
|
| new_size, arr_size);
|
| return 1;
|
| }
|
| }
|
| else if (rank == PyArray_NDIM(arr)) {
|
| npy_intp new_size = 1;
|
| int i;
|
| npy_intp d;
|
| for (i = 0; i < rank; ++i) {
|
| d = PyArray_DIM(arr, i);
|
| if (dims[i] >= 0) {
|
| if (d > 1 && d != dims[i]) {
|
| if (errmess != NULL) {
|
| strcpy(mess, errmess);
|
| }
|
| sprintf(mess + strlen(mess),
|
| " -- %d-th dimension must be fixed to %"
|
| NPY_INTP_FMT " but got %" NPY_INTP_FMT,
|
| i, dims[i], d);
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| return 1;
|
| }
|
| if (!dims[i])
|
| dims[i] = 1;
|
| }
|
| else
|
| dims[i] = d;
|
| new_size *= dims[i];
|
| }
|
| if (new_size != arr_size) {
|
| PyErr_Format(PyExc_ValueError,
|
| "unexpected array size: new_size=%" NPY_INTP_FMT
|
| ", got array with arr_size=%" NPY_INTP_FMT "\n",
|
| new_size, arr_size);
|
| return 1;
|
| }
|
| }
|
| else {
|
| int i, j;
|
| npy_intp d;
|
| int effrank;
|
| npy_intp size;
|
| for (i = 0, effrank = 0; i < PyArray_NDIM(arr); ++i)
|
| if (PyArray_DIM(arr, i) > 1)
|
| ++effrank;
|
| if (dims[rank - 1] >= 0)
|
| if (effrank > rank) {
|
| PyErr_Format(PyExc_ValueError,
|
| "too many axes: %d (effrank=%d), "
|
| "expected rank=%d\n",
|
| PyArray_NDIM(arr), effrank, rank);
|
| return 1;
|
| }
|
|
|
| for (i = 0, j = 0; i < rank; ++i) {
|
| while (j < PyArray_NDIM(arr) && PyArray_DIM(arr, j) < 2) ++j;
|
| if (j >= PyArray_NDIM(arr))
|
| d = 1;
|
| else
|
| d = PyArray_DIM(arr, j++);
|
| if (dims[i] >= 0) {
|
| if (d > 1 && d != dims[i]) {
|
| if (errmess != NULL) {
|
| strcpy(mess, errmess);
|
| }
|
| sprintf(mess + strlen(mess),
|
| " -- %d-th dimension must be fixed to %"
|
| NPY_INTP_FMT " but got %" NPY_INTP_FMT
|
| " (real index=%d)\n",
|
| i, dims[i], d, j-1);
|
| PyErr_SetString(PyExc_ValueError, mess);
|
| return 1;
|
| }
|
| if (!dims[i])
|
| dims[i] = 1;
|
| }
|
| else
|
| dims[i] = d;
|
| }
|
|
|
| for (i = rank; i < PyArray_NDIM(arr);
|
| ++i) {
|
| while (j < PyArray_NDIM(arr) && PyArray_DIM(arr, j) < 2) ++j;
|
| if (j >= PyArray_NDIM(arr))
|
| d = 1;
|
| else
|
| d = PyArray_DIM(arr, j++);
|
| dims[rank - 1] *= d;
|
| }
|
| for (i = 0, size = 1; i < rank; ++i) size *= dims[i];
|
| if (size != arr_size) {
|
| char msg[200];
|
| int len;
|
| snprintf(msg, sizeof(msg),
|
| "unexpected array size: size=%" NPY_INTP_FMT
|
| ", arr_size=%" NPY_INTP_FMT
|
| ", rank=%d, effrank=%d, arr.nd=%d, dims=[",
|
| size, arr_size, rank, effrank, PyArray_NDIM(arr));
|
| for (i = 0; i < rank; ++i) {
|
| len = strlen(msg);
|
| snprintf(msg + len, sizeof(msg) - len, " %" NPY_INTP_FMT,
|
| dims[i]);
|
| }
|
| len = strlen(msg);
|
| snprintf(msg + len, sizeof(msg) - len, " ], arr.dims=[");
|
| for (i = 0; i < PyArray_NDIM(arr); ++i) {
|
| len = strlen(msg);
|
| snprintf(msg + len, sizeof(msg) - len, " %" NPY_INTP_FMT,
|
| PyArray_DIM(arr, i));
|
| }
|
| len = strlen(msg);
|
| snprintf(msg + len, sizeof(msg) - len, " ]\n");
|
| PyErr_SetString(PyExc_ValueError, msg);
|
| return 1;
|
| }
|
| }
|
| #ifdef DEBUG_COPY_ND_ARRAY
|
| printf("check_and_fix_dimensions:end: dims=");
|
| dump_dims(rank, dims);
|
| #endif
|
| return 0;
|
| }
|
|
|
|
|
|
|
|
|
|
|
| extern int
|
| copy_ND_array(const PyArrayObject *arr, PyArrayObject *out)
|
| {
|
| F2PY_REPORT_ON_ARRAY_COPY_FROMARR;
|
| return PyArray_CopyInto(out, (PyArrayObject *)arr);
|
| }
|
|
|
|
|
|
|
| extern int
|
| f2py_describe(PyObject *obj, char *buf) {
|
| |
| |
| |
| |
| |
|
|
| char localbuf[F2PY_MESSAGE_BUFFER_SIZE];
|
| if (PyBytes_Check(obj)) {
|
| sprintf(localbuf, "%d-%s", (npy_int)PyBytes_GET_SIZE(obj), Py_TYPE(obj)->tp_name);
|
| } else if (PyUnicode_Check(obj)) {
|
| sprintf(localbuf, "%d-%s", (npy_int)PyUnicode_GET_LENGTH(obj), Py_TYPE(obj)->tp_name);
|
| } else if (PyArray_CheckScalar(obj)) {
|
| PyArrayObject* arr = (PyArrayObject*)obj;
|
| sprintf(localbuf, "%c%" NPY_INTP_FMT "-%s-scalar", PyArray_DESCR(arr)->kind, PyArray_ITEMSIZE(arr), Py_TYPE(obj)->tp_name);
|
| } else if (PyArray_Check(obj)) {
|
| int i;
|
| PyArrayObject* arr = (PyArrayObject*)obj;
|
| strcpy(localbuf, "(");
|
| for (i=0; i<PyArray_NDIM(arr); i++) {
|
| if (i) {
|
| strcat(localbuf, " ");
|
| }
|
| sprintf(localbuf + strlen(localbuf), "%" NPY_INTP_FMT ",", PyArray_DIM(arr, i));
|
| }
|
| sprintf(localbuf + strlen(localbuf), ")-%c%" NPY_INTP_FMT "-%s", PyArray_DESCR(arr)->kind, PyArray_ITEMSIZE(arr), Py_TYPE(obj)->tp_name);
|
| } else if (PySequence_Check(obj)) {
|
| sprintf(localbuf, "%d-%s", (npy_int)PySequence_Length(obj), Py_TYPE(obj)->tp_name);
|
| } else {
|
| sprintf(localbuf, "%s instance", Py_TYPE(obj)->tp_name);
|
| }
|
|
|
| strcpy(buf, localbuf);
|
| return 1;
|
| }
|
|
|
| extern npy_intp
|
| f2py_size_impl(PyArrayObject* var, ...)
|
| {
|
| npy_intp sz = 0;
|
| npy_intp dim;
|
| npy_intp rank;
|
| va_list argp;
|
| va_start(argp, var);
|
| dim = va_arg(argp, npy_int);
|
| if (dim==-1)
|
| {
|
| sz = PyArray_SIZE(var);
|
| }
|
| else
|
| {
|
| rank = PyArray_NDIM(var);
|
| if (dim>=1 && dim<=rank)
|
| sz = PyArray_DIM(var, dim-1);
|
| else
|
| fprintf(stderr, "f2py_size: 2nd argument value=%" NPY_INTP_FMT
|
| " fails to satisfy 1<=value<=%" NPY_INTP_FMT
|
| ". Result will be 0.\n", dim, rank);
|
| }
|
| va_end(argp);
|
| return sz;
|
| }
|
|
|
|
|
|
|
|
|
|
|
| PyObject *
|
| F2PyCapsule_FromVoidPtr(void *ptr, void (*dtor)(PyObject *))
|
| {
|
| PyObject *ret = PyCapsule_New(ptr, NULL, dtor);
|
| if (ret == NULL) {
|
| PyErr_Clear();
|
| }
|
| return ret;
|
| }
|
|
|
| void *
|
| F2PyCapsule_AsVoidPtr(PyObject *obj)
|
| {
|
| void *ret = PyCapsule_GetPointer(obj, NULL);
|
| if (ret == NULL) {
|
| PyErr_Clear();
|
| }
|
| return ret;
|
| }
|
|
|
| int
|
| F2PyCapsule_Check(PyObject *ptr)
|
| {
|
| return PyCapsule_CheckExact(ptr);
|
| }
|
|
|
| #ifdef __cplusplus
|
| }
|
| #endif
|
|
|
|
|