Panda3D
py_compat.h
Go to the documentation of this file.
1 /**
2  * @file py_compat.h
3  * @author rdb
4  * @date 2017-12-02
5  */
6 
7 #ifndef PY_COMPAT_H
8 #define PY_COMPAT_H
9 
10 #include "dtoolbase.h"
11 
12 #ifdef HAVE_PYTHON
13 
14 // The contents of this file were originally part of py_panda.h. It
15 // specifically contains polyfills that are required to maintain compatibility
16 // with Python 2 and older versions of Python 3.
17 
18 // These compatibility hacks are sorted by Python version that removes the
19 // need for the respective hack.
20 
21 #ifdef _POSIX_C_SOURCE
22 # undef _POSIX_C_SOURCE
23 #endif
24 
25 #ifdef _XOPEN_SOURCE
26 # undef _XOPEN_SOURCE
27 #endif
28 
29 // See PEP 353
30 #define PY_SSIZE_T_CLEAN 1
31 
32 #include <Python.h>
33 
34 #ifndef LINK_ALL_STATIC
35 # define EXPCL_PYPANDA
36 #elif defined(__GNUC__)
37 # define EXPCL_PYPANDA __attribute__((weak))
38 #else
39 # define EXPCL_PYPANDA extern inline
40 #endif
41 
42 /* Python 2.4 */
43 
44 // 2.4 macros which aren't available in 2.3
45 #ifndef Py_RETURN_NONE
46 # define Py_RETURN_NONE return Py_INCREF(Py_None), Py_None
47 #endif
48 
49 #ifndef Py_RETURN_TRUE
50 # define Py_RETURN_TRUE return Py_INCREF(Py_True), Py_True
51 #endif
52 
53 #ifndef Py_RETURN_FALSE
54 # define Py_RETURN_FALSE return Py_INCREF(Py_False), Py_False
55 #endif
56 
57 /* Python 2.5 */
58 
59 // Prior to Python 2.5, we didn't have Py_ssize_t.
60 #if PY_VERSION_HEX < 0x02050000
61 typedef int Py_ssize_t;
62 # define PyInt_FromSsize_t PyInt_FromLong
63 # define PyInt_AsSsize_t PyInt_AsLong
64 #endif
65 
66 /* Python 2.6 */
67 
68 #ifndef Py_TYPE
69 # define Py_TYPE(ob) (((PyObject*)(ob))->ob_type)
70 #endif
71 
72 /* Python 2.7, 3.1 */
73 
74 #ifndef PyVarObject_HEAD_INIT
75  #define PyVarObject_HEAD_INIT(type, size) \
76  PyObject_HEAD_INIT(type) size,
77 #endif
78 
79 /* Python 2.7, 3.2 */
80 
81 #if PY_VERSION_HEX < 0x03020000
82 # define PyErr_NewExceptionWithDoc(name, doc, base, dict) \
83  PyErr_NewException(name, base, dict)
84 #endif
85 
86 /* Python 3.0 */
87 
88 // Always on in Python 3
89 #ifndef Py_TPFLAGS_CHECKTYPES
90 # define Py_TPFLAGS_CHECKTYPES 0
91 #endif
92 
93 // Macros for writing code that will compile in both versions.
94 #if PY_MAJOR_VERSION >= 3
95 # define nb_nonzero nb_bool
96 # define nb_divide nb_true_divide
97 # define nb_inplace_divide nb_inplace_true_divide
98 
99 # define PyLongOrInt_Check(x) PyLong_Check(x)
100 # define PyLongOrInt_AS_LONG PyLong_AS_LONG
101 # define PyInt_Check PyLong_Check
102 # define PyInt_AsLong PyLong_AsLong
103 # define PyInt_AS_LONG PyLong_AS_LONG
104 # define PyLongOrInt_AsSize_t PyLong_AsSize_t
105 #else
106 # define PyLongOrInt_Check(x) (PyInt_Check(x) || PyLong_Check(x))
107 // PyInt_FromSize_t automatically picks the right type.
108 # define PyLongOrInt_AS_LONG PyInt_AsLong
109 
110 EXPCL_PYPANDA size_t PyLongOrInt_AsSize_t(PyObject *);
111 #endif
112 
113 // Which character to use in PyArg_ParseTuple et al for a byte string.
114 #if PY_MAJOR_VERSION >= 3
115 # define FMTCHAR_BYTES "y"
116 #else
117 # define FMTCHAR_BYTES "s"
118 #endif
119 
120 /* Python 3.2 */
121 
122 #if PY_VERSION_HEX < 0x03020000
123 typedef long Py_hash_t;
124 #endif
125 
126 /* Python 3.3 */
127 
128 #if PY_MAJOR_VERSION >= 3
129 // Python 3 versions before 3.3.3 defined this incorrectly.
130 # undef _PyErr_OCCURRED
131 # define _PyErr_OCCURRED() (PyThreadState_GET()->curexc_type)
132 
133 // Python versions before 3.3 did not define this.
134 # if PY_VERSION_HEX < 0x03030000
135 # define PyUnicode_AsUTF8 _PyUnicode_AsString
136 # define PyUnicode_AsUTF8AndSize _PyUnicode_AsStringAndSize
137 # endif
138 #endif
139 
140 /* Python 3.4 */
141 #if PY_VERSION_HEX < 0x03040000
142 #define PyGILState_Check() (PyGILState_GetThisThreadState() == _PyThreadState_Current)
143 #endif
144 
145 /* Python 3.6 */
146 
147 #if PY_VERSION_HEX < 0x03080000 && !defined(_PyObject_CallNoArg)
148 INLINE PyObject *_PyObject_CallNoArg(PyObject *func) {
149  static PyObject *empty_tuple = PyTuple_New(0);
150  return PyObject_Call(func, empty_tuple, nullptr);
151 }
152 # define _PyObject_CallNoArg _PyObject_CallNoArg
153 #endif
154 
155 #if PY_VERSION_HEX < 0x03080000 && !defined(_PyObject_FastCall)
156 INLINE PyObject *_PyObject_FastCall(PyObject *func, PyObject **args, Py_ssize_t nargs) {
157  PyObject *tuple = PyTuple_New(nargs);
158  for (Py_ssize_t i = 0; i < nargs; ++i) {
159  PyTuple_SET_ITEM(tuple, i, args[i]);
160  Py_INCREF(args[i]);
161  }
162  PyObject *result = PyObject_Call(func, tuple, nullptr);
163  Py_DECREF(tuple);
164  return result;
165 }
166 # define _PyObject_FastCall _PyObject_FastCall
167 #endif
168 
169 // Python versions before 3.6 didn't require longlong support to be enabled.
170 #ifndef HAVE_LONG_LONG
171 # define PyLong_FromLongLong(x) PyLong_FromLong((long) (x))
172 # define PyLong_FromUnsignedLongLong(x) PyLong_FromUnsignedLong((unsigned long) (x))
173 # define PyLong_AsLongLong(x) PyLong_AsLong(x)
174 # define PyLong_AsUnsignedLongLong(x) PyLong_AsUnsignedLong(x)
175 # define PyLong_AsUnsignedLongLongMask(x) PyLong_AsUnsignedLongMask(x)
176 # define PyLong_AsLongLongAndOverflow(x) PyLong_AsLongAndOverflow(x)
177 #endif
178 
179 /* Python 3.7 */
180 
181 #ifndef PyDict_GET_SIZE
182 # define PyDict_GET_SIZE(mp) (((PyDictObject *)mp)->ma_used)
183 #endif
184 
185 #ifndef Py_RETURN_RICHCOMPARE
186 # define Py_RETURN_RICHCOMPARE(val1, val2, op) \
187  do { \
188  switch (op) { \
189  NODEFAULT \
190  case Py_EQ: if ((val1) == (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
191  case Py_NE: if ((val1) != (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
192  case Py_LT: if ((val1) < (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
193  case Py_GT: if ((val1) > (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
194  case Py_LE: if ((val1) <= (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
195  case Py_GE: if ((val1) >= (val2)) Py_RETURN_TRUE; Py_RETURN_FALSE; \
196  } \
197  } while (0)
198 #endif
199 
200 /* Python 3.8 */
201 #if PY_VERSION_HEX < 0x03080000
202 INLINE PyObject *_PyLong_Rshift(PyObject *a, size_t shiftby) {
203  PyObject *b = PyLong_FromLong(shiftby);
204  PyObject *result = PyNumber_Rshift(a, b);
205  Py_DECREF(b);
206  return result;
207 }
208 INLINE PyObject *_PyLong_Lshift(PyObject *a, size_t shiftby) {
209  PyObject *b = PyLong_FromLong(shiftby);
210  PyObject *result = PyNumber_Lshift(a, b);
211  Py_DECREF(b);
212  return result;
213 }
214 #endif
215 
216 #if PY_VERSION_HEX < 0x03090000
217 INLINE EXPCL_PYPANDA PyObject *PyObject_CallNoArgs(PyObject *func) {
218  return _PyObject_CallNoArg(func);
219 }
220 
221 INLINE PyObject *PyObject_CallOneArg(PyObject *callable, PyObject *arg) {
222 #if PY_VERSION_HEX >= 0x03060000
223  return _PyObject_FastCall(callable, &arg, 1);
224 #else
225  return PyObject_CallFunctionObjArgs(callable, arg, nullptr);
226 #endif
227 }
228 
229 INLINE PyObject *PyObject_CallMethodNoArgs(PyObject *obj, PyObject *name) {
230  return PyObject_CallMethodObjArgs(obj, name, nullptr);
231 }
232 
233 INLINE PyObject *PyObject_CallMethodOneArg(PyObject *obj, PyObject *name, PyObject *arg) {
234  return PyObject_CallMethodObjArgs(obj, name, arg, nullptr);
235 }
236 #endif
237 
238 /* Other Python implementations */
239 
240 // _PyErr_OCCURRED is an undocumented macro version of PyErr_Occurred.
241 // Some implementations of the CPython API (e.g. PyPy's cpyext) do not define
242 // it, so in these cases we just silently fall back to PyErr_Occurred.
243 #ifndef _PyErr_OCCURRED
244 # define _PyErr_OCCURRED() PyErr_Occurred()
245 #endif
246 
247 #endif // HAVE_PYTHON
248 
249 #endif // PY_COMPAT_H
PANDA 3D SOFTWARE Copyright (c) Carnegie Mellon University.