blob: 4d0af14008b9390ef13c0555a1fcf437b8d285bc [file] [log] [blame]
Guido van Rossum7dab2422002-04-26 19:40:56 +00001/* enumerate object */
2
3#include "Python.h"
4
Serhiy Storchaka41baebd2017-01-19 18:48:17 +02005#include "clinic/enumobject.c.h"
6
7/*[clinic input]
8class enumerate "enumobject *" "&PyEnum_Type"
9class reversed "reversedobject *" "&PyReversed_Type"
10[clinic start generated code]*/
11/*[clinic end generated code: output=da39a3ee5e6b4b0d input=d2dfdf1a88c88975]*/
12
Guido van Rossum7dab2422002-04-26 19:40:56 +000013typedef struct {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000014 PyObject_HEAD
15 Py_ssize_t en_index; /* current index of enumeration */
16 PyObject* en_sit; /* secondary iterator of enumeration */
17 PyObject* en_result; /* result tuple */
18 PyObject* en_longindex; /* index for sequences >= PY_SSIZE_T_MAX */
Guido van Rossum7dab2422002-04-26 19:40:56 +000019} enumobject;
20
Serhiy Storchaka41baebd2017-01-19 18:48:17 +020021
22/*[clinic input]
23@classmethod
24enumerate.__new__ as enum_new
25
26 iterable: object
27 an object supporting iteration
28 start: object = 0
29
30Return an enumerate object.
31
32The enumerate object yields pairs containing a count (from start, which
33defaults to zero) and a value yielded by the iterable argument.
34
35enumerate is useful for obtaining an indexed list:
36 (0, seq[0]), (1, seq[1]), (2, seq[2]), ...
37[clinic start generated code]*/
38
Guido van Rossum7dab2422002-04-26 19:40:56 +000039static PyObject *
Serhiy Storchaka41baebd2017-01-19 18:48:17 +020040enum_new_impl(PyTypeObject *type, PyObject *iterable, PyObject *start)
41/*[clinic end generated code: output=e95e6e439f812c10 input=782e4911efcb8acf]*/
Guido van Rossum7dab2422002-04-26 19:40:56 +000042{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000043 enumobject *en;
Guido van Rossum7dab2422002-04-26 19:40:56 +000044
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000045 en = (enumobject *)type->tp_alloc(type, 0);
46 if (en == NULL)
47 return NULL;
48 if (start != NULL) {
49 start = PyNumber_Index(start);
50 if (start == NULL) {
51 Py_DECREF(en);
52 return NULL;
53 }
54 assert(PyLong_Check(start));
55 en->en_index = PyLong_AsSsize_t(start);
56 if (en->en_index == -1 && PyErr_Occurred()) {
57 PyErr_Clear();
58 en->en_index = PY_SSIZE_T_MAX;
59 en->en_longindex = start;
60 } else {
61 en->en_longindex = NULL;
62 Py_DECREF(start);
63 }
64 } else {
65 en->en_index = 0;
66 en->en_longindex = NULL;
67 }
Serhiy Storchaka41baebd2017-01-19 18:48:17 +020068 en->en_sit = PyObject_GetIter(iterable);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000069 if (en->en_sit == NULL) {
70 Py_DECREF(en);
71 return NULL;
72 }
73 en->en_result = PyTuple_Pack(2, Py_None, Py_None);
74 if (en->en_result == NULL) {
75 Py_DECREF(en);
76 return NULL;
77 }
78 return (PyObject *)en;
Guido van Rossum7dab2422002-04-26 19:40:56 +000079}
80
81static void
82enum_dealloc(enumobject *en)
83{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000084 PyObject_GC_UnTrack(en);
85 Py_XDECREF(en->en_sit);
86 Py_XDECREF(en->en_result);
87 Py_XDECREF(en->en_longindex);
88 Py_TYPE(en)->tp_free(en);
Guido van Rossum7dab2422002-04-26 19:40:56 +000089}
90
91static int
92enum_traverse(enumobject *en, visitproc visit, void *arg)
93{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000094 Py_VISIT(en->en_sit);
95 Py_VISIT(en->en_result);
96 Py_VISIT(en->en_longindex);
97 return 0;
Guido van Rossum7dab2422002-04-26 19:40:56 +000098}
99
100static PyObject *
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000101enum_next_long(enumobject *en, PyObject* next_item)
102{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000103 PyObject *result = en->en_result;
104 PyObject *next_index;
105 PyObject *stepped_up;
Raymond Hettinger8110dbd2017-09-25 02:15:53 -0700106 PyObject *old_index;
107 PyObject *old_item;
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000108
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000109 if (en->en_longindex == NULL) {
110 en->en_longindex = PyLong_FromSsize_t(PY_SSIZE_T_MAX);
Serhiy Storchaka0e950dd2017-09-26 08:14:58 +0300111 if (en->en_longindex == NULL) {
112 Py_DECREF(next_item);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000113 return NULL;
Serhiy Storchaka0e950dd2017-09-26 08:14:58 +0300114 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000115 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000116 next_index = en->en_longindex;
117 assert(next_index != NULL);
Serhiy Storchakaba85d692017-03-30 09:09:41 +0300118 stepped_up = PyNumber_Add(next_index, _PyLong_One);
Serhiy Storchaka0e950dd2017-09-26 08:14:58 +0300119 if (stepped_up == NULL) {
120 Py_DECREF(next_item);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000121 return NULL;
Serhiy Storchaka0e950dd2017-09-26 08:14:58 +0300122 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000123 en->en_longindex = stepped_up;
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000124
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000125 if (result->ob_refcnt == 1) {
126 Py_INCREF(result);
Raymond Hettinger8110dbd2017-09-25 02:15:53 -0700127 old_index = PyTuple_GET_ITEM(result, 0);
128 old_item = PyTuple_GET_ITEM(result, 1);
129 PyTuple_SET_ITEM(result, 0, next_index);
130 PyTuple_SET_ITEM(result, 1, next_item);
131 Py_DECREF(old_index);
132 Py_DECREF(old_item);
133 return result;
134 }
135 result = PyTuple_New(2);
136 if (result == NULL) {
137 Py_DECREF(next_index);
138 Py_DECREF(next_item);
139 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000140 }
141 PyTuple_SET_ITEM(result, 0, next_index);
142 PyTuple_SET_ITEM(result, 1, next_item);
143 return result;
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000144}
145
146static PyObject *
Guido van Rossum7dab2422002-04-26 19:40:56 +0000147enum_next(enumobject *en)
148{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000149 PyObject *next_index;
150 PyObject *next_item;
151 PyObject *result = en->en_result;
152 PyObject *it = en->en_sit;
Raymond Hettinger8110dbd2017-09-25 02:15:53 -0700153 PyObject *old_index;
154 PyObject *old_item;
Guido van Rossum7dab2422002-04-26 19:40:56 +0000155
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000156 next_item = (*Py_TYPE(it)->tp_iternext)(it);
157 if (next_item == NULL)
158 return NULL;
Guido van Rossumca5ed5b2002-07-16 21:02:42 +0000159
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000160 if (en->en_index == PY_SSIZE_T_MAX)
161 return enum_next_long(en, next_item);
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000162
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000163 next_index = PyLong_FromSsize_t(en->en_index);
164 if (next_index == NULL) {
165 Py_DECREF(next_item);
166 return NULL;
167 }
168 en->en_index++;
Raymond Hettingere8b0f042003-05-28 14:05:34 +0000169
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000170 if (result->ob_refcnt == 1) {
171 Py_INCREF(result);
Raymond Hettinger8110dbd2017-09-25 02:15:53 -0700172 old_index = PyTuple_GET_ITEM(result, 0);
173 old_item = PyTuple_GET_ITEM(result, 1);
174 PyTuple_SET_ITEM(result, 0, next_index);
175 PyTuple_SET_ITEM(result, 1, next_item);
176 Py_DECREF(old_index);
177 Py_DECREF(old_item);
178 return result;
179 }
180 result = PyTuple_New(2);
181 if (result == NULL) {
182 Py_DECREF(next_index);
183 Py_DECREF(next_item);
184 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000185 }
186 PyTuple_SET_ITEM(result, 0, next_index);
187 PyTuple_SET_ITEM(result, 1, next_item);
188 return result;
Guido van Rossum7dab2422002-04-26 19:40:56 +0000189}
190
Kristján Valur Jónsson31668b82012-04-03 10:49:41 +0000191static PyObject *
192enum_reduce(enumobject *en)
193{
194 if (en->en_longindex != NULL)
195 return Py_BuildValue("O(OO)", Py_TYPE(en), en->en_sit, en->en_longindex);
196 else
197 return Py_BuildValue("O(On)", Py_TYPE(en), en->en_sit, en->en_index);
198}
199
200PyDoc_STRVAR(reduce_doc, "Return state information for pickling.");
201
202static PyMethodDef enum_methods[] = {
203 {"__reduce__", (PyCFunction)enum_reduce, METH_NOARGS, reduce_doc},
204 {NULL, NULL} /* sentinel */
205};
206
Guido van Rossum7dab2422002-04-26 19:40:56 +0000207PyTypeObject PyEnum_Type = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000208 PyVarObject_HEAD_INIT(&PyType_Type, 0)
209 "enumerate", /* tp_name */
210 sizeof(enumobject), /* tp_basicsize */
211 0, /* tp_itemsize */
212 /* methods */
213 (destructor)enum_dealloc, /* tp_dealloc */
214 0, /* tp_print */
215 0, /* tp_getattr */
216 0, /* tp_setattr */
217 0, /* tp_reserved */
218 0, /* tp_repr */
219 0, /* tp_as_number */
220 0, /* tp_as_sequence */
221 0, /* tp_as_mapping */
222 0, /* tp_hash */
223 0, /* tp_call */
224 0, /* tp_str */
225 PyObject_GenericGetAttr, /* tp_getattro */
226 0, /* tp_setattro */
227 0, /* tp_as_buffer */
228 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200229 Py_TPFLAGS_BASETYPE, /* tp_flags */
230 enum_new__doc__, /* tp_doc */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000231 (traverseproc)enum_traverse, /* tp_traverse */
232 0, /* tp_clear */
233 0, /* tp_richcompare */
234 0, /* tp_weaklistoffset */
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200235 PyObject_SelfIter, /* tp_iter */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000236 (iternextfunc)enum_next, /* tp_iternext */
Kristján Valur Jónsson31668b82012-04-03 10:49:41 +0000237 enum_methods, /* tp_methods */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000238 0, /* tp_members */
239 0, /* tp_getset */
240 0, /* tp_base */
241 0, /* tp_dict */
242 0, /* tp_descr_get */
243 0, /* tp_descr_set */
244 0, /* tp_dictoffset */
245 0, /* tp_init */
246 PyType_GenericAlloc, /* tp_alloc */
247 enum_new, /* tp_new */
248 PyObject_GC_Del, /* tp_free */
Guido van Rossum7dab2422002-04-26 19:40:56 +0000249};
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000250
251/* Reversed Object ***************************************************************/
252
253typedef struct {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000254 PyObject_HEAD
255 Py_ssize_t index;
256 PyObject* seq;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000257} reversedobject;
258
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200259/*[clinic input]
260@classmethod
261reversed.__new__ as reversed_new
262
263 sequence as seq: object
264 /
265
266Return a reverse iterator over the values of the given sequence.
267[clinic start generated code]*/
268
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000269static PyObject *
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200270reversed_new_impl(PyTypeObject *type, PyObject *seq)
271/*[clinic end generated code: output=f7854cc1df26f570 input=aeb720361e5e3f1d]*/
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000272{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000273 Py_ssize_t n;
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200274 PyObject *reversed_meth;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000275 reversedobject *ro;
Benjamin Petersonce798522012-01-22 11:24:29 -0500276 _Py_IDENTIFIER(__reversed__);
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000277
Benjamin Petersonce798522012-01-22 11:24:29 -0500278 reversed_meth = _PyObject_LookupSpecial(seq, &PyId___reversed__);
Guido van Rossum97c1adf2016-08-18 09:22:23 -0700279 if (reversed_meth == Py_None) {
280 Py_DECREF(reversed_meth);
281 PyErr_Format(PyExc_TypeError,
282 "'%.200s' object is not reversible",
283 Py_TYPE(seq)->tp_name);
284 return NULL;
285 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000286 if (reversed_meth != NULL) {
Victor Stinnerf17c3de2016-12-06 18:46:19 +0100287 PyObject *res = _PyObject_CallNoArg(reversed_meth);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000288 Py_DECREF(reversed_meth);
289 return res;
290 }
291 else if (PyErr_Occurred())
292 return NULL;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000293
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000294 if (!PySequence_Check(seq)) {
Guido van Rossum97c1adf2016-08-18 09:22:23 -0700295 PyErr_Format(PyExc_TypeError,
296 "'%.200s' object is not reversible",
297 Py_TYPE(seq)->tp_name);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000298 return NULL;
299 }
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000300
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000301 n = PySequence_Size(seq);
302 if (n == -1)
303 return NULL;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000304
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000305 ro = (reversedobject *)type->tp_alloc(type, 0);
306 if (ro == NULL)
307 return NULL;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000308
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000309 ro->index = n-1;
310 Py_INCREF(seq);
311 ro->seq = seq;
312 return (PyObject *)ro;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000313}
314
315static void
316reversed_dealloc(reversedobject *ro)
317{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000318 PyObject_GC_UnTrack(ro);
319 Py_XDECREF(ro->seq);
320 Py_TYPE(ro)->tp_free(ro);
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000321}
322
323static int
324reversed_traverse(reversedobject *ro, visitproc visit, void *arg)
325{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000326 Py_VISIT(ro->seq);
327 return 0;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000328}
329
330static PyObject *
331reversed_next(reversedobject *ro)
332{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000333 PyObject *item;
334 Py_ssize_t index = ro->index;
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000335
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000336 if (index >= 0) {
337 item = PySequence_GetItem(ro->seq, index);
338 if (item != NULL) {
339 ro->index--;
340 return item;
341 }
342 if (PyErr_ExceptionMatches(PyExc_IndexError) ||
343 PyErr_ExceptionMatches(PyExc_StopIteration))
344 PyErr_Clear();
345 }
346 ro->index = -1;
347 Py_CLEAR(ro->seq);
348 return NULL;
Raymond Hettinger029dba52004-02-10 09:33:39 +0000349}
350
Raymond Hettinger6b27cda2005-09-24 21:23:05 +0000351static PyObject *
Raymond Hettingeref9bf402004-03-10 10:10:42 +0000352reversed_len(reversedobject *ro)
353{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000354 Py_ssize_t position, seqsize;
Raymond Hettinger7892b1c2004-04-12 18:10:01 +0000355
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000356 if (ro->seq == NULL)
357 return PyLong_FromLong(0);
358 seqsize = PySequence_Size(ro->seq);
359 if (seqsize == -1)
360 return NULL;
361 position = ro->index + 1;
362 return PyLong_FromSsize_t((seqsize < position) ? 0 : position);
Raymond Hettingeref9bf402004-03-10 10:10:42 +0000363}
364
Armin Rigof5b3e362006-02-11 21:32:43 +0000365PyDoc_STRVAR(length_hint_doc, "Private method returning an estimate of len(list(it)).");
Raymond Hettinger6b27cda2005-09-24 21:23:05 +0000366
Kristján Valur Jónsson31668b82012-04-03 10:49:41 +0000367static PyObject *
368reversed_reduce(reversedobject *ro)
369{
370 if (ro->seq)
371 return Py_BuildValue("O(O)n", Py_TYPE(ro), ro->seq, ro->index);
372 else
373 return Py_BuildValue("O(())", Py_TYPE(ro));
374}
375
376static PyObject *
377reversed_setstate(reversedobject *ro, PyObject *state)
378{
379 Py_ssize_t index = PyLong_AsSsize_t(state);
380 if (index == -1 && PyErr_Occurred())
381 return NULL;
382 if (ro->seq != 0) {
383 Py_ssize_t n = PySequence_Size(ro->seq);
384 if (n < 0)
385 return NULL;
386 if (index < -1)
387 index = -1;
388 else if (index > n-1)
389 index = n-1;
390 ro->index = index;
391 }
392 Py_RETURN_NONE;
393}
394
395PyDoc_STRVAR(setstate_doc, "Set state information for unpickling.");
396
Raymond Hettinger6b27cda2005-09-24 21:23:05 +0000397static PyMethodDef reversediter_methods[] = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000398 {"__length_hint__", (PyCFunction)reversed_len, METH_NOARGS, length_hint_doc},
Kristján Valur Jónsson31668b82012-04-03 10:49:41 +0000399 {"__reduce__", (PyCFunction)reversed_reduce, METH_NOARGS, reduce_doc},
400 {"__setstate__", (PyCFunction)reversed_setstate, METH_O, setstate_doc},
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000401 {NULL, NULL} /* sentinel */
Raymond Hettinger06353f72004-02-08 10:49:42 +0000402};
403
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000404PyTypeObject PyReversed_Type = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000405 PyVarObject_HEAD_INIT(&PyType_Type, 0)
406 "reversed", /* tp_name */
407 sizeof(reversedobject), /* tp_basicsize */
408 0, /* tp_itemsize */
409 /* methods */
410 (destructor)reversed_dealloc, /* tp_dealloc */
411 0, /* tp_print */
412 0, /* tp_getattr */
413 0, /* tp_setattr */
414 0, /* tp_reserved */
415 0, /* tp_repr */
416 0, /* tp_as_number */
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200417 0, /* tp_as_sequence */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000418 0, /* tp_as_mapping */
419 0, /* tp_hash */
420 0, /* tp_call */
421 0, /* tp_str */
422 PyObject_GenericGetAttr, /* tp_getattro */
423 0, /* tp_setattro */
424 0, /* tp_as_buffer */
425 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200426 Py_TPFLAGS_BASETYPE, /* tp_flags */
427 reversed_new__doc__, /* tp_doc */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000428 (traverseproc)reversed_traverse,/* tp_traverse */
429 0, /* tp_clear */
430 0, /* tp_richcompare */
431 0, /* tp_weaklistoffset */
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200432 PyObject_SelfIter, /* tp_iter */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000433 (iternextfunc)reversed_next, /* tp_iternext */
Serhiy Storchaka41baebd2017-01-19 18:48:17 +0200434 reversediter_methods, /* tp_methods */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000435 0, /* tp_members */
436 0, /* tp_getset */
437 0, /* tp_base */
438 0, /* tp_dict */
439 0, /* tp_descr_get */
440 0, /* tp_descr_set */
441 0, /* tp_dictoffset */
442 0, /* tp_init */
443 PyType_GenericAlloc, /* tp_alloc */
444 reversed_new, /* tp_new */
445 PyObject_GC_Del, /* tp_free */
Raymond Hettinger85c20a42003-11-06 14:06:48 +0000446};