git: bf17672f5058 - main - graphics/py-cairo: unbreak with python310
Jan Beich
jbeich at FreeBSD.org
Sun Jul 25 16:17:20 UTC 2021
The branch main has been updated by jbeich:
URL: https://cgit.FreeBSD.org/ports/commit/?id=bf17672f50580a768f536ca3c5b243324e2cd64a
commit bf17672f50580a768f536ca3c5b243324e2cd64a
Author: Jan Beich <jbeich at FreeBSD.org>
AuthorDate: 2021-07-25 15:17:51 +0000
Commit: Jan Beich <jbeich at FreeBSD.org>
CommitDate: 2021-07-25 16:17:05 +0000
graphics/py-cairo: unbreak with python310
>>> import cairo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/usr/local/lib/python3.10/site-packages/cairo/__init__.py", line 1, in <module>
from ._cairo import * # noqa: F401,F403
ImportError: /usr/local/lib/python3.10/site-packages/cairo/_cairo.cpython-310.so: Undefined symbol "PyObject_AsReadBuffer"
ImportError: /usr/local/lib/python3.10/site-packages/cairo/_cairo.cpython-310.so: Undefined symbol "PyObject_AsWriteBuffer"
---
graphics/py-cairo/Makefile | 2 +-
graphics/py-cairo/files/patch-cairo_surface.c | 215 ++++++++++++++++++++++++++
2 files changed, 216 insertions(+), 1 deletion(-)
diff --git a/graphics/py-cairo/Makefile b/graphics/py-cairo/Makefile
index 3afb3bd7cd16..3570aa42fc6d 100644
--- a/graphics/py-cairo/Makefile
+++ b/graphics/py-cairo/Makefile
@@ -2,7 +2,7 @@
PORTNAME= cairo
PORTVERSION= 1.18.1
-PORTREVISION= 1
+PORTREVISION= 2
PORTEPOCH= 1
CATEGORIES= graphics python
MASTER_SITES= https://github.com/pygobject/pycairo/releases/download/v${PORTVERSION}/
diff --git a/graphics/py-cairo/files/patch-cairo_surface.c b/graphics/py-cairo/files/patch-cairo_surface.c
new file mode 100644
index 000000000000..b4ab64c6fe21
--- /dev/null
+++ b/graphics/py-cairo/files/patch-cairo_surface.c
@@ -0,0 +1,215 @@
+https://github.com/pygobject/pycairo/commit/0f8cdc058239
+https://github.com/pygobject/pycairo/commit/4c5377787624
+https://github.com/pygobject/pycairo/commit/590bcd2ecc9c
+
+--- cairo/surface.c.orig 2018-11-03 09:30:34 UTC
++++ cairo/surface.c
+@@ -142,6 +142,7 @@ _write_func (void *closure, const unsigned char *data,
+
+ static const cairo_user_data_key_t surface_base_object_key;
+ static const cairo_user_data_key_t surface_is_mapped_image;
++static const cairo_user_data_key_t surface_buffer_view_key;
+
+ static void
+ surface_dealloc (PycairoSurface *o) {
+@@ -439,28 +440,30 @@ _destroy_mime_user_data_func (PyObject *user_data) {
+ static void
+ _destroy_mime_data_func (PyObject *user_data) {
+ cairo_surface_t *surface;
++ Py_buffer *view;
+ PyObject *mime_intern;
+
+ PyGILState_STATE gstate = PyGILState_Ensure();
+
+ /* Remove the user data holding the source object */
+ surface = PyCapsule_GetPointer(PyTuple_GET_ITEM(user_data, 0), NULL);
+- mime_intern = PyTuple_GET_ITEM(user_data, 2);
++ view = PyCapsule_GetPointer(PyTuple_GET_ITEM(user_data, 1), NULL);
++ mime_intern = PyTuple_GET_ITEM(user_data, 3);
+ cairo_surface_set_user_data(
+ surface, (cairo_user_data_key_t *)mime_intern, NULL, NULL);
+
+ /* Destroy the user data */
+- _destroy_mime_user_data_func(user_data);
++ PyBuffer_Release (view);
++ PyMem_Free (view);
++ Py_DECREF(user_data);
+
+ PyGILState_Release(gstate);
+ }
+
+ static PyObject *
+ surface_set_mime_data (PycairoSurface *o, PyObject *args) {
+- PyObject *obj, *user_data, *mime_intern, *capsule;
+- const unsigned char *buffer;
++ PyObject *obj, *user_data, *mime_intern, *surface_capsule, *view_capsule;
+ const char *mime_type;
+- Py_ssize_t buffer_len;
+ int res;
+ cairo_status_t status;
+
+@@ -475,38 +478,58 @@ surface_set_mime_data (PycairoSurface *o, PyObject *ar
+ Py_RETURN_NONE;
+ }
+
+-PYCAIRO_BEGIN_IGNORE_DEPRECATED
+- res = PyObject_AsReadBuffer (obj, (const void **)&buffer, &buffer_len);
+-PYCAIRO_END_IGNORE_DEPRECATED
+- if (res == -1)
++ Py_buffer *view = PyMem_Malloc (sizeof (Py_buffer));
++ if (view == NULL) {
++ PyErr_NoMemory ();
+ return NULL;
++ }
+
++ res = PyObject_GetBuffer (obj, view, PyBUF_READ);
++ if (res == -1) {
++ PyMem_Free (view);
++ return NULL;
++ }
++
+ /* We use the interned mime type string as user data key and store the
+ * passed in object with it. This allows us to return the same object in
+ * surface_get_mime_data().
+ */
+ mime_intern = PYCAIRO_PyUnicode_InternFromString(mime_type);
+- capsule = PyCapsule_New(o->surface, NULL, NULL);
+- user_data = Py_BuildValue("(NOO)", capsule, obj, mime_intern);
+- if (user_data == NULL)
++ surface_capsule = PyCapsule_New(o->surface, NULL, NULL);
++ view_capsule = PyCapsule_New(view, NULL, NULL);
++ user_data = Py_BuildValue("(NNOO)", surface_capsule, view_capsule, obj, mime_intern);
++ if (user_data == NULL) {
++ PyBuffer_Release (view);
++ PyMem_Free (view);
+ return NULL;
++ }
+
+ status = cairo_surface_set_user_data(
+ o->surface, (cairo_user_data_key_t *)mime_intern, user_data,
+ (cairo_destroy_func_t)_destroy_mime_user_data_func);
+- if (status != CAIRO_STATUS_SUCCESS)
++
++ if (status != CAIRO_STATUS_SUCCESS) {
++ PyBuffer_Release (view);
++ PyMem_Free (view);
+ Py_DECREF(user_data);
+- RETURN_NULL_IF_CAIRO_ERROR(status);
++ Pycairo_Check_Status (status);
++ return NULL;
++ }
+
++ Py_INCREF(user_data);
+ status = cairo_surface_set_mime_data (
+- o->surface, mime_type, buffer, (unsigned long)buffer_len,
++ o->surface, mime_type, view->buf, (unsigned long)view->len,
+ (cairo_destroy_func_t)_destroy_mime_data_func, user_data);
++
+ if (status != CAIRO_STATUS_SUCCESS) {
+ cairo_surface_set_user_data(
+ o->surface, (cairo_user_data_key_t *)mime_intern, NULL, NULL);
++ PyBuffer_Release (view);
++ PyMem_Free (view);
++ Py_DECREF(user_data);
++ Pycairo_Check_Status (status);
++ return NULL;
+ }
+- RETURN_NULL_IF_CAIRO_ERROR(status);
+- Py_INCREF(user_data);
+
+ Py_RETURN_NONE;
+ }
+@@ -534,7 +557,7 @@ surface_get_mime_data (PycairoSurface *o, PyObject *ar
+ /* In case the mime data wasn't set through the Python API just copy it */
+ return Py_BuildValue(PYCAIRO_DATA_FORMAT "#", buffer, buffer_len);
+ } else {
+- obj = PyTuple_GET_ITEM(user_data, 1);
++ obj = PyTuple_GET_ITEM(user_data, 2);
+ Py_INCREF(obj);
+ return obj;
+ }
+@@ -804,14 +827,22 @@ image_surface_new (PyTypeObject *type, PyObject *args,
+ NULL);
+ }
+
++static void
++_release_buffer_destroy_func (void *user_data) {
++ Py_buffer *view = (Py_buffer *)user_data;
++ PyGILState_STATE gstate = PyGILState_Ensure();
++ PyBuffer_Release (view);
++ PyMem_Free (view);
++ PyGILState_Release(gstate);
++}
++
+ /* METH_CLASS */
+ static PyObject *
+ image_surface_create_for_data (PyTypeObject *type, PyObject *args) {
+ cairo_surface_t *surface;
+ cairo_format_t format;
+- unsigned char *buffer;
++ cairo_status_t status;
+ int width, height, stride = -1, res, format_arg;
+- Py_ssize_t buffer_len;
+ PyObject *obj;
+
+ if (!PyArg_ParseTuple (args, "Oiii|i:ImageSurface.create_for_data",
+@@ -820,12 +851,6 @@ image_surface_create_for_data (PyTypeObject *type, PyO
+
+ format = (cairo_format_t)format_arg;
+
+-PYCAIRO_BEGIN_IGNORE_DEPRECATED
+- res = PyObject_AsWriteBuffer (obj, (void **)&buffer, &buffer_len);
+-PYCAIRO_END_IGNORE_DEPRECATED
+- if (res == -1)
+- return NULL;
+-
+ if (width <= 0) {
+ PyErr_SetString(PyExc_ValueError, "width must be positive");
+ return NULL;
+@@ -843,15 +868,42 @@ PYCAIRO_END_IGNORE_DEPRECATED
+ return NULL;
+ }
+ }
+- if (height * stride > buffer_len) {
++
++ Py_buffer *view = PyMem_Malloc (sizeof (Py_buffer));
++ if (view == NULL) {
++ PyErr_NoMemory ();
++ return NULL;
++ }
++
++ res = PyObject_GetBuffer (obj, view, PyBUF_WRITABLE);
++ if (res == -1) {
++ PyMem_Free (view);
++ return NULL;
++ }
++
++ if (height * stride > view->len) {
++ PyBuffer_Release (view);
++ PyMem_Free (view);
+ PyErr_SetString(PyExc_TypeError, "buffer is not long enough");
+ return NULL;
+ }
++
+ Py_BEGIN_ALLOW_THREADS;
+- surface = cairo_image_surface_create_for_data (buffer, format, width,
+- height, stride);
++ surface = cairo_image_surface_create_for_data (view->buf, format, width,
++ height, stride);
+ Py_END_ALLOW_THREADS;
+- return _surface_create_with_object(surface, obj);
++
++ status = cairo_surface_set_user_data(
++ surface, &surface_buffer_view_key, view,
++ (cairo_destroy_func_t)_release_buffer_destroy_func);
++ if (Pycairo_Check_Status (status)) {
++ cairo_surface_destroy (surface);
++ PyBuffer_Release (view);
++ PyMem_Free (view);
++ return NULL;
++ }
++
++ return PycairoSurface_FromSurface(surface, NULL);
+ }
+
+
More information about the dev-commits-ports-all
mailing list