mirror of
https://github.com/python/cpython.git
synced 2025-07-24 03:35:53 +00:00
Fix obscure set crashers (#4420). Backport of d56b3cafb1e6, reviewed by Raymond.
This commit is contained in:
parent
1fd26525a4
commit
f079c9bfbf
2 changed files with 59 additions and 12 deletions
|
@ -1588,6 +1588,39 @@ class TestVariousIteratorArgs(unittest.TestCase):
|
|||
self.assertRaises(TypeError, getattr(set('january'), methname), N(data))
|
||||
self.assertRaises(ZeroDivisionError, getattr(set('january'), methname), E(data))
|
||||
|
||||
class bad_eq:
|
||||
def __eq__(self, other):
|
||||
if be_bad:
|
||||
set2.clear()
|
||||
raise ZeroDivisionError
|
||||
return self is other
|
||||
def __hash__(self):
|
||||
return 0
|
||||
|
||||
class bad_dict_clear:
|
||||
def __eq__(self, other):
|
||||
if be_bad:
|
||||
dict2.clear()
|
||||
return self is other
|
||||
def __hash__(self):
|
||||
return 0
|
||||
|
||||
class TestWeirdBugs(unittest.TestCase):
|
||||
def test_8420_set_merge(self):
|
||||
# This used to segfault
|
||||
global be_bad, set2, dict2
|
||||
be_bad = False
|
||||
set1 = {bad_eq()}
|
||||
set2 = {bad_eq() for i in range(75)}
|
||||
be_bad = True
|
||||
self.assertRaises(ZeroDivisionError, set1.update, set2)
|
||||
|
||||
be_bad = False
|
||||
set1 = {bad_dict_clear()}
|
||||
dict2 = {bad_dict_clear(): None}
|
||||
be_bad = True
|
||||
set1.symmetric_difference_update(dict2)
|
||||
|
||||
# Application tests (based on David Eppstein's graph recipes ====================================
|
||||
|
||||
def powerset(U):
|
||||
|
@ -1729,6 +1762,7 @@ def test_main(verbose=None):
|
|||
TestIdentities,
|
||||
TestVariousIteratorArgs,
|
||||
TestGraphs,
|
||||
TestWeirdBugs,
|
||||
)
|
||||
|
||||
test_support.run_unittest(*test_classes)
|
||||
|
|
|
@ -362,12 +362,14 @@ static int
|
|||
set_add_entry(register PySetObject *so, setentry *entry)
|
||||
{
|
||||
register Py_ssize_t n_used;
|
||||
PyObject *key = entry->key;
|
||||
long hash = entry->hash;
|
||||
|
||||
assert(so->fill <= so->mask); /* at least one empty slot */
|
||||
n_used = so->used;
|
||||
Py_INCREF(entry->key);
|
||||
if (set_insert_key(so, entry->key, entry->hash) == -1) {
|
||||
Py_DECREF(entry->key);
|
||||
Py_INCREF(key);
|
||||
if (set_insert_key(so, key, hash) == -1) {
|
||||
Py_DECREF(key);
|
||||
return -1;
|
||||
}
|
||||
if (!(so->used > n_used && so->fill*3 >= (so->mask+1)*2))
|
||||
|
@ -647,6 +649,8 @@ static int
|
|||
set_merge(PySetObject *so, PyObject *otherset)
|
||||
{
|
||||
PySetObject *other;
|
||||
PyObject *key;
|
||||
long hash;
|
||||
register Py_ssize_t i;
|
||||
register setentry *entry;
|
||||
|
||||
|
@ -667,11 +671,13 @@ set_merge(PySetObject *so, PyObject *otherset)
|
|||
}
|
||||
for (i = 0; i <= other->mask; i++) {
|
||||
entry = &other->table[i];
|
||||
if (entry->key != NULL &&
|
||||
entry->key != dummy) {
|
||||
Py_INCREF(entry->key);
|
||||
if (set_insert_key(so, entry->key, entry->hash) == -1) {
|
||||
Py_DECREF(entry->key);
|
||||
key = entry->key;
|
||||
hash = entry->hash;
|
||||
if (key != NULL &&
|
||||
key != dummy) {
|
||||
Py_INCREF(key);
|
||||
if (set_insert_key(so, key, hash) == -1) {
|
||||
Py_DECREF(key);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
@ -1640,15 +1646,22 @@ set_symmetric_difference_update(PySetObject *so, PyObject *other)
|
|||
while (_PyDict_Next(other, &pos, &key, &value, &hash)) {
|
||||
setentry an_entry;
|
||||
|
||||
Py_INCREF(key);
|
||||
an_entry.hash = hash;
|
||||
an_entry.key = key;
|
||||
|
||||
rv = set_discard_entry(so, &an_entry);
|
||||
if (rv == -1)
|
||||
if (rv == -1) {
|
||||
Py_DECREF(key);
|
||||
return NULL;
|
||||
if (rv == DISCARD_NOTFOUND) {
|
||||
if (set_add_entry(so, &an_entry) == -1)
|
||||
return NULL;
|
||||
}
|
||||
if (rv == DISCARD_NOTFOUND) {
|
||||
if (set_add_entry(so, &an_entry) == -1) {
|
||||
Py_DECREF(key);
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
Py_DECREF(key);
|
||||
}
|
||||
Py_RETURN_NONE;
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue