From 9275c182708ae80ac62eaf0e4152df7b88163076 Mon Sep 17 00:00:00 2001 From: Damien George Date: Fri, 3 Mar 2017 11:21:19 +1100 Subject: [PATCH] py/map: Fix bugs with deletion of elements from OrderedDict. There were 2 bugs, now fixed by this patch: - after deleting an element the len of the dict did not decrease by 1 - after deleting an element searching through the dict could lead to a seg fault due to there being an MP_OBJ_SENTINEL in the ordered array --- py/map.c | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/py/map.c b/py/map.c index c45f71e17c84..50d74f38f562 100644 --- a/py/map.c +++ b/py/map.c @@ -178,8 +178,14 @@ mp_map_elem_t *mp_map_lookup(mp_map_t *map, mp_obj_t index, mp_map_lookup_kind_t for (mp_map_elem_t *elem = &map->table[0], *top = &map->table[map->used]; elem < top; elem++) { if (elem->key == index || (!compare_only_ptrs && mp_obj_equal(elem->key, index))) { if (MP_UNLIKELY(lookup_kind == MP_MAP_LOOKUP_REMOVE_IF_FOUND)) { - elem->key = MP_OBJ_SENTINEL; - // keep elem->value so that caller can access it if needed + // remove the found element by moving the rest of the array down + mp_obj_t value = elem->value; + --map->used; + memmove(elem, elem + 1, (top - elem - 1) * sizeof(*elem)); + // put the found element after the end so the caller can access it if needed + elem = &map->table[map->used]; + elem->key = MP_OBJ_NULL; + elem->value = value; } return elem; } @@ -187,7 +193,6 @@ mp_map_elem_t *mp_map_lookup(mp_map_t *map, mp_obj_t index, mp_map_lookup_kind_t if (MP_LIKELY(lookup_kind != MP_MAP_LOOKUP_ADD_IF_NOT_FOUND)) { return NULL; } - // TODO shrink array down over any previously-freed slots if (map->used == map->alloc) { // TODO: Alloc policy map->alloc += 4;