Skip to content

Commit

Permalink
merge 3.4-slp (Stackless python#96, python#98)
Browse files Browse the repository at this point in the history
  • Loading branch information
Anselm Kruis committed Nov 9, 2016
2 parents 4717ee3 + 0125e06 commit 44c7230
Show file tree
Hide file tree
Showing 4 changed files with 86 additions and 6 deletions.
10 changes: 8 additions & 2 deletions Modules/_pickle.c
Original file line number Diff line number Diff line change
Expand Up @@ -3706,8 +3706,14 @@ save(PicklerObject *self, PyObject *obj, int pers_save)

#ifdef STACKLESS
/* but we save the stack after a fixed watermark */
if (CSTACK_SAVE_NOW(PyThreadState_GET(), self))
return slp_safe_pickling((void *)&save, (PyObject *)self, obj, pers_save);
if (CSTACK_SAVE_NOW(PyThreadState_GET(), self)) {
int res;
if (Py_EnterRecursiveCall(" while pickling an object"))
return -1;
res = slp_safe_pickling((void *)&save, (PyObject *)self, obj, pers_save);
Py_LeaveRecursiveCall();
return res;
}
#endif
if (Py_EnterRecursiveCall(" while pickling an object"))
return -1;
Expand Down
13 changes: 11 additions & 2 deletions Stackless/changelog.txt
Original file line number Diff line number Diff line change
@@ -1,15 +1,24 @@
Stackless-Python News
+++++++++++++++++++++

This file documents changes to the Stackless extension of C-Python.
This file documents user visible changes to the Stackless extension of C-Python.
For other changes see Misc/NEWS.


What's New in Stackless 3.X.X?
==============================

*Release date: 20XX-XX-XX*

- https://bitbucket.org/stackless-dev/stackless/issues/98
Prevent an unlikely NULL-pointer access in safe_pickle.c pickle_callback().
The bug is very old and might affect multiprocessing.

- https://bitbucket.org/stackless-dev/stackless/issues/96
Impose a very high limit on the recursion depth of cPickle.
Previously an infinite recursion could eat up all you memory
and finally crash Stackless. No the recursion stops after using
about 170MB (32bit) / 300MB (64bit) of memory.

- https://bitbucket.org/stackless-dev/stackless/issues/93
Unify tasklet.kill(), tasklet.throw() and tasklet.raise_exception().
They now behave almost identically. This affects the error handling in some
Expand Down
9 changes: 7 additions & 2 deletions Stackless/pickling/safe_pickle.c
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,12 @@ pickle_callback(PyFrameObject *f, int exc, PyObject *retval)
*/
saved_base = ts->st.cstack_root;
ts->st.cstack_root = STACK_REFPLUS + (intptr_t *) &f;
Py_DECREF(retval);
cf->i = cPickle_save(cf->ob1, cf->ob2, cf->n);
if (retval) {
Py_DECREF(retval);
cf->i = cPickle_save(cf->ob1, cf->ob2, cf->n);
} else {
cf->i = -1;
}
ts->st.cstack_root = saved_base;

/* jump back. No decref, frame contains result. */
Expand All @@ -41,6 +45,7 @@ pickle_callback(PyFrameObject *f, int exc, PyObject *retval)
ts->frame = cf->f_back;
slp_transfer_return(cst);
/* never come here */
assert(0);
return NULL;
}

Expand Down
60 changes: 60 additions & 0 deletions Stackless/unittests/test_defects.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
import gc
import sys
import types
from io import BytesIO
import time
try:
import threading
withThreads = True
Expand Down Expand Up @@ -428,6 +430,64 @@ def test_invalid_args_tasklet_kill(self):
self.assertRaises(TypeError, func, False, None)


class TestCPickleBombHandling_Dict(dict):
pass


class TestCPickleBombHandling_Cls(object):
def __getstate__(self):
try:
started = self.started
except AttributeError:
pass
else:
self.started = None
started.set()
# print("started")
time.sleep(0.05) # give the other thread a chance to run
return self.__dict__


class TestCPickleBombHandling(StacklessTestCase):
def other_thread(self, pickler, c):
try:
pickler.dump(c)
except TaskletExit:
self.killed = None
except:
self.killed = sys.exc_info()
else:
self.killed = False

def test_kill_during_cPickle_stack_switch(self):
# this test kills the main/current tasklet of a other-thread,
# which is fast-pickling a recursive structure. This leads to an
# infinite recursion, which gets interrupted by a bomb thrown from
# main-thread. Until issue #98 got fixed, this caused a crash.
# See https://bitbucket.org/stackless-dev/stackless/issues/98
buf = BytesIO()
import _pickle as pickle
pickler = pickle.Pickler(buf, protocol=-1)
pickler.fast = 1

started = threading.Event()

c = TestCPickleBombHandling_Cls()
c.started = started
d = TestCPickleBombHandling_Dict()
d[1] = d
c.recursive = d
self.killed = "undefined"
t = threading.Thread(target=self.other_thread, name="other_thread", args=(pickler, c))
t.start()
started.wait()
stackless.get_thread_info(t.ident)[0].kill(pending=True)
# print("killing")
t.join()
if isinstance(self.killed, tuple):
raise (self.killed[0], self.killed[1], self.killed[2])
self.assertIsNone(self.killed)

if __name__ == '__main__':
if not sys.argv[1:]:
sys.argv.append('-v')
Expand Down

0 comments on commit 44c7230

Please sign in to comment.