From 8ca92ae54c3c0958bf073fe04d897f8f01e02547 Mon Sep 17 00:00:00 2001 From: Raymond Hettinger Date: Thu, 11 Mar 2004 09:13:12 +0000 Subject: [PATCH] Eliminate a big block of duplicate code in PySequence_List() by exposing _PyList_Extend(). --- Include/listobject.h | 1 + Objects/abstract.c | 60 +++++--------------------------------------- Objects/listobject.c | 6 +++++ 3 files changed, 13 insertions(+), 54 deletions(-) diff --git a/Include/listobject.h b/Include/listobject.h index 6221b80d1f9..b4935ae8946 100644 --- a/Include/listobject.h +++ b/Include/listobject.h @@ -41,6 +41,7 @@ PyAPI_FUNC(int) PyList_SetSlice(PyObject *, int, int, PyObject *); PyAPI_FUNC(int) PyList_Sort(PyObject *); PyAPI_FUNC(int) PyList_Reverse(PyObject *); PyAPI_FUNC(PyObject *) PyList_AsTuple(PyObject *); +PyAPI_FUNC(PyObject *) _PyList_Extend(PyListObject *, PyObject *); /* Macro, trading safety for speed */ #define PyList_GET_ITEM(op, i) (((PyListObject *)(op))->ob_item[i]) diff --git a/Objects/abstract.c b/Objects/abstract.c index 4ac92608ab9..90efd9010f2 100644 --- a/Objects/abstract.c +++ b/Objects/abstract.c @@ -1427,69 +1427,21 @@ PySequence_Tuple(PyObject *v) PyObject * PySequence_List(PyObject *v) { - PyObject *it; /* iter(v) */ PyObject *result; /* result list */ - int n; /* guess for result list size */ - int i; + PyObject *rv; /* return value from PyList_Extend */ if (v == NULL) return null_error(); - /* Special-case list(a_list), for speed. */ - if (PyList_Check(v)) - return PyList_GetSlice(v, 0, PyList_GET_SIZE(v)); - - /* Get iterator. There may be some low-level efficiency to be gained - * by caching the tp_iternext slot instead of using PyIter_Next() - * later, but premature optimization is the root etc. - */ - it = PyObject_GetIter(v); - if (it == NULL) + result = PyList_New(0); + if (result == NULL) return NULL; - /* Guess a result list size. */ - n = PyObject_Size(v); - if (n < 0) { - PyErr_Clear(); - n = 8; /* arbitrary */ - } - result = PyList_New(n); - if (result == NULL) { - Py_DECREF(it); + rv = _PyList_Extend((PyListObject *)result, v); + if (rv == NULL) { + Py_DECREF(result); return NULL; } - - /* Run iterator to exhaustion. */ - for (i = 0; ; i++) { - PyObject *item = PyIter_Next(it); - if (item == NULL) { - if (PyErr_Occurred()) { - Py_DECREF(result); - result = NULL; - } - break; - } - if (i < n) - PyList_SET_ITEM(result, i, item); /* steals ref */ - else { - int status = PyList_Append(result, item); - Py_DECREF(item); /* append creates a new ref */ - if (status < 0) { - Py_DECREF(result); - result = NULL; - break; - } - } - } - - /* Cut back result list if initial guess was too large. */ - if (i < n && result != NULL) { - if (PyList_SetSlice(result, i, n, (PyObject *)NULL) != 0) { - Py_DECREF(result); - result = NULL; - } - } - Py_DECREF(it); return result; } diff --git a/Objects/listobject.c b/Objects/listobject.c index 6bb6d8c7973..ed6ed3e5610 100644 --- a/Objects/listobject.c +++ b/Objects/listobject.c @@ -776,6 +776,12 @@ listextend(PyListObject *self, PyObject *b) return NULL; } +PyObject * +_PyList_Extend(PyListObject *self, PyObject *b) +{ + return listextend(self, b); +} + static PyObject * list_inplace_concat(PyListObject *self, PyObject *other) {