summaryrefslogtreecommitdiffstats
path: root/Objects/listobject.c
diff options
context:
space:
mode:
Diffstat (limited to 'Objects/listobject.c')
-rw-r--r--Objects/listobject.c61
1 files changed, 54 insertions, 7 deletions
diff --git a/Objects/listobject.c b/Objects/listobject.c
index 6992a0f..996c289 100644
--- a/Objects/listobject.c
+++ b/Objects/listobject.c
@@ -26,6 +26,8 @@ OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#include "allobjects.h"
#include "modsupport.h"
+#include "compile.h" /* Needed by ceval.h */
+#include "ceval.h" /* For call_object() */
object *
newlistobject(size)
@@ -471,11 +473,46 @@ listappend(self, args)
return ins(self, (int) self->ob_size, args);
}
+static object *cmpfunc;
+
static int
cmp(v, w)
char *v, *w;
{
- return cmpobject(* (object **) v, * (object **) w);
+ object *t, *res;
+ long i;
+
+ if (err_occurred())
+ return 0;
+
+ if (cmpfunc == NULL)
+ return cmpobject(* (object **) v, * (object **) w);
+
+ /* Call the user-supplied comparison function */
+ t = newtupleobject(2);
+ if (t == NULL)
+ return 0;
+ INCREF(* (object **) v);
+ settupleitem(t, 0, * (object **) v);
+ INCREF(* (object **) w);
+ settupleitem(t, 1, * (object **) w);
+ res = call_object(cmpfunc, t);
+ DECREF(t);
+ if (res == NULL)
+ return 0;
+ if (!is_intobject(res)) {
+ err_setstr(TypeError, "comparison function should return int");
+ i = 0;
+ }
+ else {
+ i = getintvalue(res);
+ if (i < 0)
+ i = -1;
+ else if (i > 0)
+ i = 1;
+ }
+ DECREF(res);
+ return (int) i;
}
static object *
@@ -483,14 +520,24 @@ listsort(self, args)
listobject *self;
object *args;
{
- if (args != NULL) {
- err_badarg();
- return NULL;
+ object *save_cmpfunc;
+ if (self->ob_size <= 1) {
+ INCREF(None);
+ return None;
+ }
+ save_cmpfunc = cmpfunc;
+ cmpfunc = args;
+ if (cmpfunc != NULL) {
+ /* Test the comparison function for obvious errors */
+ (void) cmp(&self->ob_item[0], &self->ob_item[1]);
+ if (err_occurred()) {
+ cmpfunc = save_cmpfunc;
+ return NULL;
+ }
}
- err_clear();
- if (self->ob_size > 1)
- qsort((char *)self->ob_item,
+ qsort((char *)self->ob_item,
(int) self->ob_size, sizeof(object *), cmp);
+ cmpfunc = save_cmpfunc;
if (err_occurred())
return NULL;
INCREF(None);