Logo ROOT  
Reference Guide
Pythonize.cxx
Go to the documentation of this file.
1// Bindings
2#include "CPyCppyy.h"
3#include "Pythonize.h"
4#include "Converters.h"
5#include "CPPInstance.h"
6#include "CPPOverload.h"
7#include "CustomPyTypes.h"
8#include "LowLevelViews.h"
9#include "ProxyWrappers.h"
10#include "PyCallable.h"
11#include "PyStrings.h"
12#include "TypeManip.h"
13#include "Utility.h"
14
15// Standard
16#include <algorithm>
17#include <complex>
18#include <set>
19#include <stdexcept>
20#include <sstream>
21#include <string>
22#include <utility>
23
24
25//- data and local helpers ---------------------------------------------------
26namespace CPyCppyy {
27 extern PyObject* gThisModule;
28 extern std::map<std::string, std::vector<PyObject*>> gPythonizations;
29}
30
31namespace {
32
33// for convenience
34using namespace CPyCppyy;
35
36//-----------------------------------------------------------------------------
37bool HasAttrDirect(PyObject* pyclass, PyObject* pyname, bool mustBeCPyCppyy = false) {
38// prevents calls to Py_TYPE(pyclass)->tp_getattr, which is unnecessary for our
39// purposes here and could tickle problems w/ spurious lookups into ROOT meta
40 PyObject* dct = PyObject_GetAttr(pyclass, PyStrings::gDict);
41 if (dct) {
42 PyObject* attr = PyObject_GetItem(dct, pyname);
43 Py_DECREF(dct);
44 if (attr) {
45 bool ret = !mustBeCPyCppyy || CPPOverload_Check(attr);
46 Py_DECREF(attr);
47 return ret;
48 }
49 }
50 PyErr_Clear();
51 return false;
52}
53
54//-----------------------------------------------------------------------------
55inline bool IsTemplatedSTLClass(const std::string& name, const std::string& klass) {
56// Scan the name of the class and determine whether it is a template instantiation.
57 auto pos = name.find(klass);
58 return (pos == 0 || pos == 5) && name.find("::", name.rfind(">")) == std::string::npos;
59}
60
61// to prevent compiler warnings about const char* -> char*
62inline PyObject* CallPyObjMethod(PyObject* obj, const char* meth)
63{
64// Helper; call method with signature: obj->meth().
65 Py_INCREF(obj);
66 PyObject* result = PyObject_CallMethod(obj, const_cast<char*>(meth), const_cast<char*>(""));
67 Py_DECREF(obj);
68 return result;
69}
70
71//-----------------------------------------------------------------------------
72inline PyObject* CallPyObjMethod(PyObject* obj, const char* meth, PyObject* arg1)
73{
74// Helper; call method with signature: obj->meth(arg1).
75 Py_INCREF(obj);
76 PyObject* result = PyObject_CallMethod(
77 obj, const_cast<char*>(meth), const_cast<char*>("O"), arg1);
78 Py_DECREF(obj);
79 return result;
80}
81
82//-----------------------------------------------------------------------------
84{
85// Helper; converts python index into straight C index.
86 Py_ssize_t idx = PyInt_AsSsize_t(index);
87 if (idx == (Py_ssize_t)-1 && PyErr_Occurred())
88 return nullptr;
89
90 Py_ssize_t size = PySequence_Size(self);
91 if (idx >= size || (idx < 0 && idx < -size)) {
92 PyErr_SetString(PyExc_IndexError, "index out of range");
93 return nullptr;
94 }
95
96 PyObject* pyindex = nullptr;
97 if (idx >= 0) {
98 Py_INCREF(index);
99 pyindex = index;
100 } else
101 pyindex = PyLong_FromSsize_t(size+idx);
102
103 return pyindex;
104}
105
106//-----------------------------------------------------------------------------
107inline bool AdjustSlice(const Py_ssize_t nlen, Py_ssize_t& start, Py_ssize_t& stop, Py_ssize_t& step)
108{
109// Helper; modify slice range to match the container.
110 if ((step > 0 && stop <= start) || (step < 0 && start <= stop))
111 return false;
112
113 if (start < 0) start = 0;
114 if (start >= nlen) start = nlen-1;
115 if (step >= nlen) step = nlen;
116
117 stop = step > 0 ? std::min(nlen, stop) : (stop >= 0 ? stop : -1);
118 return true;
119}
120
121//-----------------------------------------------------------------------------
122inline PyObject* CallSelfIndex(CPPInstance* self, PyObject* idx, PyObject* pymeth)
123{
124// Helper; call method with signature: meth(pyindex).
125 Py_INCREF((PyObject*)self);
126 PyObject* pyindex = PyStyleIndex((PyObject*)self, idx);
127 if (!pyindex) {
128 Py_DECREF((PyObject*)self);
129 return nullptr;
130 }
131
132 PyObject* result = PyObject_CallMethodObjArgs((PyObject*)self, pymeth, pyindex, nullptr);
133 Py_DECREF(pyindex);
134 Py_DECREF((PyObject*)self);
135 return result;
136}
137
138//- "smart pointer" behavior ---------------------------------------------------
139PyObject* DeRefGetAttr(PyObject* self, PyObject* name)
140{
141// Follow operator*() if present (available in python as __deref__), so that
142// smart pointers behave as expected.
144 // TODO: these calls come from TemplateProxy and are unlikely to be needed in practice,
145 // whereas as-is, they can accidentally dereference the result of end() on some STL
146 // containers. Obviously, this is a dumb hack that should be resolved more fundamentally.
147 PyErr_SetString(PyExc_AttributeError, CPyCppyy_PyText_AsString(name));
148 return nullptr;
149 }
150
152 PyErr_SetString(PyExc_TypeError, "getattr(): attribute name must be string");
153
154 PyObject* pyptr = PyObject_CallMethodObjArgs(self, PyStrings::gDeref, nullptr);
155 if (!pyptr)
156 return nullptr;
157
158// prevent a potential infinite loop
159 if (Py_TYPE(pyptr) == Py_TYPE(self)) {
160 PyObject* val1 = PyObject_Str(self);
161 PyObject* val2 = PyObject_Str(name);
162 PyErr_Format(PyExc_AttributeError, "%s has no attribute \'%s\'",
164 Py_DECREF(val2);
165 Py_DECREF(val1);
166
167 Py_DECREF(pyptr);
168 return nullptr;
169 }
170
171 PyObject* result = PyObject_GetAttr(pyptr, name);
172 Py_DECREF(pyptr);
173 return result;
174}
175
176//-----------------------------------------------------------------------------
177PyObject* FollowGetAttr(PyObject* self, PyObject* name)
178{
179// Follow operator->() if present (available in python as __follow__), so that
180// smart pointers behave as expected.
182 PyErr_SetString(PyExc_TypeError, "getattr(): attribute name must be string");
183
184 PyObject* pyptr = PyObject_CallMethodObjArgs(self, PyStrings::gFollow, nullptr);
185 if (!pyptr)
186 return nullptr;
187
188 PyObject* result = PyObject_GetAttr(pyptr, name);
189 Py_DECREF(pyptr);
190 return result;
191}
192
193
194//- vector behavior as primitives ----------------------------------------------
195#if PY_VERSION_HEX < 0x03040000
196#define PyObject_LengthHint _PyObject_LengthHint
197#endif
198
199// TODO: can probably use the below getters in the InitializerListConverter
200struct ItemGetter {
201 ItemGetter(PyObject* pyobj) : fPyObject(pyobj) { Py_INCREF(fPyObject); }
202 virtual ~ItemGetter() { Py_DECREF(fPyObject); }
203 virtual Py_ssize_t size() = 0;
204 virtual PyObject* get() = 0;
205 PyObject* fPyObject;
206};
207
208struct CountedItemGetter : public ItemGetter {
209 CountedItemGetter(PyObject* pyobj) : ItemGetter(pyobj), fCur(0) {}
210 Py_ssize_t fCur;
211};
212
213struct TupleItemGetter : public CountedItemGetter {
214 using CountedItemGetter::CountedItemGetter;
215 virtual Py_ssize_t size() { return PyTuple_GET_SIZE(fPyObject); }
216 virtual PyObject* get() {
217 if (fCur < PyTuple_GET_SIZE(fPyObject)) {
218 PyObject* item = PyTuple_GET_ITEM(fPyObject, fCur++);
219 Py_INCREF(item);
220 return item;
221 }
222 PyErr_SetString(PyExc_StopIteration, "end of tuple");
223 return nullptr;
224 }
225};
226
227struct ListItemGetter : public CountedItemGetter {
228 using CountedItemGetter::CountedItemGetter;
229 virtual Py_ssize_t size() { return PyList_GET_SIZE(fPyObject); }
230 virtual PyObject* get() {
231 if (fCur < PyList_GET_SIZE(fPyObject)) {
232 PyObject* item = PyList_GET_ITEM(fPyObject, fCur++);
233 Py_INCREF(item);
234 return item;
235 }
236 PyErr_SetString(PyExc_StopIteration, "end of list");
237 return nullptr;
238 }
239};
240
241struct SequenceItemGetter : public CountedItemGetter {
242 using CountedItemGetter::CountedItemGetter;
243 virtual Py_ssize_t size() {
244 Py_ssize_t sz = PySequence_Size(fPyObject);
245 if (sz < 0) {
246 PyErr_Clear();
247 return PyObject_LengthHint(fPyObject, 8);
248 }
249 return sz;
250 }
251 virtual PyObject* get() { return PySequence_GetItem(fPyObject, fCur++); }
252};
253
254struct IterItemGetter : public ItemGetter {
255 using ItemGetter::ItemGetter;
256 virtual Py_ssize_t size() { return PyObject_LengthHint(fPyObject, 8); }
257 virtual PyObject* get() { return (*(Py_TYPE(fPyObject)->tp_iternext))(fPyObject); }
258};
259
260PyObject* VectorInit(PyObject* self, PyObject* args, PyObject* /* kwds */)
261{
262// Specialized vector constructor to allow construction from containers; allowing
263// such construction from initializer_list instead would possible, but can be
264// error-prone. This use case is common enough for std::vector to implement it
265// directly, except for arrays (which can be passed wholesale) and strings (which
266// won't convert properly as they'll be seen as buffers)
267
268 ItemGetter* getter = nullptr;
269 if (PyTuple_GET_SIZE(args) == 1) {
270 PyObject* fi = PyTuple_GET_ITEM(args, 0);
271 if (CPyCppyy_PyText_Check(fi) || PyBytes_Check(fi)) {
272 PyErr_SetString(PyExc_TypeError, "can not convert string to vector");
273 return nullptr;
274 }
275 // TODO: this only tests for new-style buffers, which is too strict, but a
276 // generic check for Py_TYPE(fi)->tp_as_buffer is too loose (note that the
277 // main use case is numpy, which offers the new interface)
278 if (!PyObject_CheckBuffer(fi)) {
279 if (PyTuple_CheckExact(fi))
280 getter = new TupleItemGetter(fi);
281 else if (PyList_CheckExact(fi))
282 getter = new ListItemGetter(fi);
283 else if (PySequence_Check(fi))
284 getter = new SequenceItemGetter(fi);
285 else {
286 PyObject* iter = PyObject_GetIter(fi);
287 if (iter) {
288 getter = new IterItemGetter{iter};
289 Py_DECREF(iter);
290 }
291 else PyErr_Clear();
292 }
293 }
294 }
295
296 if (getter) {
297 // construct an empty vector, then back-fill it
298 PyObject* mname = CPyCppyy_PyText_FromString("__real_init");
299 PyObject* result = PyObject_CallMethodObjArgs(self, mname, nullptr);
300 Py_DECREF(mname);
301 if (!result) {
302 delete getter;
303 return result;
304 }
305
306 Py_ssize_t sz = getter->size();
307 if (sz < 0) {
308 delete getter;
309 return nullptr;
310 }
311
312 // reserve memory as appliable
313 if (0 < sz) {
314 PyObject* res = PyObject_CallMethod(self, (char*)"reserve", (char*)"n", sz);
315 Py_DECREF(res);
316 } else { // empty container
317 delete getter;
318 return result;
319 }
320
321 bool fill_ok = true;
322
323 // two main options: a list of lists (or tuples), or a list of objects; the former
324 // are emplace_back'ed, the latter push_back'ed
325 PyObject* fi = PySequence_GetItem(PyTuple_GET_ITEM(args, 0), 0);
326 if (!fi) PyErr_Clear();
327 if (fi && (PyTuple_CheckExact(fi) || PyList_CheckExact(fi))) {
328 // use emplace_back to construct the vector entries one by one
329 PyObject* eb_call = PyObject_GetAttrString(self, (char*)"emplace_back");
330 PyObject* vtype = PyObject_GetAttrString((PyObject*)Py_TYPE(self), "value_type");
331 bool value_is_vector = false;
332 if (vtype && CPyCppyy_PyText_Check(vtype)) {
333 // if the value_type is a vector, then allow for initialization from sequences
334 if (std::string(CPyCppyy_PyText_AsString(vtype)).rfind("std::vector", 0) != std::string::npos)
335 value_is_vector = true;
336 } else
337 PyErr_Clear();
338 Py_XDECREF(vtype);
339
340 if (eb_call) {
341 PyObject* eb_args;
342 for (int i = 0; /* until break */; ++i) {
343 PyObject* item = getter->get();
344 if (item) {
345 if (value_is_vector && PySequence_Check(item)) {
346 eb_args = PyTuple_New(1);
347 PyTuple_SET_ITEM(eb_args, 0, item);
348 } else if (PyTuple_CheckExact(item)) {
349 eb_args = item;
350 } else if (PyList_CheckExact(item)) {
351 Py_ssize_t isz = PyList_GET_SIZE(item);
352 eb_args = PyTuple_New(isz);
353 for (Py_ssize_t j = 0; j < isz; ++j) {
354 PyObject* iarg = PyList_GET_ITEM(item, j);
355 Py_INCREF(iarg);
356 PyTuple_SET_ITEM(eb_args, j, iarg);
357 }
358 Py_DECREF(item);
359 } else {
360 Py_DECREF(item);
361 PyErr_Format(PyExc_TypeError, "argument %d is not a tuple or list", i);
362 fill_ok = false;
363 break;
364 }
365 PyObject* ebres = PyObject_CallObject(eb_call, eb_args);
366 Py_DECREF(eb_args);
367 if (!ebres) {
368 fill_ok = false;
369 break;
370 }
371 Py_DECREF(ebres);
372 } else {
373 if (PyErr_Occurred()) {
374 if (!(PyErr_ExceptionMatches(PyExc_IndexError) ||
375 PyErr_ExceptionMatches(PyExc_StopIteration)))
376 fill_ok = false;
377 else { PyErr_Clear(); }
378 }
379 break;
380 }
381 }
382 Py_DECREF(eb_call);
383 }
384 } else {
385 // use push_back to add the vector entries one by one
386 PyObject* pb_call = PyObject_GetAttrString(self, (char*)"push_back");
387 if (pb_call) {
388 for (;;) {
389 PyObject* item = getter->get();
390 if (item) {
391 PyObject* pbres = PyObject_CallFunctionObjArgs(pb_call, item, nullptr);
392 Py_DECREF(item);
393 if (!pbres) {
394 fill_ok = false;
395 break;
396 }
397 Py_DECREF(pbres);
398 } else {
399 if (PyErr_Occurred()) {
400 if (!(PyErr_ExceptionMatches(PyExc_IndexError) ||
401 PyErr_ExceptionMatches(PyExc_StopIteration)))
402 fill_ok = false;
403 else { PyErr_Clear(); }
404 }
405 break;
406 }
407 }
408 Py_DECREF(pb_call);
409 }
410 }
411 Py_XDECREF(fi);
412 delete getter;
413
414 if (!fill_ok) {
415 Py_DECREF(result);
416 return nullptr;
417 }
418
419 return result;
420 }
421
422// The given argument wasn't iterable: simply forward to regular constructor
423 PyObject* realInit = PyObject_GetAttrString(self, "__real_init");
424 if (realInit) {
425 PyObject* result = PyObject_Call(realInit, args, nullptr);
426 Py_DECREF(realInit);
427 return result;
428 }
429
430 return nullptr;
431}
432
433//---------------------------------------------------------------------------
434PyObject* VectorData(PyObject* self, PyObject*)
435{
436 PyObject* pydata = CallPyObjMethod(self, "__real_data");
437 if (!LowLevelView_Check(pydata)) return pydata;
438
439 PyObject* pylen = PyObject_CallMethodObjArgs(self, PyStrings::gSize, nullptr);
440 if (!pylen) {
441 PyErr_Clear();
442 return pydata;
443 }
444
445 long clen = PyInt_AsLong(pylen);
446 Py_DECREF(pylen);
447
448// TODO: should be a LowLevelView helper
449 Py_buffer& bi = ((LowLevelView*)pydata)->fBufInfo;
450 bi.len = clen * bi.itemsize;
451 if (bi.ndim == 1 && bi.shape)
452 bi.shape[0] = clen;
453
454 return pydata;
455}
456
457
458//-----------------------------------------------------------------------------
459static PyObject* vector_iter(PyObject* v) {
460 vectoriterobject* vi = PyObject_GC_New(vectoriterobject, &VectorIter_Type);
461 if (!vi) return nullptr;
462
463 Py_INCREF(v);
464 vi->ii_container = v;
465 vi->vi_flags = v->ob_refcnt <= 2 ? 1 : 0; // 2, b/c of preceding INCREF
466
467 PyObject* pyvalue_type = PyObject_GetAttrString((PyObject*)Py_TYPE(v), "value_type");
468 PyObject* pyvalue_size = PyObject_GetAttrString((PyObject*)Py_TYPE(v), "value_size");
469
470 vi->vi_klass = 0;
471 if (pyvalue_type && pyvalue_size) {
472 PyObject* pydata = CallPyObjMethod(v, "data");
473 if (!pydata || Utility::GetBuffer(pydata, '*', 1, vi->vi_data, false) == 0) {
474 if (CPPInstance_Check(pydata)) {
475 vi->vi_data = ((CPPInstance*)pydata)->GetObjectRaw();
476 vi->vi_klass = ((CPPInstance*)pydata)->ObjectIsA(false);
477 } else
478 vi->vi_data = nullptr;
479 }
480 Py_XDECREF(pydata);
481
482 vi->vi_converter = vi->vi_klass ? nullptr : CPyCppyy::CreateConverter(CPyCppyy_PyText_AsString(pyvalue_type));
483 vi->vi_stride = PyLong_AsLong(pyvalue_size);
484 } else {
485 PyErr_Clear();
486 vi->vi_data = nullptr;
487 vi->vi_converter = nullptr;
488 vi->vi_stride = 0;
489 }
490
491 Py_XDECREF(pyvalue_size);
492 Py_XDECREF(pyvalue_type);
493
494 vi->ii_pos = 0;
495 vi->ii_len = PySequence_Size(v);
496
497 PyObject_GC_Track(vi);
498 return (PyObject*)vi;
499}
500
501PyObject* VectorGetItem(CPPInstance* self, PySliceObject* index)
502{
503// Implement python's __getitem__ for std::vector<>s.
504 if (PySlice_Check(index)) {
505 if (!self->GetObject()) {
506 PyErr_SetString(PyExc_TypeError, "unsubscriptable object");
507 return nullptr;
508 }
509
510 PyObject* pyclass = (PyObject*)Py_TYPE((PyObject*)self);
511 PyObject* nseq = PyObject_CallObject(pyclass, nullptr);
512
513 Py_ssize_t start, stop, step;
514 PySlice_GetIndices((CPyCppyy_PySliceCast)index, PyObject_Length((PyObject*)self), &start, &stop, &step);
515
516 const Py_ssize_t nlen = PySequence_Size((PyObject*)self);
517 if (!AdjustSlice(nlen, start, stop, step))
518 return nseq;
519
520 const Py_ssize_t sign = step < 0 ? -1 : 1;
521 for (Py_ssize_t i = start; i*sign < stop*sign; i += step) {
522 PyObject* pyidx = PyInt_FromSsize_t(i);
523 PyObject* item = PyObject_CallMethodObjArgs((PyObject*)self, PyStrings::gGetNoCheck, pyidx, nullptr);
524 CallPyObjMethod(nseq, "push_back", item);
525 Py_DECREF(item);
526 Py_DECREF(pyidx);
527 }
528
529 return nseq;
530 }
531
532 return CallSelfIndex(self, (PyObject*)index, PyStrings::gGetNoCheck);
533}
534
535
536static Cppyy::TCppType_t sVectorBoolTypeID = (Cppyy::TCppType_t)0;
537
538PyObject* VectorBoolGetItem(CPPInstance* self, PyObject* idx)
539{
540// std::vector<bool> is a special-case in C++, and its return type depends on
541// the compiler: treat it special here as well
542 if (!CPPInstance_Check(self) || self->ObjectIsA() != sVectorBoolTypeID) {
543 PyErr_Format(PyExc_TypeError,
544 "require object of type std::vector<bool>, but %s given",
545 Cppyy::GetScopedFinalName(self->ObjectIsA()).c_str());
546 return nullptr;
547 }
548
549 if (!self->GetObject()) {
550 PyErr_SetString(PyExc_TypeError, "unsubscriptable object");
551 return nullptr;
552 }
553
554 if (PySlice_Check(idx)) {
555 PyObject* pyclass = (PyObject*)Py_TYPE((PyObject*)self);
556 PyObject* nseq = PyObject_CallObject(pyclass, nullptr);
557
558 Py_ssize_t start, stop, step;
559 PySlice_GetIndices((CPyCppyy_PySliceCast)idx, PyObject_Length((PyObject*)self), &start, &stop, &step);
560 const Py_ssize_t nlen = PySequence_Size((PyObject*)self);
561 if (!AdjustSlice(nlen, start, stop, step))
562 return nseq;
563
564 const Py_ssize_t sign = step < 0 ? -1 : 1;
565 for (Py_ssize_t i = start; i*sign < stop*sign; i += step) {
566 PyObject* pyidx = PyInt_FromSsize_t(i);
567 PyObject* item = PyObject_CallMethodObjArgs((PyObject*)self, PyStrings::gGetItem, pyidx, nullptr);
568 CallPyObjMethod(nseq, "push_back", item);
569 Py_DECREF(item);
570 Py_DECREF(pyidx);
571 }
572
573 return nseq;
574 }
575
576 PyObject* pyindex = PyStyleIndex((PyObject*)self, idx);
577 if (!pyindex)
578 return nullptr;
579
580 int index = (int)PyLong_AsLong(pyindex);
581 Py_DECREF(pyindex);
582
583// get hold of the actual std::vector<bool> (no cast, as vector is never a base)
584 std::vector<bool>* vb = (std::vector<bool>*)self->GetObject();
585
586// finally, return the value
587 if (bool((*vb)[index]))
590}
591
592PyObject* VectorBoolSetItem(CPPInstance* self, PyObject* args)
593{
594// std::vector<bool> is a special-case in C++, and its return type depends on
595// the compiler: treat it special here as well
596 if (!CPPInstance_Check(self) || self->ObjectIsA() != sVectorBoolTypeID) {
597 PyErr_Format(PyExc_TypeError,
598 "require object of type std::vector<bool>, but %s given",
599 Cppyy::GetScopedFinalName(self->ObjectIsA()).c_str());
600 return nullptr;
601 }
602
603 if (!self->GetObject()) {
604 PyErr_SetString(PyExc_TypeError, "unsubscriptable object");
605 return nullptr;
606 }
607
608 int bval = 0; PyObject* idx = nullptr;
609 if (!PyArg_ParseTuple(args, const_cast<char*>("Oi:__setitem__"), &idx, &bval))
610 return nullptr;
611
612 PyObject* pyindex = PyStyleIndex((PyObject*)self, idx);
613 if (!pyindex)
614 return nullptr;
615
616 int index = (int)PyLong_AsLong(pyindex);
617 Py_DECREF(pyindex);
618
619// get hold of the actual std::vector<bool> (no cast, as vector is never a base)
620 std::vector<bool>* vb = (std::vector<bool>*)self->GetObject();
621
622// finally, set the value
623 (*vb)[index] = (bool)bval;
624
626}
627
628//- map behavior as primitives ------------------------------------------------
629PyObject* MapContains(PyObject* self, PyObject* obj)
630{
631// Implement python's __contains__ for std::map<>s
632 PyObject* result = nullptr;
633
634 PyObject* iter = CallPyObjMethod(self, "find", obj);
635 if (CPPInstance_Check(iter)) {
636 PyObject* end = PyObject_CallMethodObjArgs(self, PyStrings::gEnd, nullptr);
637 if (CPPInstance_Check(end)) {
638 if (!PyObject_RichCompareBool(iter, end, Py_EQ)) {
639 Py_INCREF(Py_True);
640 result = Py_True;
641 }
642 }
643 Py_XDECREF(end);
644 }
645 Py_XDECREF(iter);
646
647 if (!result) {
648 PyErr_Clear(); // e.g. wrong argument type, which should always lead to False
649 Py_INCREF(Py_False);
650 result = Py_False;
651 }
652
653 return result;
654}
655
656//- STL container iterator support --------------------------------------------
657PyObject* StlSequenceIter(PyObject* self)
658{
659// Implement python's __iter__ for std::iterator<>s
660 PyObject* iter = PyObject_CallMethodObjArgs(self, PyStrings::gBegin, nullptr);
661 if (iter) {
662 PyObject* end = PyObject_CallMethodObjArgs(self, PyStrings::gEnd, nullptr);
663 if (end)
664 PyObject_SetAttr(iter, PyStrings::gEnd, end);
665 Py_XDECREF(end);
666
667 // add iterated collection as attribute so its refcount stays >= 1 while it's being iterated over
668 PyObject_SetAttr(iter, CPyCppyy_PyText_FromString("_collection"), self);
669 }
670 return iter;
671}
672
673//- generic iterator support over a sequence with operator[] and size ---------
674//-----------------------------------------------------------------------------
675static PyObject* index_iter(PyObject* c) {
676 indexiterobject* ii = PyObject_GC_New(indexiterobject, &IndexIter_Type);
677 if (!ii) return nullptr;
678
679 Py_INCREF(c);
680 ii->ii_container = c;
681 ii->ii_pos = 0;
682 ii->ii_len = PySequence_Size(c);
683
684 PyObject_GC_Track(ii);
685 return (PyObject*)ii;
686}
687
688
689//- safe indexing for STL-like vector w/o iterator dictionaries ---------------
690/* replaced by indexiterobject iteration, but may still have some future use ...
691PyObject* CheckedGetItem(PyObject* self, PyObject* obj)
692{
693// Implement a generic python __getitem__ for STL-like classes that are missing the
694// reflection info for their iterators. This is then used for iteration by means of
695// consecutive indeces, it such index is of integer type.
696 Py_ssize_t size = PySequence_Size(self);
697 Py_ssize_t idx = PyInt_AsSsize_t(obj);
698 if ((size == (Py_ssize_t)-1 || idx == (Py_ssize_t)-1) && PyErr_Occurred()) {
699 // argument conversion problem: let method itself resolve anew and report
700 PyErr_Clear();
701 return PyObject_CallMethodObjArgs(self, PyStrings::gGetNoCheck, obj, nullptr);
702 }
703
704 bool inbounds = false;
705 if (idx < 0) idx += size;
706 if (0 <= idx && 0 <= size && idx < size)
707 inbounds = true;
708
709 if (inbounds)
710 return PyObject_CallMethodObjArgs(self, PyStrings::gGetNoCheck, obj, nullptr);
711 else
712 PyErr_SetString( PyExc_IndexError, "index out of range" );
713
714 return nullptr;
715}*/
716
717//- pair as sequence to allow tuple unpacking --------------------------------
718PyObject* PairUnpack(PyObject* self, PyObject* pyindex)
719{
720// For std::map<> iteration, unpack std::pair<>s into tuples for the loop.
721 long idx = PyLong_AsLong(pyindex);
722 if (idx == -1 && PyErr_Occurred())
723 return nullptr;
724
725 if (!CPPInstance_Check(self) || !((CPPInstance*)self)->GetObject()) {
726 PyErr_SetString(PyExc_TypeError, "unsubscriptable object");
727 return nullptr;
728 }
729
730 if ((int)idx == 0)
731 return PyObject_GetAttr(self, PyStrings::gFirst);
732 else if ((int)idx == 1)
733 return PyObject_GetAttr(self, PyStrings::gSecond);
734
735// still here? Trigger stop iteration
736 PyErr_SetString(PyExc_IndexError, "out of bounds");
737 return nullptr;
738}
739
740//- simplistic len() functions -----------------------------------------------
741PyObject* ReturnTwo(CPPInstance*, PyObject*) {
742 return PyInt_FromLong(2);
743}
744
745
746//- shared_ptr behavior --------------------------------------------------------
747PyObject* SharedPtrInit(PyObject* self, PyObject* args, PyObject* /* kwds */)
748{
749// since the shared pointer will take ownership, we need to relinquish it
750 PyObject* realInit = PyObject_GetAttrString(self, "__real_init");
751 if (realInit) {
752 PyObject* result = PyObject_Call(realInit, args, nullptr);
753 Py_DECREF(realInit);
754 if (result && PyTuple_GET_SIZE(args) == 1 && CPPInstance_Check(PyTuple_GET_ITEM(args, 0)))
755 PyObject_SetAttrString(PyTuple_GET_ITEM(args, 0), "__python_owns__", Py_False);
756 return result;
757 }
758 return nullptr;
759}
760
761
762//- string behavior as primitives --------------------------------------------
763#if PY_VERSION_HEX >= 0x03000000
764// TODO: this is wrong, b/c it doesn't order
765static int PyObject_Compare(PyObject* one, PyObject* other) {
766 return !PyObject_RichCompareBool(one, other, Py_EQ);
767}
768#endif
769static inline PyObject* CPyCppyy_PyString_FromCppString(std::string* s) {
770 return CPyCppyy_PyText_FromStringAndSize(s->c_str(), s->size());
771}
772
773static inline PyObject* CPyCppyy_PyString_FromCppString(std::wstring* s) {
774 return PyUnicode_FromWideChar(s->c_str(), s->size());
775}
776
777#define CPPYY_IMPL_STRING_PYTHONIZATION(type, name) \
778static PyObject* name##StringGetData(PyObject* self) \
779{ \
780 if (CPyCppyy::CPPInstance_Check(self)) { \
781 type* obj = ((type*)((CPPInstance*)self)->GetObject()); \
782 if (obj) { \
783 return CPyCppyy_PyString_FromCppString(obj); \
784 } else { \
785 return CPPInstance_Type.tp_str(self); \
786 } \
787 } \
788 PyErr_Format(PyExc_TypeError, "object mismatch (%s expected)", #type); \
789 return nullptr; \
790} \
791 \
792PyObject* name##StringRepr(PyObject* self) \
793{ \
794 PyObject* data = name##StringGetData(self); \
795 if (data) { \
796 PyObject* repr = PyObject_Repr(data); \
797 Py_DECREF(data); \
798 return repr; \
799 } \
800 return nullptr; \
801} \
802 \
803PyObject* name##StringIsEqual(PyObject* self, PyObject* obj) \
804{ \
805 PyObject* data = name##StringGetData(self); \
806 if (data) { \
807 PyObject* result = PyObject_RichCompare(data, obj, Py_EQ); \
808 Py_DECREF(data); \
809 return result; \
810 } \
811 return nullptr; \
812} \
813 \
814PyObject* name##StringIsNotEqual(PyObject* self, PyObject* obj) \
815{ \
816 PyObject* data = name##StringGetData(self); \
817 if (data) { \
818 PyObject* result = PyObject_RichCompare(data, obj, Py_NE); \
819 Py_DECREF(data); \
820 return result; \
821 } \
822 return nullptr; \
823}
824
825// Only define StlStringCompare:
826#define CPPYY_IMPL_STRING_PYTHONIZATION_CMP(type, name) \
827CPPYY_IMPL_STRING_PYTHONIZATION(type, name) \
828PyObject* name##StringCompare(PyObject* self, PyObject* obj) \
829{ \
830 PyObject* data = name##StringGetData(self); \
831 int result = 0; \
832 if (data) { \
833 result = PyObject_Compare(data, obj); \
834 Py_DECREF(data); \
835 } \
836 if (PyErr_Occurred()) \
837 return nullptr; \
838 return PyInt_FromLong(result); \
839}
840
842CPPYY_IMPL_STRING_PYTHONIZATION_CMP(std::wstring, StlW)
843
844
845//- STL iterator behavior ----------------------------------------------------
846PyObject* StlIterNext(PyObject* self)
847{
848// Python iterator protocol __next__ for STL forward iterators.
849 PyObject* next = nullptr;
850 PyObject* last = PyObject_GetAttr(self, PyStrings::gEnd);
851
852 if (last) {
853 // handle special case of empty container (i.e. self is end)
854 if (PyObject_RichCompareBool(last, self, Py_EQ) == 0) {
855 // first, get next from the _current_ iterator as internal state may change
856 // when call post or pre increment
857 next = PyObject_CallMethodObjArgs(self, PyStrings::gDeref, nullptr);
858 if (!next) PyErr_Clear();
859
860 // use postinc, even as the C++11 range-based for loops prefer preinc b/c
861 // that allows the current value from the iterator to be had from __deref__,
862 // an issue that does not come up in C++
863 static PyObject* dummy = PyInt_FromLong(1l);
864 PyObject* iter = PyObject_CallMethodObjArgs(self, PyStrings::gPostInc, dummy, nullptr);
865 if (!iter) {
866 // allow preinc, as in that case likely __deref__ is not defined and it
867 // is the iterator rather that is returned in the loop
868 PyErr_Clear();
869 iter = PyObject_CallMethodObjArgs(self, PyStrings::gPreInc, nullptr);
870 }
871 if (iter) {
872 // prefer != as per C++11 range-based for
873 int isNotEnd = PyObject_RichCompareBool(last, iter, Py_NE);
874 if (isNotEnd && !next) {
875 // if no dereference, continue iterating over the iterator
876 Py_INCREF(iter);
877 next = iter;
878 }
879 Py_DECREF(iter);
880 } else {
881 // fail current next, even if available
882 Py_XDECREF(next);
883 next = nullptr;
884 }
885 } else {
886 PyErr_SetString(PyExc_StopIteration, "");
887 }
888 Py_DECREF(last);
889 }
890
891 if (!next) PyErr_SetString(PyExc_StopIteration, "");
892 return next;
893}
894
895
896//- STL complex<T> behavior --------------------------------------------------
897#define COMPLEX_METH_GETSET(name, cppname) \
898static PyObject* name##ComplexGet(PyObject* self, void*) { \
899 return PyObject_CallMethodObjArgs(self, cppname, nullptr); \
900} \
901static int name##ComplexSet(PyObject* self, PyObject* value, void*) { \
902 PyObject* result = PyObject_CallMethodObjArgs(self, cppname, value, nullptr);\
903 if (result) { \
904 Py_DECREF(result); \
905 return 0; \
906 } \
907 return -1; \
908} \
909PyGetSetDef name##Complex{(char*)#name, (getter)name##ComplexGet, (setter)name##ComplexSet, nullptr, nullptr};
910
913
914static PyObject* ComplexComplex(PyObject* self) {
915 PyObject* real = PyObject_CallMethodObjArgs(self, PyStrings::gCppReal, nullptr);
916 if (!real) return nullptr;
917 double r = PyFloat_AsDouble(real);
918 Py_DECREF(real);
919 if (r == -1. && PyErr_Occurred())
920 return nullptr;
921
922 PyObject* imag = PyObject_CallMethodObjArgs(self, PyStrings::gCppImag, nullptr);
923 if (!imag) return nullptr;
924 double i = PyFloat_AsDouble(imag);
925 Py_DECREF(imag);
926 if (i == -1. && PyErr_Occurred())
927 return nullptr;
928
929 return PyComplex_FromDoubles(r, i);
930}
931
932static PyObject* ComplexRepr(PyObject* self) {
933 PyObject* real = PyObject_CallMethodObjArgs(self, PyStrings::gCppReal, nullptr);
934 if (!real) return nullptr;
935 double r = PyFloat_AsDouble(real);
936 Py_DECREF(real);
937 if (r == -1. && PyErr_Occurred())
938 return nullptr;
939
940 PyObject* imag = PyObject_CallMethodObjArgs(self, PyStrings::gCppImag, nullptr);
941 if (!imag) return nullptr;
942 double i = PyFloat_AsDouble(imag);
943 Py_DECREF(imag);
944 if (i == -1. && PyErr_Occurred())
945 return nullptr;
946
947 std::ostringstream s;
948 s << '(' << r << '+' << i << "j)";
949 return CPyCppyy_PyText_FromString(s.str().c_str());
950}
951
952static PyObject* ComplexDRealGet(CPPInstance* self, void*)
953{
954 return PyFloat_FromDouble(((std::complex<double>*)self->GetObject())->real());
955}
956
957static int ComplexDRealSet(CPPInstance* self, PyObject* value, void*)
958{
959 double d = PyFloat_AsDouble(value);
960 if (d == -1.0 && PyErr_Occurred())
961 return -1;
962 ((std::complex<double>*)self->GetObject())->real(d);
963 return 0;
964}
965
966PyGetSetDef ComplexDReal{(char*)"real", (getter)ComplexDRealGet, (setter)ComplexDRealSet, nullptr, nullptr};
967
968
969static PyObject* ComplexDImagGet(CPPInstance* self, void*)
970{
971 return PyFloat_FromDouble(((std::complex<double>*)self->GetObject())->imag());
972}
973
974static int ComplexDImagSet(CPPInstance* self, PyObject* value, void*)
975{
976 double d = PyFloat_AsDouble(value);
977 if (d == -1.0 && PyErr_Occurred())
978 return -1;
979 ((std::complex<double>*)self->GetObject())->imag(d);
980 return 0;
981}
982
983PyGetSetDef ComplexDImag{(char*)"imag", (getter)ComplexDImagGet, (setter)ComplexDImagSet, nullptr, nullptr};
984
985static PyObject* ComplexDComplex(CPPInstance* self)
986{
987 double r = ((std::complex<double>*)self->GetObject())->real();
988 double i = ((std::complex<double>*)self->GetObject())->imag();
989 return PyComplex_FromDoubles(r, i);
990}
991
992
993} // unnamed namespace
994
995
996//- public functions ---------------------------------------------------------
997namespace CPyCppyy {
998 std::set<std::string> gIteratorTypes;
999}
1000
1001bool CPyCppyy::Pythonize(PyObject* pyclass, const std::string& name)
1002{
1003// Add pre-defined pythonizations (for STL and ROOT) to classes based on their
1004// signature and/or class name.
1005 if (!pyclass)
1006 return false;
1007
1008 CPPScope* klass = (CPPScope*)pyclass;
1009
1010//- method name based pythonization ------------------------------------------
1011
1012// for smart pointer style classes that are otherwise not known as such; would
1013// prefer operator-> as that returns a pointer (which is simpler since it never
1014// has to deal with ref-assignment), but operator* plays better with STL iters
1015// and algorithms
1016 if (HasAttrDirect(pyclass, PyStrings::gDeref) && !Cppyy::IsSmartPtr(klass->fCppType))
1017 Utility::AddToClass(pyclass, "__getattr__", (PyCFunction)DeRefGetAttr, METH_O);
1018 else if (HasAttrDirect(pyclass, PyStrings::gFollow) && !Cppyy::IsSmartPtr(klass->fCppType))
1019 Utility::AddToClass(pyclass, "__getattr__", (PyCFunction)FollowGetAttr, METH_O);
1020
1021// for STL containers, and user classes modeled after them
1022 if (HasAttrDirect(pyclass, PyStrings::gSize))
1023 Utility::AddToClass(pyclass, "__len__", "size");
1024
1025 if (!IsTemplatedSTLClass(name, "vector") && // vector is dealt with below
1026 !((PyTypeObject*)pyclass)->tp_iter) {
1027 if (HasAttrDirect(pyclass, PyStrings::gBegin) && HasAttrDirect(pyclass, PyStrings::gEnd)) {
1028 // obtain the name of the return type
1029 const auto& v = Cppyy::GetMethodIndicesFromName(klass->fCppType, "begin");
1030 if (!v.empty()) {
1031 // check return type; if not explicitly an iterator, add it to the "known" return
1032 // types to add the "next" method on use
1034 const std::string& resname = Cppyy::GetMethodResultType(meth);
1035 if (Cppyy::GetScope(resname)) {
1036 if (resname.find("iterator") == std::string::npos)
1037 gIteratorTypes.insert(resname);
1038
1039 // install iterator protocol a la STL
1040 ((PyTypeObject*)pyclass)->tp_iter = (getiterfunc)StlSequenceIter;
1041 Utility::AddToClass(pyclass, "__iter__", (PyCFunction)StlSequenceIter, METH_NOARGS);
1042 }
1043 }
1044 }
1045 if (!((PyTypeObject*)pyclass)->tp_iter && // no iterator resolved
1046 HasAttrDirect(pyclass, PyStrings::gGetItem) && HasAttrDirect(pyclass, PyStrings::gLen)) {
1047 // Python will iterate over __getitem__ using integers, but C++ operator[] will never raise
1048 // a StopIteration. A checked getitem (raising IndexError if beyond size()) works in some
1049 // cases but would mess up if operator[] is meant to implement an associative container. So,
1050 // this has to be implemented as an interator protocol.
1051 ((PyTypeObject*)pyclass)->tp_iter = (getiterfunc)index_iter;
1052 Utility::AddToClass(pyclass, "__iter__", (PyCFunction)index_iter, METH_NOARGS);
1053 }
1054 }
1055
1056// operator==/!= are used in op_richcompare of CPPInstance, which subsequently allows
1057// comparisons to None; if no operator is available, a hook is installed for lazy
1058// lookups in the global and/or class namespace
1059 if (HasAttrDirect(pyclass, PyStrings::gEq, true)) {
1060 PyObject* cppol = PyObject_GetAttr(pyclass, PyStrings::gEq);
1061 if (!klass->fOperators) klass->fOperators = new Utility::PyOperators();
1062 klass->fOperators->fEq = cppol;
1063 // re-insert the forwarding __eq__ from the CPPInstance in case there was a Python-side
1064 // override in the base class
1065 static PyObject* top_eq = nullptr;
1066 if (!top_eq) {
1067 PyObject* top_cls = PyObject_GetAttrString(gThisModule, "CPPInstance");
1068 top_eq = PyObject_GetAttr(top_cls, PyStrings::gEq);
1069 Py_DECREF(top_eq); // make it borrowed
1070 Py_DECREF(top_cls);
1071 }
1072 PyObject_SetAttr(pyclass, PyStrings::gEq, top_eq);
1073 }
1074
1075 if (HasAttrDirect(pyclass, PyStrings::gNe, true)) {
1076 PyObject* cppol = PyObject_GetAttr(pyclass, PyStrings::gNe);
1077 if (!klass->fOperators) klass->fOperators = new Utility::PyOperators();
1078 klass->fOperators->fNe = cppol;
1079 // re-insert the forwarding __ne__ (same reason as above for __eq__)
1080 static PyObject* top_ne = nullptr;
1081 if (!top_ne) {
1082 PyObject* top_cls = PyObject_GetAttrString(gThisModule, "CPPInstance");
1083 top_ne = PyObject_GetAttr(top_cls, PyStrings::gNe);
1084 Py_DECREF(top_ne); // make it borrowed
1085 Py_DECREF(top_cls);
1086 }
1087 PyObject_SetAttr(pyclass, PyStrings::gNe, top_ne);
1088 }
1089
1090
1091//- class name based pythonization -------------------------------------------
1092
1093 if (IsTemplatedSTLClass(name, "vector")) {
1094
1095 // std::vector<bool> is a special case in C++
1096 if (!sVectorBoolTypeID) sVectorBoolTypeID = (Cppyy::TCppType_t)Cppyy::GetScope("std::vector<bool>");
1097 if (klass->fCppType == sVectorBoolTypeID) {
1098 Utility::AddToClass(pyclass, "__getitem__", (PyCFunction)VectorBoolGetItem, METH_O);
1099 Utility::AddToClass(pyclass, "__setitem__", (PyCFunction)VectorBoolSetItem);
1100 } else {
1101 // constructor that takes python collections
1102 Utility::AddToClass(pyclass, "__real_init", "__init__");
1103 Utility::AddToClass(pyclass, "__init__", (PyCFunction)VectorInit, METH_VARARGS | METH_KEYWORDS);
1104
1105 // data with size
1106 Utility::AddToClass(pyclass, "__real_data", "data");
1107 Utility::AddToClass(pyclass, "data", (PyCFunction)VectorData);
1108
1109 // checked getitem
1110 if (HasAttrDirect(pyclass, PyStrings::gLen)) {
1111 Utility::AddToClass(pyclass, "_getitem__unchecked", "__getitem__");
1112 Utility::AddToClass(pyclass, "__getitem__", (PyCFunction)VectorGetItem, METH_O);
1113 }
1114
1115 // vector-optimized iterator protocol
1116 ((PyTypeObject*)pyclass)->tp_iter = (getiterfunc)vector_iter;
1117
1118 // helpers for iteration
1119 const std::string& vtype = Cppyy::ResolveName(name+"::value_type");
1120 size_t typesz = Cppyy::SizeOf(vtype);
1121 if (typesz) {
1122 PyObject* pyvalue_size = PyLong_FromSsize_t(typesz);
1123 PyObject_SetAttrString(pyclass, "value_size", pyvalue_size);
1124 Py_DECREF(pyvalue_size);
1125
1126 PyObject* pyvalue_type = CPyCppyy_PyText_FromString(vtype.c_str());
1127 PyObject_SetAttrString(pyclass, "value_type", pyvalue_type);
1128 Py_DECREF(pyvalue_type);
1129 }
1130 }
1131 }
1132
1133 else if (IsTemplatedSTLClass(name, "map")) {
1134 Utility::AddToClass(pyclass, "__contains__", (PyCFunction)MapContains, METH_O);
1135 }
1136
1137 else if (IsTemplatedSTLClass(name, "pair")) {
1138 Utility::AddToClass(pyclass, "__getitem__", (PyCFunction)PairUnpack, METH_O);
1139 Utility::AddToClass(pyclass, "__len__", (PyCFunction)ReturnTwo, METH_NOARGS);
1140 }
1141
1142 if (IsTemplatedSTLClass(name, "shared_ptr")) {
1143 Utility::AddToClass(pyclass, "__real_init", "__init__");
1144 Utility::AddToClass(pyclass, "__init__", (PyCFunction)SharedPtrInit, METH_VARARGS | METH_KEYWORDS);
1145 }
1146
1147 else if (name.find("iterator") != std::string::npos || gIteratorTypes.find(name) != gIteratorTypes.end()) {
1148 ((PyTypeObject*)pyclass)->tp_iternext = (iternextfunc)StlIterNext;
1149 Utility::AddToClass(pyclass, CPPYY__next__, (PyCFunction)StlIterNext, METH_NOARGS);
1150 ((PyTypeObject*)pyclass)->tp_iter = (getiterfunc)PyObject_SelfIter;
1151 Utility::AddToClass(pyclass, "__iter__", (PyCFunction)PyObject_SelfIter, METH_NOARGS);
1152 }
1153
1154 else if (name == "string" || name == "std::string") { // TODO: ask backend as well
1155 Utility::AddToClass(pyclass, "__repr__", (PyCFunction)StlStringRepr, METH_NOARGS);
1156 Utility::AddToClass(pyclass, "__str__", (PyCFunction)StlStringGetData, METH_NOARGS);
1157 Utility::AddToClass(pyclass, "__cmp__", (PyCFunction)StlStringCompare, METH_O);
1158 Utility::AddToClass(pyclass, "__eq__", (PyCFunction)StlStringIsEqual, METH_O);
1159 Utility::AddToClass(pyclass, "__ne__", (PyCFunction)StlStringIsNotEqual, METH_O);
1160 }
1161
1162 else if (name == "basic_string<wchar_t,char_traits<wchar_t>,allocator<wchar_t> >" || \
1163 name == "std::basic_string<wchar_t,char_traits<wchar_t>,allocator<wchar_t> >") {
1164 Utility::AddToClass(pyclass, "__repr__", (PyCFunction)StlWStringRepr, METH_NOARGS);
1165 Utility::AddToClass(pyclass, "__str__", (PyCFunction)StlWStringGetData, METH_NOARGS);
1166 Utility::AddToClass(pyclass, "__cmp__", (PyCFunction)StlWStringCompare, METH_O);
1167 Utility::AddToClass(pyclass, "__eq__", (PyCFunction)StlWStringIsEqual, METH_O);
1168 Utility::AddToClass(pyclass, "__ne__", (PyCFunction)StlWStringIsNotEqual, METH_O);
1169 }
1170
1171 else if (name == "complex<double>" || name == "std::complex<double>") {
1172 Utility::AddToClass(pyclass, "__cpp_real", "real");
1173 PyObject_SetAttrString(pyclass, "real", PyDescr_NewGetSet((PyTypeObject*)pyclass, &ComplexDReal));
1174 Utility::AddToClass(pyclass, "__cpp_imag", "imag");
1175 PyObject_SetAttrString(pyclass, "imag", PyDescr_NewGetSet((PyTypeObject*)pyclass, &ComplexDImag));
1176 Utility::AddToClass(pyclass, "__complex__", (PyCFunction)ComplexDComplex, METH_NOARGS);
1177 Utility::AddToClass(pyclass, "__repr__", (PyCFunction)ComplexRepr, METH_NOARGS);
1178 }
1179
1180 else if (IsTemplatedSTLClass(name, "complex")) {
1181 Utility::AddToClass(pyclass, "__cpp_real", "real");
1182 PyObject_SetAttrString(pyclass, "real", PyDescr_NewGetSet((PyTypeObject*)pyclass, &realComplex));
1183 Utility::AddToClass(pyclass, "__cpp_imag", "imag");
1184 PyObject_SetAttrString(pyclass, "imag", PyDescr_NewGetSet((PyTypeObject*)pyclass, &imagComplex));
1185 Utility::AddToClass(pyclass, "__complex__", (PyCFunction)ComplexComplex, METH_NOARGS);
1186 Utility::AddToClass(pyclass, "__repr__", (PyCFunction)ComplexRepr, METH_NOARGS);
1187 }
1188
1189// direct user access; there are two calls here:
1190// - explicit pythonization: won't fall through to the base classes and is preferred if present
1191// - normal pythonization: only called if explicit isn't present, falls through to base classes
1192 bool bUserOk = true; PyObject* res = nullptr;
1194 if (HasAttrDirect(pyclass, PyStrings::gExPythonize)) {
1195 res = PyObject_CallMethodObjArgs(pyclass, PyStrings::gExPythonize, pyclass, pyname, nullptr);
1196 bUserOk = (bool)res;
1197 } else {
1198 PyObject* func = PyObject_GetAttr(pyclass, PyStrings::gPythonize);
1199 if (func) {
1200 res = PyObject_CallFunctionObjArgs(func, pyclass, pyname, nullptr);
1201 Py_DECREF(func);
1202 bUserOk = (bool)res;
1203 } else
1204 PyErr_Clear();
1205 }
1206 if (!bUserOk) {
1207 Py_DECREF(pyname);
1208 return false;
1209 } else {
1210 Py_XDECREF(res);
1211 // pyname handed to tuple below
1212 }
1213
1214// call registered pythonizors, if any
1215 PyObject* args = PyTuple_New(2);
1216 Py_INCREF(pyclass);
1217 PyTuple_SET_ITEM(args, 0, pyclass);
1218
1219 std::string outer_scope = TypeManip::extract_namespace(name);
1220
1221 bool pstatus = true;
1222 auto p = outer_scope.empty() ? gPythonizations.end() : gPythonizations.find(outer_scope);
1223 if (p == gPythonizations.end()) {
1224 p = gPythonizations.find("");
1225 PyTuple_SET_ITEM(args, 1, pyname);
1226 } else {
1227 PyTuple_SET_ITEM(args, 1, CPyCppyy_PyText_FromString(
1228 name.substr(outer_scope.size()+2, std::string::npos).c_str()));
1229 Py_DECREF(pyname);
1230 }
1231
1232 if (p != gPythonizations.end()) {
1233 for (auto pythonizor : p->second) {
1234 PyObject* result = PyObject_CallObject(pythonizor, args);
1235 if (!result) {
1236 // TODO: detail error handling for the pythonizors
1237 pstatus = false;
1238 break;
1239 }
1240 Py_DECREF(result);
1241 }
1242 }
1243
1244 Py_DECREF(args);
1245
1246// phew! all done ...
1247 return pstatus;
1248}
#define Py_TYPE(ob)
Definition: CPyCppyy.h:209
#define Py_RETURN_TRUE
Definition: CPyCppyy.h:285
#define Py_RETURN_FALSE
Definition: CPyCppyy.h:289
#define PyInt_FromSsize_t
Definition: CPyCppyy.h:230
#define CPyCppyy_PyText_FromStringAndSize
Definition: CPyCppyy.h:106
#define PyBytes_Check
Definition: CPyCppyy.h:83
#define PyInt_AsSsize_t
Definition: CPyCppyy.h:229
#define CPyCppyy_PySliceCast
Definition: CPyCppyy.h:202
#define CPyCppyy_PyText_AsString
Definition: CPyCppyy.h:97
#define Py_RETURN_NONE
Definition: CPyCppyy.h:281
#define CPPYY__next__
Definition: CPyCppyy.h:131
#define CPyCppyy_PyText_FromString
Definition: CPyCppyy.h:102
#define CPyCppyy_PyText_Check
Definition: CPyCppyy.h:95
PyFloat_FromDouble
Definition: Converters.cxx:921
PyFloat_AsDouble
Definition: Converters.cxx:921
PyInt_FromLong
Definition: Converters.cxx:858
ROOT::R::TRInterface & r
Definition: Object.C:4
_object PyObject
Definition: PyMethodBase.h:41
#define CPPYY_IMPL_STRING_PYTHONIZATION_CMP(type, name)
Definition: Pythonize.cxx:826
#define COMPLEX_METH_GETSET(name, cppname)
Definition: Pythonize.cxx:897
#define PyObject_LengthHint
Definition: Pythonize.cxx:196
PyObject * CallPyObjMethod(PyObject *obj, const char *meth)
Set of helper functions that are invoked from the C++ implementation of pythonizations.
#define d(i)
Definition: RSha256.hxx:102
#define c(i)
Definition: RSha256.hxx:101
static RooMathCoreReg dummy
static PyObject * PyStyleIndex(PyObject *self, PyObject *index)
char name[80]
Definition: TGX11.cxx:109
#define pyname
Definition: TMCParticle.cxx:19
Cppyy::TCppType_t ObjectIsA(bool check_smart=true) const
Definition: CPPInstance.h:106
Utility::PyOperators * fOperators
Definition: CPPScope.h:56
Cppyy::TCppType_t fCppType
Definition: CPPScope.h:50
PyObject * gDict
Definition: PyStrings.cxx:14
PyObject * gCTypesType
Definition: PyStrings.cxx:29
PyObject * gDeref
Definition: PyStrings.cxx:11
PyObject * gExPythonize
Definition: PyStrings.cxx:57
PyObject * gGetItem
Definition: PyStrings.cxx:18
PyObject * gCppReal
Definition: PyStrings.cxx:50
PyObject * gBegin
Definition: PyStrings.cxx:42
PyObject * gFollow
Definition: PyStrings.cxx:17
PyObject * gPreInc
Definition: PyStrings.cxx:12
PyObject * gPythonize
Definition: PyStrings.cxx:58
PyObject * gTypeCode
Definition: PyStrings.cxx:28
PyObject * gSecond
Definition: PyStrings.cxx:45
PyObject * gPostInc
Definition: PyStrings.cxx:13
PyObject * gCppImag
Definition: PyStrings.cxx:51
PyObject * gSize
Definition: PyStrings.cxx:46
PyObject * gGetNoCheck
Definition: PyStrings.cxx:19
PyObject * gFirst
Definition: PyStrings.cxx:44
std::string extract_namespace(const std::string &name)
Definition: TypeManip.cxx:159
Py_ssize_t GetBuffer(PyObject *pyobject, char tc, int size, void *&buf, bool check=true)
Definition: Utility.cxx:614
bool AddToClass(PyObject *pyclass, const char *label, PyCFunction cfunc, int flags=METH_VARARGS)
Definition: Utility.cxx:169
PyTypeObject VectorIter_Type
bool Pythonize(PyObject *pyclass, const std::string &name)
Definition: Pythonize.cxx:1001
bool CPPOverload_Check(T *object)
Definition: CPPOverload.h:79
std::map< std::string, std::vector< PyObject * > > gPythonizations
bool LowLevelView_Check(T *object)
Definition: LowLevelViews.h:65
bool CPPInstance_Check(T *object)
Definition: CPPInstance.h:118
PyTypeObject IndexIter_Type
PyObject * gThisModule
Definition: API.cxx:32
std::set< std::string > gIteratorTypes
Definition: Pythonize.cxx:998
CPYCPPYY_EXTERN Converter * CreateConverter(const std::string &name, Py_ssize_t *dims=nullptr)
RPY_EXPORTED std::vector< TCppIndex_t > GetMethodIndicesFromName(TCppScope_t scope, const std::string &name)
intptr_t TCppMethod_t
Definition: cpp_cppyy.h:22
TCppScope_t TCppType_t
Definition: cpp_cppyy.h:19
RPY_EXPORTED TCppMethod_t GetMethod(TCppScope_t scope, TCppIndex_t imeth)
RPY_EXPORTED std::string ResolveName(const std::string &cppitem_name)
RPY_EXPORTED size_t SizeOf(TCppType_t klass)
RPY_EXPORTED std::string GetScopedFinalName(TCppType_t type)
RPY_EXPORTED bool IsSmartPtr(TCppType_t type)
RPY_EXPORTED TCppScope_t GetScope(const std::string &scope_name)
RPY_EXPORTED std::string GetMethodResultType(TCppMethod_t)
static constexpr double s
PyObject_HEAD PyObject * ii_container
Definition: CustomPyTypes.h:80
Cppyy::TCppType_t vi_klass
Definition: CustomPyTypes.h:92
CPyCppyy::Converter * vi_converter
Definition: CustomPyTypes.h:91