Implemented proto 2 NEWTRUE and NEWFALSE in cPickle.

This commit is contained in:
Tim Peters 2003-02-02 17:59:11 +00:00
parent d156c2d782
commit 3c67d795ef
2 changed files with 40 additions and 2 deletions

View File

@ -502,6 +502,12 @@ class AbstractPickleTests(unittest.TestCase):
y = self.loads(s)
self.assert_(x is y, (proto, x, s, y))
# Test that proto >= 2 really uses the bool opcodes.
if proto >= 2 and x in (False, True):
expected = x and pickle.NEWTRUE or pickle.NEWFALSE
# Skip the PROTO opcode at the start.
self.assertEqual(s[2], expected)
def test_newobj_tuple(self):
x = MyTuple([1, 2, 3])
x.foo = 42

View File

@ -963,9 +963,13 @@ save_bool(Picklerobject *self, PyObject *args)
static char len[2] = {sizeof(FALSE)-1, sizeof(TRUE)-1};
long l = PyInt_AS_LONG((PyIntObject *)args);
if ((*self->write_func)(self, buf[l], len[l]) < 0)
if (self->proto >= 2) {
char opcode = l ? NEWTRUE : NEWFALSE;
if (self->write_func(self, &opcode, 1) < 0)
return -1;
}
else if (self->write_func(self, buf[l], len[l]) < 0)
return -1;
return 0;
}
@ -2789,6 +2793,15 @@ load_int(Unpicklerobject *self)
return res;
}
static int
load_bool(Unpicklerobject *self, PyObject *boolean)
{
assert(boolean == Py_True || boolean == Py_False);
Py_INCREF(boolean);
PDATA_PUSH(self->stack, boolean, -1);
return 0;
}
/* s contains x bytes of a little-endian integer. Return its value as a
* C int. Obscure: when x is 1 or 2, this is an unsigned little-endian
* int, but when x is 4 it's a signed one. This is an historical source
@ -4144,6 +4157,16 @@ load(Unpicklerobject *self)
break;
continue;
case NEWTRUE:
if (load_bool(self, Py_True) < 0)
break;
continue;
case NEWFALSE:
if (load_bool(self, Py_False) < 0)
break;
continue;
case '\0':
/* end of file */
PyErr_SetNone(PyExc_EOFError);
@ -4462,6 +4485,15 @@ noload(Unpicklerobject *self)
break;
continue;
case NEWTRUE:
if (load_bool(self, Py_True) < 0)
break;
continue;
case NEWFALSE:
if (load_bool(self, Py_False) < 0)
break;
continue;
default:
cPickle_ErrFormat(UnpicklingError,
"invalid load key, '%s'.",