mirror of
https://github.com/python/cpython.git
synced 2025-07-24 11:44:31 +00:00
bpo-30061: Check if PyObject_Size()/PySequence_Size()/PyMapping_Size() (#1096)
raised an error. Replace them with using concrete types API that never fails if appropriate.
This commit is contained in:
parent
c209b70d61
commit
bf623ae884
14 changed files with 108 additions and 48 deletions
|
@ -543,6 +543,22 @@ class IOTest(unittest.TestCase):
|
||||||
with self.open(support.TESTFN, "r") as f:
|
with self.open(support.TESTFN, "r") as f:
|
||||||
self.assertRaises(TypeError, f.readline, 5.3)
|
self.assertRaises(TypeError, f.readline, 5.3)
|
||||||
|
|
||||||
|
def test_readline_nonsizeable(self):
|
||||||
|
# Issue #30061
|
||||||
|
# Crash when readline() returns an object without __len__
|
||||||
|
class R(self.IOBase):
|
||||||
|
def readline(self):
|
||||||
|
return None
|
||||||
|
self.assertRaises((TypeError, StopIteration), next, R())
|
||||||
|
|
||||||
|
def test_next_nonsizeable(self):
|
||||||
|
# Issue #30061
|
||||||
|
# Crash when __next__() returns an object without __len__
|
||||||
|
class R(self.IOBase):
|
||||||
|
def __next__(self):
|
||||||
|
return None
|
||||||
|
self.assertRaises(TypeError, R().readlines, 1)
|
||||||
|
|
||||||
def test_raw_bytes_io(self):
|
def test_raw_bytes_io(self):
|
||||||
f = self.BytesIO()
|
f = self.BytesIO()
|
||||||
self.write_ops(f)
|
self.write_ops(f)
|
||||||
|
|
|
@ -313,6 +313,11 @@ Extension Modules
|
||||||
Library
|
Library
|
||||||
-------
|
-------
|
||||||
|
|
||||||
|
- bpo-30061: Fixed crashes in IOBase methods __next__() and readlines() when
|
||||||
|
readline() or __next__() respectively return non-sizeable object.
|
||||||
|
Fixed possible other errors caused by not checking results of PyObject_Size(),
|
||||||
|
PySequence_Size(), or PyMapping_Size().
|
||||||
|
|
||||||
- bpo-10076: Compiled regular expression and match objects in the re module
|
- bpo-10076: Compiled regular expression and match objects in the re module
|
||||||
now support copy.copy() and copy.deepcopy() (they are considered atomic).
|
now support copy.copy() and copy.deepcopy() (they are considered atomic).
|
||||||
|
|
||||||
|
|
|
@ -5143,7 +5143,7 @@ comerror_init(PyObject *self, PyObject *args, PyObject *kwds)
|
||||||
if (!PyArg_ParseTuple(args, "OOO:COMError", &hresult, &text, &details))
|
if (!PyArg_ParseTuple(args, "OOO:COMError", &hresult, &text, &details))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
a = PySequence_GetSlice(args, 1, PySequence_Size(args));
|
a = PySequence_GetSlice(args, 1, PyTuple_GET_SIZE(args));
|
||||||
if (!a)
|
if (!a)
|
||||||
return -1;
|
return -1;
|
||||||
status = PyObject_SetAttrString(self, "args", a);
|
status = PyObject_SetAttrString(self, "args", a);
|
||||||
|
|
|
@ -1878,7 +1878,7 @@ element_ass_subscr(PyObject* self_, PyObject* item, PyObject* value)
|
||||||
);
|
);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
newlen = PySequence_Size(seq);
|
newlen = PySequence_Fast_GET_SIZE(seq);
|
||||||
|
|
||||||
if (step != 1 && newlen != slicelen)
|
if (step != 1 && newlen != slicelen)
|
||||||
{
|
{
|
||||||
|
@ -3660,7 +3660,7 @@ _elementtree_XMLParser__setevents_impl(XMLParserObject *self,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < PySequence_Size(events_seq); ++i) {
|
for (i = 0; i < PySequence_Fast_GET_SIZE(events_seq); ++i) {
|
||||||
PyObject *event_name_obj = PySequence_Fast_GET_ITEM(events_seq, i);
|
PyObject *event_name_obj = PySequence_Fast_GET_ITEM(events_seq, i);
|
||||||
const char *event_name = NULL;
|
const char *event_name = NULL;
|
||||||
if (PyUnicode_Check(event_name_obj)) {
|
if (PyUnicode_Check(event_name_obj)) {
|
||||||
|
|
|
@ -618,7 +618,8 @@ iobase_iternext(PyObject *self)
|
||||||
if (line == NULL)
|
if (line == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
if (PyObject_Size(line) == 0) {
|
if (PyObject_Size(line) <= 0) {
|
||||||
|
/* Error or empty */
|
||||||
Py_DECREF(line);
|
Py_DECREF(line);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -670,6 +671,7 @@ _io__IOBase_readlines_impl(PyObject *self, Py_ssize_t hint)
|
||||||
}
|
}
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
|
Py_ssize_t line_length;
|
||||||
PyObject *line = PyIter_Next(it);
|
PyObject *line = PyIter_Next(it);
|
||||||
if (line == NULL) {
|
if (line == NULL) {
|
||||||
if (PyErr_Occurred()) {
|
if (PyErr_Occurred()) {
|
||||||
|
@ -683,11 +685,14 @@ _io__IOBase_readlines_impl(PyObject *self, Py_ssize_t hint)
|
||||||
Py_DECREF(line);
|
Py_DECREF(line);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
length += PyObject_Size(line);
|
line_length = PyObject_Size(line);
|
||||||
Py_DECREF(line);
|
Py_DECREF(line);
|
||||||
|
if (line_length < 0) {
|
||||||
if (length > hint)
|
goto error;
|
||||||
|
}
|
||||||
|
if (line_length > hint - length)
|
||||||
break;
|
break;
|
||||||
|
length += line_length;
|
||||||
}
|
}
|
||||||
|
|
||||||
Py_DECREF(it);
|
Py_DECREF(it);
|
||||||
|
|
|
@ -722,17 +722,22 @@ getenvironment(PyObject* environment)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
envsize = PyMapping_Length(environment);
|
|
||||||
|
|
||||||
keys = PyMapping_Keys(environment);
|
keys = PyMapping_Keys(environment);
|
||||||
values = PyMapping_Values(environment);
|
values = PyMapping_Values(environment);
|
||||||
if (!keys || !values)
|
if (!keys || !values)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
envsize = PySequence_Fast_GET_SIZE(keys);
|
||||||
|
if (PySequence_Fast_GET_SIZE(values) != envsize) {
|
||||||
|
PyErr_SetString(PyExc_RuntimeError,
|
||||||
|
"environment changed size during iteration");
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
totalsize = 1; /* trailing null character */
|
totalsize = 1; /* trailing null character */
|
||||||
for (i = 0; i < envsize; i++) {
|
for (i = 0; i < envsize; i++) {
|
||||||
PyObject* key = PyList_GET_ITEM(keys, i);
|
PyObject* key = PySequence_Fast_GET_ITEM(keys, i);
|
||||||
PyObject* value = PyList_GET_ITEM(values, i);
|
PyObject* value = PySequence_Fast_GET_ITEM(values, i);
|
||||||
|
|
||||||
if (! PyUnicode_Check(key) || ! PyUnicode_Check(value)) {
|
if (! PyUnicode_Check(key) || ! PyUnicode_Check(value)) {
|
||||||
PyErr_SetString(PyExc_TypeError,
|
PyErr_SetString(PyExc_TypeError,
|
||||||
|
@ -760,8 +765,8 @@ getenvironment(PyObject* environment)
|
||||||
end = buffer + totalsize;
|
end = buffer + totalsize;
|
||||||
|
|
||||||
for (i = 0; i < envsize; i++) {
|
for (i = 0; i < envsize; i++) {
|
||||||
PyObject* key = PyList_GET_ITEM(keys, i);
|
PyObject* key = PySequence_Fast_GET_ITEM(keys, i);
|
||||||
PyObject* value = PyList_GET_ITEM(values, i);
|
PyObject* value = PySequence_Fast_GET_ITEM(values, i);
|
||||||
if (!PyUnicode_AsUCS4(key, p, end - p, 0))
|
if (!PyUnicode_AsUCS4(key, p, end - p, 0))
|
||||||
goto error;
|
goto error;
|
||||||
p += PyUnicode_GET_LENGTH(key);
|
p += PyUnicode_GET_LENGTH(key);
|
||||||
|
|
|
@ -1670,6 +1670,9 @@ _multibytecodec_MultibyteStreamWriter_writelines(MultibyteStreamWriterObject *se
|
||||||
if (r == -1)
|
if (r == -1)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
/* PySequence_Length() can fail */
|
||||||
|
if (PyErr_Occurred())
|
||||||
|
return NULL;
|
||||||
|
|
||||||
Py_RETURN_NONE;
|
Py_RETURN_NONE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -4325,7 +4325,7 @@ zip_longest_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
||||||
PyObject *ittuple; /* tuple of iterators */
|
PyObject *ittuple; /* tuple of iterators */
|
||||||
PyObject *result;
|
PyObject *result;
|
||||||
PyObject *fillvalue = Py_None;
|
PyObject *fillvalue = Py_None;
|
||||||
Py_ssize_t tuplesize = PySequence_Length(args);
|
Py_ssize_t tuplesize;
|
||||||
|
|
||||||
if (kwds != NULL && PyDict_CheckExact(kwds) && PyDict_GET_SIZE(kwds) > 0) {
|
if (kwds != NULL && PyDict_CheckExact(kwds) && PyDict_GET_SIZE(kwds) > 0) {
|
||||||
fillvalue = PyDict_GetItemString(kwds, "fillvalue");
|
fillvalue = PyDict_GetItemString(kwds, "fillvalue");
|
||||||
|
@ -4338,6 +4338,7 @@ zip_longest_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
||||||
|
|
||||||
/* args must be a tuple */
|
/* args must be a tuple */
|
||||||
assert(PyTuple_Check(args));
|
assert(PyTuple_Check(args));
|
||||||
|
tuplesize = PyTuple_GET_SIZE(args);
|
||||||
|
|
||||||
/* obtain iterators */
|
/* obtain iterators */
|
||||||
ittuple = PyTuple_New(tuplesize);
|
ittuple = PyTuple_New(tuplesize);
|
||||||
|
|
|
@ -6642,7 +6642,7 @@ static PyObject *
|
||||||
os_setgroups(PyObject *module, PyObject *groups)
|
os_setgroups(PyObject *module, PyObject *groups)
|
||||||
/*[clinic end generated code: output=3fcb32aad58c5ecd input=fa742ca3daf85a7e]*/
|
/*[clinic end generated code: output=3fcb32aad58c5ecd input=fa742ca3daf85a7e]*/
|
||||||
{
|
{
|
||||||
int i, len;
|
Py_ssize_t i, len;
|
||||||
gid_t grouplist[MAX_GROUPS];
|
gid_t grouplist[MAX_GROUPS];
|
||||||
|
|
||||||
if (!PySequence_Check(groups)) {
|
if (!PySequence_Check(groups)) {
|
||||||
|
@ -6650,6 +6650,9 @@ os_setgroups(PyObject *module, PyObject *groups)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
len = PySequence_Size(groups);
|
len = PySequence_Size(groups);
|
||||||
|
if (len < 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
if (len > MAX_GROUPS) {
|
if (len > MAX_GROUPS) {
|
||||||
PyErr_SetString(PyExc_ValueError, "too many groups");
|
PyErr_SetString(PyExc_ValueError, "too many groups");
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -7877,9 +7880,9 @@ os_read_impl(PyObject *module, int fd, Py_ssize_t length)
|
||||||
#if (defined(HAVE_SENDFILE) && (defined(__FreeBSD__) || defined(__DragonFly__) \
|
#if (defined(HAVE_SENDFILE) && (defined(__FreeBSD__) || defined(__DragonFly__) \
|
||||||
|| defined(__APPLE__))) || defined(HAVE_READV) || defined(HAVE_WRITEV)
|
|| defined(__APPLE__))) || defined(HAVE_READV) || defined(HAVE_WRITEV)
|
||||||
static Py_ssize_t
|
static Py_ssize_t
|
||||||
iov_setup(struct iovec **iov, Py_buffer **buf, PyObject *seq, int cnt, int type)
|
iov_setup(struct iovec **iov, Py_buffer **buf, PyObject *seq, Py_ssize_t cnt, int type)
|
||||||
{
|
{
|
||||||
int i, j;
|
Py_ssize_t i, j;
|
||||||
Py_ssize_t blen, total = 0;
|
Py_ssize_t blen, total = 0;
|
||||||
|
|
||||||
*iov = PyMem_New(struct iovec, cnt);
|
*iov = PyMem_New(struct iovec, cnt);
|
||||||
|
@ -7956,8 +7959,7 @@ static Py_ssize_t
|
||||||
os_readv_impl(PyObject *module, int fd, PyObject *buffers)
|
os_readv_impl(PyObject *module, int fd, PyObject *buffers)
|
||||||
/*[clinic end generated code: output=792da062d3fcebdb input=e679eb5dbfa0357d]*/
|
/*[clinic end generated code: output=792da062d3fcebdb input=e679eb5dbfa0357d]*/
|
||||||
{
|
{
|
||||||
int cnt;
|
Py_ssize_t cnt, n;
|
||||||
Py_ssize_t n;
|
|
||||||
int async_err = 0;
|
int async_err = 0;
|
||||||
struct iovec *iov;
|
struct iovec *iov;
|
||||||
Py_buffer *buf;
|
Py_buffer *buf;
|
||||||
|
@ -7969,6 +7971,8 @@ os_readv_impl(PyObject *module, int fd, PyObject *buffers)
|
||||||
}
|
}
|
||||||
|
|
||||||
cnt = PySequence_Size(buffers);
|
cnt = PySequence_Size(buffers);
|
||||||
|
if (cnt < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
if (iov_setup(&iov, &buf, buffers, cnt, PyBUF_WRITABLE) < 0)
|
if (iov_setup(&iov, &buf, buffers, cnt, PyBUF_WRITABLE) < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -8107,15 +8111,24 @@ posix_sendfile(PyObject *self, PyObject *args, PyObject *kwdict)
|
||||||
"sendfile() headers must be a sequence");
|
"sendfile() headers must be a sequence");
|
||||||
return NULL;
|
return NULL;
|
||||||
} else {
|
} else {
|
||||||
Py_ssize_t i = 0; /* Avoid uninitialized warning */
|
Py_ssize_t i = PySequence_Size(headers);
|
||||||
sf.hdr_cnt = PySequence_Size(headers);
|
if (i < 0)
|
||||||
if (sf.hdr_cnt > 0 &&
|
|
||||||
(i = iov_setup(&(sf.headers), &hbuf,
|
|
||||||
headers, sf.hdr_cnt, PyBUF_SIMPLE)) < 0)
|
|
||||||
return NULL;
|
return NULL;
|
||||||
|
if (i > INT_MAX) {
|
||||||
|
PyErr_SetString(PyExc_OverflowError,
|
||||||
|
"sendfile() header is too large");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if (i > 0) {
|
||||||
|
sf.hdr_cnt = (int)i;
|
||||||
|
i = iov_setup(&(sf.headers), &hbuf,
|
||||||
|
headers, sf.hdr_cnt, PyBUF_SIMPLE);
|
||||||
|
if (i < 0)
|
||||||
|
return NULL;
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
sbytes += i;
|
sbytes += i;
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (trailers != NULL) {
|
if (trailers != NULL) {
|
||||||
|
@ -8124,15 +8137,24 @@ posix_sendfile(PyObject *self, PyObject *args, PyObject *kwdict)
|
||||||
"sendfile() trailers must be a sequence");
|
"sendfile() trailers must be a sequence");
|
||||||
return NULL;
|
return NULL;
|
||||||
} else {
|
} else {
|
||||||
Py_ssize_t i = 0; /* Avoid uninitialized warning */
|
Py_ssize_t i = PySequence_Size(trailers);
|
||||||
sf.trl_cnt = PySequence_Size(trailers);
|
if (i < 0)
|
||||||
if (sf.trl_cnt > 0 &&
|
|
||||||
(i = iov_setup(&(sf.trailers), &tbuf,
|
|
||||||
trailers, sf.trl_cnt, PyBUF_SIMPLE)) < 0)
|
|
||||||
return NULL;
|
return NULL;
|
||||||
|
if (i > INT_MAX) {
|
||||||
|
PyErr_SetString(PyExc_OverflowError,
|
||||||
|
"sendfile() trailer is too large");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if (i > 0) {
|
||||||
|
sf.trl_cnt = (int)i;
|
||||||
|
i = iov_setup(&(sf.trailers), &tbuf,
|
||||||
|
trailers, sf.trl_cnt, PyBUF_SIMPLE);
|
||||||
|
if (i < 0)
|
||||||
|
return NULL;
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
sbytes += i;
|
sbytes += i;
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8402,7 +8424,7 @@ static Py_ssize_t
|
||||||
os_writev_impl(PyObject *module, int fd, PyObject *buffers)
|
os_writev_impl(PyObject *module, int fd, PyObject *buffers)
|
||||||
/*[clinic end generated code: output=56565cfac3aac15b input=5b8d17fe4189d2fe]*/
|
/*[clinic end generated code: output=56565cfac3aac15b input=5b8d17fe4189d2fe]*/
|
||||||
{
|
{
|
||||||
int cnt;
|
Py_ssize_t cnt;
|
||||||
Py_ssize_t result;
|
Py_ssize_t result;
|
||||||
int async_err = 0;
|
int async_err = 0;
|
||||||
struct iovec *iov;
|
struct iovec *iov;
|
||||||
|
@ -8414,6 +8436,8 @@ os_writev_impl(PyObject *module, int fd, PyObject *buffers)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
cnt = PySequence_Size(buffers);
|
cnt = PySequence_Size(buffers);
|
||||||
|
if (cnt < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
if (iov_setup(&iov, &buf, buffers, cnt, PyBUF_SIMPLE) < 0) {
|
if (iov_setup(&iov, &buf, buffers, cnt, PyBUF_SIMPLE) < 0) {
|
||||||
return -1;
|
return -1;
|
||||||
|
|
|
@ -2021,8 +2021,8 @@ newKqueue_Object(PyTypeObject *type, SOCKET fd)
|
||||||
static PyObject *
|
static PyObject *
|
||||||
kqueue_queue_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
kqueue_queue_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
||||||
{
|
{
|
||||||
if ((args != NULL && PyObject_Size(args)) ||
|
if (PyTuple_GET_SIZE(args) ||
|
||||||
(kwds != NULL && PyObject_Size(kwds))) {
|
(kwds != NULL && PyDict_GET_SIZE(kwds))) {
|
||||||
PyErr_SetString(PyExc_ValueError,
|
PyErr_SetString(PyExc_ValueError,
|
||||||
"select.kqueue doesn't accept arguments");
|
"select.kqueue doesn't accept arguments");
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
|
@ -2790,7 +2790,7 @@ _PyErr_TrySetFromCause(const char *format, ...)
|
||||||
/* Ensure the instance dict is also empty */
|
/* Ensure the instance dict is also empty */
|
||||||
dictptr = _PyObject_GetDictPtr(val);
|
dictptr = _PyObject_GetDictPtr(val);
|
||||||
if (dictptr != NULL && *dictptr != NULL &&
|
if (dictptr != NULL && *dictptr != NULL &&
|
||||||
PyObject_Length(*dictptr) > 0) {
|
PyDict_GET_SIZE(*dictptr) > 0) {
|
||||||
/* While we could potentially copy a non-empty instance dictionary
|
/* While we could potentially copy a non-empty instance dictionary
|
||||||
* to the replacement exception, for now we take the more
|
* to the replacement exception, for now we take the more
|
||||||
* conservative path of leaving exceptions with attributes set
|
* conservative path of leaving exceptions with attributes set
|
||||||
|
|
|
@ -40,15 +40,9 @@ namespace_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
||||||
static int
|
static int
|
||||||
namespace_init(_PyNamespaceObject *ns, PyObject *args, PyObject *kwds)
|
namespace_init(_PyNamespaceObject *ns, PyObject *args, PyObject *kwds)
|
||||||
{
|
{
|
||||||
// ignore args if it's NULL or empty
|
if (PyTuple_GET_SIZE(args) != 0) {
|
||||||
if (args != NULL) {
|
PyErr_Format(PyExc_TypeError, "no positional arguments expected");
|
||||||
Py_ssize_t argcount = PyObject_Size(args);
|
return -1;
|
||||||
if (argcount < 0)
|
|
||||||
return -1;
|
|
||||||
else if (argcount > 0) {
|
|
||||||
PyErr_Format(PyExc_TypeError, "no positional arguments expected");
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (kwds == NULL)
|
if (kwds == NULL)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -1546,20 +1546,26 @@ set_difference(PySetObject *so, PyObject *other)
|
||||||
PyObject *key;
|
PyObject *key;
|
||||||
Py_hash_t hash;
|
Py_hash_t hash;
|
||||||
setentry *entry;
|
setentry *entry;
|
||||||
Py_ssize_t pos = 0;
|
Py_ssize_t pos = 0, other_size;
|
||||||
int rv;
|
int rv;
|
||||||
|
|
||||||
if (PySet_GET_SIZE(so) == 0) {
|
if (PySet_GET_SIZE(so) == 0) {
|
||||||
return set_copy(so);
|
return set_copy(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!PyAnySet_Check(other) && !PyDict_CheckExact(other)) {
|
if (PyAnySet_Check(other)) {
|
||||||
|
other_size = PySet_GET_SIZE(other);
|
||||||
|
}
|
||||||
|
else if (PyDict_CheckExact(other)) {
|
||||||
|
other_size = PyDict_GET_SIZE(other);
|
||||||
|
}
|
||||||
|
else {
|
||||||
return set_copy_and_difference(so, other);
|
return set_copy_and_difference(so, other);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If len(so) much more than len(other), it's more efficient to simply copy
|
/* If len(so) much more than len(other), it's more efficient to simply copy
|
||||||
* so and then iterate other looking for common elements. */
|
* so and then iterate other looking for common elements. */
|
||||||
if ((PySet_GET_SIZE(so) >> 2) > PyObject_Size(other)) {
|
if ((PySet_GET_SIZE(so) >> 2) > other_size) {
|
||||||
return set_copy_and_difference(so, other);
|
return set_copy_and_difference(so, other);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2442,13 +2442,14 @@ zip_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
|
||||||
Py_ssize_t i;
|
Py_ssize_t i;
|
||||||
PyObject *ittuple; /* tuple of iterators */
|
PyObject *ittuple; /* tuple of iterators */
|
||||||
PyObject *result;
|
PyObject *result;
|
||||||
Py_ssize_t tuplesize = PySequence_Length(args);
|
Py_ssize_t tuplesize;
|
||||||
|
|
||||||
if (type == &PyZip_Type && !_PyArg_NoKeywords("zip()", kwds))
|
if (type == &PyZip_Type && !_PyArg_NoKeywords("zip()", kwds))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
/* args must be a tuple */
|
/* args must be a tuple */
|
||||||
assert(PyTuple_Check(args));
|
assert(PyTuple_Check(args));
|
||||||
|
tuplesize = PyTuple_GET_SIZE(args);
|
||||||
|
|
||||||
/* obtain iterators */
|
/* obtain iterators */
|
||||||
ittuple = PyTuple_New(tuplesize);
|
ittuple = PyTuple_New(tuplesize);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue