56 for (
int i = 0; i < (int)fConverters.size(); ++i )
57 delete fConverters[ i ];
72 }
catch ( std::exception&
e ) {
80 std::string exception_type;
81 if (cl) exception_type = cl->
GetName();
85 if (errorCode) exception_type =
typeid(
e).
name();
86 else exception_type = demangled.get();
88 PyObject* pyexc = PyDict_GetItemString( pyUserExcepts, exception_type.c_str() );
91 pyexc = PyDict_GetItemString( pyUserExcepts, (
"std::"+exception_type).c_str() );
95 pyexc = PyDict_GetItemString( pyUserExcepts, (
"ROOT::"+exception_type).c_str() );
97 Py_DECREF( pyUserExcepts );
100 PyErr_Format( pyexc,
"%s",
e.what() );
102 PyErr_Format( PyExc_Exception,
"%s (C++ exception of type %s)",
e.what(), exception_type.c_str() );
106 PyErr_SetString( PyExc_Exception,
"unhandled, unknown C++ exception" );
121 result = CallFast( self, offset, ctxt );
123 PyErr_SetString( PyExc_SystemError,
"problem in C++; program state has been reset" );
137 fConverters.resize( nArgs );
140 for (
size_t iarg = 0; iarg < nArgs; ++iarg ) {
150 ( fullType ==
"const std::string&" || fullType ==
"const std::string &"
151 || fullType ==
"const string&" || fullType ==
"const string &" ) ) {
157 if ( ! fConverters[ iarg ] ) {
158 PyErr_Format( PyExc_TypeError,
"argument type %s not handled", fullType.c_str() );
187 std::stringstream sig; sig <<
"(";
190 for (
size_t iarg = 0; iarg < nArgs; ++iarg ) {
191 if ( ifirst ) sig <<
", ";
196 if ( ! parname.empty() )
197 sig <<
" " << parname;
200 if ( ! defvalue.empty() )
201 sig <<
" = " << defvalue;
214 PyErr_Fetch( &etype, &evalue, &etrace );
216 std::string details =
"";
218 PyObject* descr = PyObject_Str( evalue );
225 Py_XDECREF( evalue ); Py_XDECREF( etrace );
228 PyObject* errtype = etype ? etype : PyExc_TypeError;
229 if ( details.empty() ) {
233 PyErr_Format( errtype,
"%s =>\n %s (%s)",
236 PyErr_Format( errtype,
"%s =>\n %s",
248 fMethod( method ), fScope( scope ), fExecutor( nullptr ), fArgsRequired( -1 ),
255 PyCallable( other ), fMethod( other.fMethod ), fScope( other.fScope )
266 if (
this != &other ) {
293 GetSignatureString().c_str() );
306 for (
size_t iarg = 0; iarg < nArgs; ++iarg ) {
314 if ( strstr( aname.c_str(),
"void*" ) )
317 else if ( strstr( aname.c_str(),
"float" ) )
319 else if ( strstr( aname.c_str(),
"long double" ) )
321 else if ( strstr( aname.c_str(),
"double" ) )
324 else if ( strstr( aname.c_str(),
"bool" ) )
329 if ( aname[ aname.size() - 1 ] ==
'&' )
337 if ( aname ==
"IBaseFunctionMultiDim")
339 else if ( aname ==
"RooAbsReal" )
370 int co_argcount = (int)GetMaxArgs() ;
374 PyObject* co_varnames = PyTuple_New( co_argcount + 1 );
376 for (
int iarg = 0; iarg < co_argcount; ++iarg ) {
379 if ( ! parname.empty() ) {
386 PyTuple_SET_ITEM( co_varnames, iarg + 1, pyspec );
397 if ( iarg >= (
int)GetMaxArgs() )
401 if ( ! defvalue.empty() ) {
406 if ( ! pyval && PyErr_Occurred() ) {
430 if ( fIsInitialized ==
kTRUE )
433 if ( ! InitConverters_() )
436 if ( ! InitExecutor_( fExecutor, ctxt ) )
443 fIsInitialized =
kTRUE;
459 if ( PyTuple_GET_SIZE( args ) != 0 ) {
473 return PyTuple_GetSlice( args, 1, PyTuple_GET_SIZE( args ) );
479 "unbound method %s::%s must be called with a %s instance as first argument",
489 int argc = PyTuple_GET_SIZE( args );
490 int argMax = fConverters.size();
493 if ( argc < fArgsRequired ) {
495 "takes at least %d arguments (%d given)", fArgsRequired, argc ) );
497 }
else if ( argMax < argc ) {
499 "takes at most %d arguments (%d given)", argMax, argc ) );
504 ctxt->
fArgs.resize( argc );
505 for (
int i = 0; i < argc; ++i ) {
506 if ( ! fConverters[ i ]->SetArg(
507 PyTuple_GET_ITEM( args, i ), ctxt->
fArgs[i], ctxt ) ) {
525 result = CallFast( self, offset, ctxt );
528 result = CallSafe( self, offset, ctxt );
535 }
else if ( ! result && PyErr_Occurred() )
547 if ( kwds != 0 && PyDict_Size( kwds ) ) {
548 PyErr_SetString( PyExc_TypeError,
"keyword arguments are not yet supported" );
557 if ( ! ( args = PreProcessArgs( self, args, kwds ) ) )
561 Bool_t bConvertOk = ConvertAndSetArgs( args, ctxt );
564 if ( bConvertOk ==
kFALSE )
572 PyErr_SetString( PyExc_ReferenceError,
"attempt to access a null-pointer" );
580 ptrdiff_t offset = 0;
581 if ( derived && derived != fScope )
588 derived && pyobj->
ObjectIsA() == derived &&
#define PyROOT_PyUnicode_AsString
#define PyROOT_PyUnicode_FromString
#define PyROOT_PyUnicode_FromFormat
void Throw(int code)
If an exception context has been set (using the TRY and RETRY macros) jump back to where it was set.
Cppyy::TCppType_t ObjectIsA() const
virtual Int_t GetPriority()
Method priorities exist (in lieu of true overloading) there to prevent void* or <unknown>* from usurp...
virtual Bool_t Initialize(TCallContext *ctxt=0)
done if cache is already setup
virtual PyObject * GetCoVarNames()
Build a tuple of the argument types/names.
virtual PyObject * GetSignature()
virtual Int_t GetMaxArgs()
virtual ~TMethodHolder()
destructor
virtual PyObject * Execute(void *self, ptrdiff_t offset, TCallContext *ctxt=0)
call the interface method
void Copy_(const TMethodHolder &)
void SetPyError_(PyObject *msg)
helper to report errors in a consistent format (derefs msg)
TMethodHolder & operator=(const TMethodHolder &)
assignment operator
TMethodHolder(Cppyy::TCppScope_t scope, Cppyy::TCppMethod_t method)
virtual PyObject * PreProcessArgs(ObjectProxy *&self, PyObject *args, PyObject *kwds)
verify existence of self, return if ok
virtual PyObject * Call(ObjectProxy *&self, PyObject *args, PyObject *kwds, TCallContext *ctxt=0)
preliminary check in case keywords are accidently used (they are ignored otherwise)
std::string GetSignatureString()
built a signature representation (used for doc strings)
Cppyy::TCppScope_t fScope
std::string GetReturnTypeName()
virtual PyObject * GetArgDefault(Int_t iarg)
get the default value (if any) of argument iarg of this method
virtual Bool_t ConvertAndSetArgs(PyObject *args, TCallContext *ctxt=0)
Cppyy::TCppMethod_t fMethod
void Destroy_() const
destroy executor and argument converters
PyObject * CallFast(void *, ptrdiff_t, TCallContext *)
Helper code to prevent some duplication; this is called from CallSafe() as well as directly from TMet...
Bool_t InitConverters_()
build buffers for argument dispatching
virtual PyObject * GetPrototype()
virtual Bool_t InitExecutor_(TExecutor *&, TCallContext *ctxt=0)
install executor conform to the return type
PyObject * CallSafe(void *, ptrdiff_t, TCallContext *)
Helper code to prevent some code duplication; this code embeds a ROOT "try/catch" block that saves th...
virtual PyObject * GetScopeProxy()
Get or build the scope of this method.
TClass instances represent classes, structs and namespaces in the ROOT type system.
static TClass * GetClass(const char *name, Bool_t load=kTRUE, Bool_t silent=kFALSE)
Static method returning pointer to TClass of the specified class name.
virtual const char * GetName() const
Returns name of object.
ptrdiff_t GetBaseOffset(TCppType_t derived, TCppType_t base, TCppObject_t address, int direction, bool rerror=false)
std::string GetMethodArgType(TCppMethod_t, int iarg)
std::string GetMethodName(TCppMethod_t)
TCppIndex_t GetMethodReqArgs(TCppMethod_t)
std::string ResolveName(const std::string &cppitem_name)
Bool_t IsComplete(const std::string &type_name)
std::string GetScopedFinalName(TCppType_t type)
Bool_t IsConstMethod(TCppMethod_t)
std::string GetMethodArgDefault(TCppMethod_t, int iarg)
TCppIndex_t GetMethodNumArgs(TCppMethod_t)
TCppScope_t GetScope(const std::string &scope_name)
Bool_t IsStaticMethod(TCppMethod_t method)
std::string GetMethodArgName(TCppMethod_t, int iarg)
Bool_t IsBuiltin(const std::string &type_name)
std::string GetMethodResultType(TCppMethod_t)
std::string GetFinalName(TCppType_t type)
Bool_t IsSubtype(TCppType_t derived, TCppType_t base)
PyObject * PyErr_Occurred_WithGIL()
Re-acquire the GIL before calling PyErr_Occurred() in case it has been released; note that the p2....
TExecutor * CreateExecutor(const std::string &fullType, Bool_t manage_smart_ptr=kTRUE)
Bool_t ObjectProxy_Check(T *object)
PyObject * CreateScopeProxy(Cppyy::TCppScope_t)
Convenience function with a lookup first through the known existing proxies.
TConverter * CreateConverter(const std::string &fullType, Long_t size=-1)
R__EXTERN PyObject * gRootModule
Bool_t ManagesSmartPtr(TCallContext *ctxt)
char * DemangleTypeIdName(const std::type_info &ti, int &errorCode)
Demangle in a portable way the type id name.
void Initialize(Bool_t useTMVAStyle=kTRUE)
std::vector< TParameter > fArgs
static ECallFlags sSignalPolicy