]> git.ipfire.org Git - thirdparty/Python/cpython.git/commitdiff
Iterators phase 1. This comprises:
authorGuido van Rossum <guido@python.org>
Fri, 20 Apr 2001 19:13:02 +0000 (19:13 +0000)
committerGuido van Rossum <guido@python.org>
Fri, 20 Apr 2001 19:13:02 +0000 (19:13 +0000)
new slot tp_iter in type object, plus new flag Py_TPFLAGS_HAVE_ITER
new C API PyObject_GetIter(), calls tp_iter
new builtin iter(), with two forms: iter(obj), and iter(function, sentinel)
new internal object types iterobject and calliterobject
new exception StopIteration
new opcodes for "for" loops, GET_ITER and FOR_ITER (also supported by dis.py)
new magic number for .pyc files
new special method for instances: __iter__() returns an iterator
iteration over dictionaries: "for x in dict" iterates over the keys
iteration over files: "for x in file" iterates over lines

TODO:

documentation
test suite
decide whether to use a different way to spell iter(function, sentinal)
decide whether "for key in dict" is a good idea
use iterators in map/filter/reduce, min/max, and elsewhere (in/not in?)
speed tuning (make next() a slot tp_next???)

16 files changed:
Include/Python.h
Include/abstract.h
Include/object.h
Include/opcode.h
Include/pyerrors.h
Lib/dis.py
Makefile.pre.in
Objects/abstract.c
Objects/classobject.c
Objects/dictobject.c
Objects/stringobject.c
Python/bltinmodule.c
Python/ceval.c
Python/compile.c
Python/exceptions.c
Python/import.c

index b771c8cf702ef2644061c01254f6c5bc0d3d5dbf..23938720049f61c4bca1c8eda969feff5ac0b279 100644 (file)
@@ -82,6 +82,7 @@
 #include "traceback.h"
 #include "sliceobject.h"
 #include "cellobject.h"
+#include "iterobject.h"
 
 #include "codecs.h"
 #include "pyerrors.h"
index e0765e5cb1f3349d1a32982987f06ace51bc1167..c56c8876de16ae559b1c8d408dcb72e0550468a9 100644 (file)
@@ -470,6 +470,11 @@ xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx*/
 
        */
 
+     DL_IMPORT(PyObject *) PyObject_GetIter(PyObject *);
+     /* Takes an object and returns an iterator for it.
+        This is typically a new iterator but if the argument
+       is an iterator, this returns itself. */
+
 /*  Number Protocol:*/
 
      DL_IMPORT(int) PyNumber_Check(PyObject *o);
index 80669da7d946c2823d065082b35115366df5c6d3..4a53835116affca5bc71a87670f6244a192e79c8 100644 (file)
@@ -200,6 +200,7 @@ typedef int (*cmpfunc)(PyObject *, PyObject *);
 typedef PyObject *(*reprfunc)(PyObject *);
 typedef long (*hashfunc)(PyObject *);
 typedef PyObject *(*richcmpfunc) (PyObject *, PyObject *, int);
+typedef PyObject *(*getiterfunc) (PyObject *);
 
 typedef struct _typeobject {
        PyObject_VAR_HEAD
@@ -249,8 +250,11 @@ typedef struct _typeobject {
        /* weak reference enabler */
        long tp_weaklistoffset;
 
+       /* Iterators */
+       getiterfunc tp_iter;
+
 #ifdef COUNT_ALLOCS
-       /* these must be last */
+       /* these must be last and never explicitly initialized */
        int tp_alloc;
        int tp_free;
        int tp_maxalloc;
@@ -342,20 +346,22 @@ given type object has a specified feature.
 /* PyNumberMethods do their own coercion */
 #define Py_TPFLAGS_CHECKTYPES (1L<<4)
 
+/* tp_richcompare is defined */
 #define Py_TPFLAGS_HAVE_RICHCOMPARE (1L<<5)
 
 /* Objects which are weakly referencable if their tp_weaklistoffset is >0 */
-/* XXX Should this have the same value as Py_TPFLAGS_HAVE_RICHCOMPARE?
- * These both indicate a feature that appeared in the same alpha release.
- */
 #define Py_TPFLAGS_HAVE_WEAKREFS (1L<<6)
 
+/* tp_iter is defined */
+#define Py_TPFLAGS_HAVE_ITER (1L<<7)
+
 #define Py_TPFLAGS_DEFAULT  ( \
                              Py_TPFLAGS_HAVE_GETCHARBUFFER | \
                              Py_TPFLAGS_HAVE_SEQUENCE_IN | \
                              Py_TPFLAGS_HAVE_INPLACEOPS | \
                              Py_TPFLAGS_HAVE_RICHCOMPARE | \
                              Py_TPFLAGS_HAVE_WEAKREFS | \
+                             Py_TPFLAGS_HAVE_ITER | \
                             0)
 
 #define PyType_HasFeature(t,f)  (((t)->tp_flags & (f)) != 0)
index 546ad08856079183b1123ebcdca9224888efcfe7..0cace8edbf3cfa459554143eb41a31b412e4cc84 100644 (file)
@@ -53,6 +53,7 @@ extern "C" {
 #define BINARY_XOR     65
 #define BINARY_OR      66
 #define INPLACE_POWER  67
+#define GET_ITER       68
 
 #define PRINT_EXPR     70
 #define PRINT_ITEM     71
@@ -80,6 +81,7 @@ extern "C" {
 #define STORE_NAME     90      /* Index in name list */
 #define DELETE_NAME    91      /* "" */
 #define UNPACK_SEQUENCE        92      /* Number of sequence items */
+#define FOR_ITER       93
 
 #define STORE_ATTR     95      /* Index in name list */
 #define DELETE_ATTR    96      /* "" */
index 622af2d43bbdbab8c544f84c6917643a17f20614..3c57e2f03144fda23d53ff789c0d1415997a6f02 100644 (file)
@@ -24,6 +24,7 @@ DL_IMPORT(void) PyErr_NormalizeException(PyObject**, PyObject**, PyObject**);
 /* Predefined exceptions */
 
 extern DL_IMPORT(PyObject *) PyExc_Exception;
+extern DL_IMPORT(PyObject *) PyExc_StopIteration;
 extern DL_IMPORT(PyObject *) PyExc_StandardError;
 extern DL_IMPORT(PyObject *) PyExc_ArithmeticError;
 extern DL_IMPORT(PyObject *) PyExc_LookupError;
index dc3025721f2c81d4dec822818eb0a7c5f0af3d86..fb9736983c66de858d2e04cb9641d33372c86c91 100644 (file)
@@ -205,6 +205,7 @@ def_op('BINARY_AND', 64)
 def_op('BINARY_XOR', 65)
 def_op('BINARY_OR', 66)
 def_op('INPLACE_POWER', 67)
+def_op('GET_ITER', 68)
 
 def_op('PRINT_EXPR', 70)
 def_op('PRINT_ITEM', 71)
@@ -232,6 +233,7 @@ HAVE_ARGUMENT = 90              # Opcodes from here have an argument:
 name_op('STORE_NAME', 90)       # Index in name list
 name_op('DELETE_NAME', 91)      # ""
 def_op('UNPACK_SEQUENCE', 92)   # Number of tuple items
+def_op('FOR_ITER', 93)
 
 name_op('STORE_ATTR', 95)       # Index in name list
 name_op('DELETE_ATTR', 96)      # ""
index f2f33362355d2803c9c44a1edf1e46840bf5d6a9..103a36b345894be7f49c3b994725b1cd791f7be5 100644 (file)
@@ -238,6 +238,7 @@ OBJECT_OBJS=        \
                Objects/frameobject.o \
                Objects/funcobject.o \
                Objects/intobject.o \
+               Objects/iterobject.o \
                Objects/listobject.o \
                Objects/longobject.o \
                Objects/dictobject.o \
@@ -433,6 +434,7 @@ PYTHON_HEADERS= \
                Include/bufferobject.h \
                Include/tupleobject.h \
                Include/listobject.h \
+               Include/iterobject.h \
                Include/dictobject.h \
                Include/methodobject.h \
                Include/moduleobject.h \
index 3ccac7140eb88a63c77049da00c960b7b03f7d6f..8a6df76fc31a5cb63ec97d46d041e5ebcec20a01 100644 (file)
@@ -1738,3 +1738,20 @@ PyObject_IsSubclass(PyObject *derived, PyObject *cls)
 
        return retval;
 }
+
+PyObject *
+PyObject_GetIter(PyObject *o)
+{
+       PyTypeObject *t = o->ob_type;
+       getiterfunc f = NULL;
+       if (PyType_HasFeature(t, Py_TPFLAGS_HAVE_ITER))
+               f = t->tp_iter;
+       if (f == NULL) {
+               if (PySequence_Check(o))
+                       return PyIter_New(o);
+               PyErr_SetString(PyExc_TypeError, "iter() of non-sequence");
+               return NULL;
+       }
+       else
+               return (*f)(o);
+}
index fa71c4ea3a74fa187150a8f1146d00c8b674f060..742e4729103998edbc786e73c6a0d34130c0103b 100644 (file)
@@ -848,7 +848,7 @@ instance_traverse(PyInstanceObject *o, visitproc visit, void *arg)
        return 0;
 }
 
-static PyObject *getitemstr, *setitemstr, *delitemstr, *lenstr;
+static PyObject *getitemstr, *setitemstr, *delitemstr, *lenstr, *iterstr;
 
 static int
 instance_length(PyInstanceObject *inst)
@@ -1712,6 +1712,32 @@ instance_richcompare(PyObject *v, PyObject *w, int op)
 }
 
 
+/* Get the iterator */
+static PyObject *
+instance_getiter(PyInstanceObject *self)
+{
+       PyObject *func;
+
+       if (iterstr == NULL)
+               iterstr = PyString_InternFromString("__iter__");
+       if (getitemstr == NULL)
+               getitemstr = PyString_InternFromString("__getitem__");
+
+       if ((func = instance_getattr(self, iterstr)) != NULL) {
+               PyObject *res = PyEval_CallObject(func, (PyObject *)NULL);
+               Py_DECREF(func);
+               return res;
+       }
+       PyErr_Clear();
+       if ((func = instance_getattr(self, getitemstr)) == NULL) {
+               PyErr_SetString(PyExc_TypeError, "iter() of non-sequence");
+               return NULL;
+       }
+       Py_DECREF(func);
+       return PyIter_New((PyObject *)self);
+}
+
+
 static PyNumberMethods instance_as_number = {
        (binaryfunc)instance_add,               /* nb_add */
        (binaryfunc)instance_sub,               /* nb_subtract */
@@ -1775,7 +1801,8 @@ PyTypeObject PyInstance_Type = {
        (traverseproc)instance_traverse,        /* tp_traverse */
        0,                                      /* tp_clear */
        instance_richcompare,                   /* tp_richcompare */
-       offsetof(PyInstanceObject, in_weakreflist) /* tp_weaklistoffset */
+       offsetof(PyInstanceObject, in_weakreflist), /* tp_weaklistoffset */
+       (getiterfunc)instance_getiter,          /* tp_iter */
 };
 
 
index 95d5b71bd0ba9d30eeb1a19a3ea1815d778b989d..17b6a043e2a7cf757ef38136c2feea8f73c6e993 100644 (file)
@@ -1324,6 +1324,8 @@ static PySequenceMethods dict_as_sequence = {
        0,                                      /* sq_inplace_repeat */
 };
 
+staticforward PyObject *dictiter_new(dictobject *);
+
 PyTypeObject PyDict_Type = {
        PyObject_HEAD_INIT(&PyType_Type)
        0,
@@ -1350,6 +1352,8 @@ PyTypeObject PyDict_Type = {
        (traverseproc)dict_traverse,            /* tp_traverse */
        (inquiry)dict_tp_clear,                 /* tp_clear */
        0,                                      /* tp_richcompare */
+       0,                                      /* tp_weaklistoffset */
+       (getiterfunc)dictiter_new,              /* tp_iter */
 };
 
 /* For backward compatibility with old dictionary interface */
@@ -1392,3 +1396,102 @@ PyDict_DelItemString(PyObject *v, char *key)
        Py_DECREF(kv);
        return err;
 }
+
+/* Dictionary iterator type */
+
+extern PyTypeObject PyDictIter_Type; /* Forward */
+
+typedef struct {
+       PyObject_HEAD
+       dictobject *di_dict;
+       int di_size;
+       int di_pos;
+} dictiterobject;
+
+static PyObject *
+dictiter_new(dictobject *dict)
+{
+       dictiterobject *di;
+       di = PyObject_NEW(dictiterobject, &PyDictIter_Type);
+       if (di == NULL)
+               return NULL;
+       Py_INCREF(dict);
+       di->di_dict = dict;
+       di->di_size = dict->ma_size;
+       di->di_pos = 0;
+       return (PyObject *)di;
+}
+
+static void
+dictiter_dealloc(dictiterobject *di)
+{
+       Py_DECREF(di->di_dict);
+       PyObject_DEL(di);
+}
+
+static PyObject *
+dictiter_next(dictiterobject *di, PyObject *args)
+{
+       PyObject *key;
+       if (di->di_size != di->di_dict->ma_size) {
+               PyErr_SetString(PyExc_RuntimeError,
+                               "dictionary changed size during iteration");
+               return NULL;
+       }
+       if (PyDict_Next((PyObject *)(di->di_dict), &di->di_pos, &key, NULL)) {
+               Py_INCREF(key);
+               return key;
+       }
+       PyErr_SetObject(PyExc_StopIteration, Py_None);
+       return NULL;
+}
+
+static PyObject *
+dictiter_getiter(PyObject *it)
+{
+       Py_INCREF(it);
+       return it;
+}
+
+static PyMethodDef dictiter_methods[] = {
+       {"next",        (PyCFunction)dictiter_next,     METH_VARARGS,
+        "it.next() -- get the next value, or raise StopIteration"},
+       {NULL,          NULL}           /* sentinel */
+};
+
+static PyObject *
+dictiter_getattr(dictiterobject *it, char *name)
+{
+       return Py_FindMethod(dictiter_methods, (PyObject *)it, name);
+}
+
+PyTypeObject PyDictIter_Type = {
+       PyObject_HEAD_INIT(&PyType_Type)
+       0,                                      /* ob_size */
+       "dictionary-iterator",                  /* tp_name */
+       sizeof(dictiterobject),                 /* tp_basicsize */
+       0,                                      /* tp_itemsize */
+       /* methods */
+       (destructor)dictiter_dealloc,           /* tp_dealloc */
+       0,                                      /* tp_print */
+       (getattrfunc)dictiter_getattr,          /* tp_getattr */
+       0,                                      /* tp_setattr */
+       0,                                      /* tp_compare */
+       0,                                      /* tp_repr */
+       0,                                      /* tp_as_number */
+       0,                                      /* tp_as_sequence */
+       0,                                      /* tp_as_mapping */
+       0,                                      /* tp_hash */
+       0,                                      /* tp_call */
+       0,                                      /* tp_str */
+       0,                                      /* tp_getattro */
+       0,                                      /* tp_setattro */
+       0,                                      /* tp_as_buffer */
+       Py_TPFLAGS_DEFAULT,                     /* tp_flags */
+       0,                                      /* tp_doc */
+       0,                                      /* tp_traverse */
+       0,                                      /* tp_clear */
+       0,                                      /* tp_richcompare */
+       0,                                      /* tp_weaklistoffset */
+       (getiterfunc)dictiter_getiter,          /* tp_iter */
+};
index 1701b2fcc39abaebb82704b69ca379b0a5a9ea2b..861cade973479fe13452e5c25af1d4814a6fed2c 100644 (file)
@@ -3232,6 +3232,8 @@ PyString_Fini(void)
 void _Py_ReleaseInternedStrings(void)
 {
        if (interned) {
+               fprintf(stderr, "releasing interned strings\n");
+               PyDict_Clear(interned);
                Py_DECREF(interned);
                interned = NULL;
        }
index 576447cabc83427e6f7f70c1883cc7eb36428e25..7e8f55584765caed9528dcf230036bea515c99fc 100644 (file)
@@ -1311,6 +1311,32 @@ static char float_doc[] =
 Convert a string or number to a floating point number, if possible.";
 
 
+static PyObject *
+builtin_iter(PyObject *self, PyObject *args)
+{
+       PyObject *v, *w = NULL;
+
+       if (!PyArg_ParseTuple(args, "O|O:iter", &v, &w))
+               return NULL;
+       if (w == NULL)
+               return PyObject_GetIter(v);
+       if (!PyCallable_Check(v)) {
+               PyErr_SetString(PyExc_TypeError,
+                               "iter(v, w): v must be callable");
+               return NULL;
+       }
+       return PyCallIter_New(v, w);
+}
+
+static char iter_doc[] =
+"iter(collection) -> iterator\n\
+iter(callable, sentinel) -> iterator\n\
+\n\
+Get an iterator from an object.  In the first form, the argument must\n\
+supply its own iterator, or be a sequence.\n\
+In the second form, the callable is called until it returns the sentinel.";
+
+
 static PyObject *
 builtin_len(PyObject *self, PyObject *args)
 {
@@ -2148,6 +2174,7 @@ static PyMethodDef builtin_methods[] = {
        {"int",         builtin_int, 1, int_doc},
        {"isinstance",  builtin_isinstance, 1, isinstance_doc},
        {"issubclass",  builtin_issubclass, 1, issubclass_doc},
+       {"iter",        builtin_iter, 1, iter_doc},
        {"len",         builtin_len, 1, len_doc},
        {"list",        builtin_list, 1, list_doc},
        {"locals",      builtin_locals, 1, locals_doc},
index b6686b6e1bf8648fdc6fcb382215ac5cabef09c7..4f4a646149304ea2585c39782b70b45234e63521 100644 (file)
@@ -381,6 +381,7 @@ eval_code2(PyCodeObject *co, PyObject *globals, PyObject *locals,
        /* Make it easier to find out where we are with a debugger */
        char *filename = PyString_AsString(co->co_filename);
 #endif
+       static PyObject *nextstr;
 
 /* Code access macros */
 
@@ -416,6 +417,11 @@ eval_code2(PyCodeObject *co, PyObject *globals, PyObject *locals,
                                     GETLOCAL(i) = value; } while (0)
 
 /* Start of code */
+       if (nextstr == NULL) {
+               nextstr = PyString_InternFromString("next");
+               if (nextstr == NULL)
+                       return NULL;
+       }
 
 #ifdef USE_STACKCHECK
        if (tstate->recursion_depth%10 == 0 && PyOS_CheckStack()) {
@@ -1875,6 +1881,41 @@ eval_code2(PyCodeObject *co, PyObject *globals, PyObject *locals,
                        JUMPTO(oparg);
                        continue;
 
+               case GET_ITER:
+                       /* before: [obj]; after [getiter(obj)] */
+                       v = POP();
+                       x = PyObject_GetIter(v);
+                       Py_DECREF(v);
+                       if (x != NULL) {
+                               w = x;
+                               x = PyObject_GetAttr(w, nextstr);
+                               Py_DECREF(w);
+                               if (x != NULL) {
+                                       PUSH(x);
+                                       continue;
+                               }
+                       }
+                       break;
+
+               case FOR_ITER:
+                       /* before: [iter]; after: [iter, iter()] *or* [] */
+                       v = TOP();
+                       x = PyObject_CallObject(v, NULL);
+                       if (x == NULL) {
+                               if (PyErr_ExceptionMatches(
+                                       PyExc_StopIteration))
+                               {
+                                       PyErr_Clear();
+                                       x = v = POP();
+                                       Py_DECREF(v);
+                                       JUMPBY(oparg);
+                                       continue;
+                               }
+                               break;
+                       }
+                       PUSH(x);
+                       continue;
+
                case FOR_LOOP:
                        /* for v in s: ...
                           On entry: stack contains s, i.
index 0939f059c40ff1bb2e5a377a3b3d91cad12e8580..cb85ce3e010748284ca7e57c3625850253656d0a 100644 (file)
@@ -1232,21 +1232,15 @@ parsestrplus(node *n)
 static void
 com_list_for(struct compiling *c, node *n, node *e, char *t)
 {
-       PyObject *v;
        int anchor = 0;
        int save_begin = c->c_begin;
 
        /* list_iter: for v in expr [list_iter] */
        com_node(c, CHILD(n, 3)); /* expr */
-       v = PyInt_FromLong(0L);
-       if (v == NULL)
-               c->c_errors++;
-       com_addoparg(c, LOAD_CONST, com_addconst(c, v));
-       com_push(c, 1);
-       Py_XDECREF(v);
+       com_addbyte(c, GET_ITER);
        c->c_begin = c->c_nexti;
        com_addoparg(c, SET_LINENO, n->n_lineno);
-       com_addfwref(c, FOR_LOOP, &anchor);
+       com_addfwref(c, FOR_ITER, &anchor);
        com_push(c, 1);
        com_assign(c, CHILD(n, 1), OP_ASSIGN, NULL);
        c->c_loops++;
@@ -1255,7 +1249,7 @@ com_list_for(struct compiling *c, node *n, node *e, char *t)
        com_addoparg(c, JUMP_ABSOLUTE, c->c_begin);
        c->c_begin = save_begin;
        com_backpatch(c, anchor);
-       com_pop(c, 2); /* FOR_LOOP has popped these */
+       com_pop(c, 1); /* FOR_ITER has popped this */
 }  
 
 static void
@@ -2873,7 +2867,6 @@ com_while_stmt(struct compiling *c, node *n)
 static void
 com_for_stmt(struct compiling *c, node *n)
 {
-       PyObject *v;
        int break_anchor = 0;
        int anchor = 0;
        int save_begin = c->c_begin;
@@ -2882,15 +2875,10 @@ com_for_stmt(struct compiling *c, node *n)
        com_addfwref(c, SETUP_LOOP, &break_anchor);
        block_push(c, SETUP_LOOP);
        com_node(c, CHILD(n, 3));
-       v = PyInt_FromLong(0L);
-       if (v == NULL)
-               c->c_errors++;
-       com_addoparg(c, LOAD_CONST, com_addconst(c, v));
-       com_push(c, 1);
-       Py_XDECREF(v);
+       com_addbyte(c, GET_ITER);
        c->c_begin = c->c_nexti;
        com_addoparg(c, SET_LINENO, n->n_lineno);
-       com_addfwref(c, FOR_LOOP, &anchor);
+       com_addfwref(c, FOR_ITER, &anchor);
        com_push(c, 1);
        com_assign(c, CHILD(n, 1), OP_ASSIGN, NULL);
        c->c_loops++;
@@ -2899,7 +2887,7 @@ com_for_stmt(struct compiling *c, node *n)
        com_addoparg(c, JUMP_ABSOLUTE, c->c_begin);
        c->c_begin = save_begin;
        com_backpatch(c, anchor);
-       com_pop(c, 2); /* FOR_LOOP has popped these */
+       com_pop(c, 1); /* FOR_ITER has popped this */
        com_addbyte(c, POP_BLOCK);
        block_pop(c, SETUP_LOOP);
        if (NCH(n) > 8)
index ad8021e0db1a0c26d042cd00943d34aae18c1e82..214d8e5e623bd5aa3fd709c10d289b6fdf44edfc 100644 (file)
@@ -52,6 +52,7 @@ recommended that user defined class based exceptions be derived from the\n\
 Exception\n\
  |\n\
  +-- SystemExit\n\
+ +-- StopIteration\n\
  +-- StandardError\n\
  |    |\n\
  |    +-- KeyboardInterrupt\n\
@@ -369,6 +370,9 @@ StandardError__doc__[] = "Base class for all standard Python exceptions.";
 static char
 TypeError__doc__[] = "Inappropriate argument type.";
 
+static char
+StopIteration__doc__[] = "Signal the end from iterator.next().";
+
 
 \f
 static char
@@ -924,6 +928,7 @@ static PyMethodDef functions[] = {
 /* Global C API defined exceptions */
 
 PyObject *PyExc_Exception;
+PyObject *PyExc_StopIteration;
 PyObject *PyExc_StandardError;
 PyObject *PyExc_ArithmeticError;
 PyObject *PyExc_LookupError;
@@ -985,6 +990,8 @@ static struct {
   * The first three classes MUST appear in exactly this order
   */
  {"Exception", &PyExc_Exception},
+ {"StopIteration", &PyExc_StopIteration, &PyExc_Exception,
+  StopIteration__doc__},
  {"StandardError", &PyExc_StandardError, &PyExc_Exception,
   StandardError__doc__},
  {"TypeError", &PyExc_TypeError, 0, TypeError__doc__},
index bde49ce91cb4d0bfcba29bb2eb188803dd4ae70e..2c15d1050c84266d9c953d315c19962d0711d4e5 100644 (file)
@@ -43,7 +43,7 @@ extern time_t PyOS_GetLastModificationTime(char *, FILE *);
 /* XXX Perhaps the magic number should be frozen and a version field
    added to the .pyc file header? */
 /* New way to come up with the magic number: (YEAR-1995), MONTH, DAY */
-#define MAGIC (60202 | ((long)'\r'<<16) | ((long)'\n'<<24))
+#define MAGIC (60420 | ((long)'\r'<<16) | ((long)'\n'<<24))
 
 /* Magic word as global; note that _PyImport_Init() can change the
    value of this global to accommodate for alterations of how the