diff options
Diffstat (limited to 'Objects/enumobject.c')
| -rw-r--r-- | Objects/enumobject.c | 55 | 
1 files changed, 49 insertions, 6 deletions
| diff --git a/Objects/enumobject.c b/Objects/enumobject.c index 6997fdccf7..0a3694d002 100644 --- a/Objects/enumobject.c +++ b/Objects/enumobject.c @@ -7,6 +7,7 @@ typedef struct {  	long      en_index;        /* current index of enumeration */  	PyObject* en_sit;          /* secondary iterator of enumeration */  	PyObject* en_result;	   /* result tuple  */ +	PyObject* en_longindex;	   /* index for sequences >= LONG_MAX */  } enumobject;  static PyObject * @@ -25,6 +26,7 @@ enum_new(PyTypeObject *type, PyObject *args, PyObject *kwds)  		return NULL;  	en->en_index = 0;  	en->en_sit = PyObject_GetIter(seq); +	en->en_longindex = NULL;  	if (en->en_sit == NULL) {  		Py_DECREF(en);  		return NULL; @@ -43,6 +45,7 @@ enum_dealloc(enumobject *en)  	PyObject_GC_UnTrack(en);  	Py_XDECREF(en->en_sit);  	Py_XDECREF(en->en_result); +	Py_XDECREF(en->en_longindex);  	Py_Type(en)->tp_free(en);  } @@ -51,10 +54,53 @@ enum_traverse(enumobject *en, visitproc visit, void *arg)  {  	Py_VISIT(en->en_sit);  	Py_VISIT(en->en_result); +	Py_VISIT(en->en_longindex);  	return 0;  }  static PyObject * +enum_next_long(enumobject *en, PyObject* next_item) +{ +	static PyObject *one = NULL; +	PyObject *result = en->en_result; +	PyObject *next_index; +	PyObject *stepped_up; + +	if (en->en_longindex == NULL) { +		en->en_longindex = PyInt_FromLong(LONG_MAX); +		if (en->en_longindex == NULL) +			return NULL; +	} +	if (one == NULL) { +		one = PyInt_FromLong(1); +		if (one == NULL) +			return NULL; +	} +	next_index = en->en_longindex; +	assert(next_index != NULL); +	stepped_up = PyNumber_Add(next_index, one); +	if (stepped_up == NULL) +		return NULL; +	en->en_longindex = stepped_up; + +	if (result->ob_refcnt == 1) { +		Py_INCREF(result); +		Py_DECREF(PyTuple_GET_ITEM(result, 0)); +		Py_DECREF(PyTuple_GET_ITEM(result, 1)); +	} else { +		result = PyTuple_New(2); +		if (result == NULL) { +			Py_DECREF(next_index); +			Py_DECREF(next_item); +			return NULL; +		} +	} +	PyTuple_SET_ITEM(result, 0, next_index); +	PyTuple_SET_ITEM(result, 1, next_item); +	return result; +} + +static PyObject *  enum_next(enumobject *en)  {  	PyObject *next_index; @@ -62,16 +108,13 @@ enum_next(enumobject *en)  	PyObject *result = en->en_result;  	PyObject *it = en->en_sit; -	if (en->en_index == LONG_MAX) { -		PyErr_SetString(PyExc_OverflowError, -			"enumerate() is limited to LONG_MAX items");                 -		return NULL;          -	} -  	next_item = (*Py_Type(it)->tp_iternext)(it);  	if (next_item == NULL)  		return NULL; +	if (en->en_index == LONG_MAX) +		return enum_next_long(en, next_item); +  	next_index = PyInt_FromLong(en->en_index);  	if (next_index == NULL) {  		Py_DECREF(next_item); | 
