
On Fri, May 10, 2019 at 12:09 AM Jeroen Demeyer J.Demeyer@ugent.be wrote:
Dear C API lovers,
I have a question about the usual args/kwargs call protocol (as used by PyObject_Call and tp_call): is the "args" tuple supposed to be of type exactly tuple or are tuple subclasses allowed? Same question for "kwargs" and dict.
[ munch ]
I'm writing a native extension module which is also affected by this. I' wrapping a few C functions which take a length argument and a pointer argument. The Python wrapper just takes a sequence argument and, for when the 3rd Python argument is the sequence, extracts the C length argument with
count = PySequenceLength(PyTuple_GetItem(args, 2));
(For those concerned about code safety, PyArg_ParseTuple has already checked that the third argument exists and is a sequence, and the method is flagged as positional arguments only.)
I'm using GetItem because it's a borrowed reference and I don't need to adjust any reference counts.
So, my extension module will break if the args tuple is not compatible with the C API for the builtin tuple type.
This is related I think to Victor's work on the levels of C API compatibility. Is there a requirement that a subclass of a builtin type must implement the entire C API?
(I'd like to say yes, but this isn't production code so it wouldn't be big problem if the consensus is no.)
Also, the implementation of the call protocol has assertions of both forms: you see both assert(PyTuple_Check(args)) and assert(PyTuple_CheckExact(args)) and similar for dict. It seems pretty random whether or not an exact type check is done.
I'm pretty sure that there are code paths that would result in an assertion failure because of this.
So basically my question is: what is the right behavior? Given the implementation of the Python bytecode interpreter, I would be inclined to say that type checks should be exact.
I agree for the current implementation. The internals of the Python interpreter can be much more limited about what they accept.