diff options
author | Nicola Pero <nicola.pero@meta-innovation.com> | 2011-10-14 10:19:01 +0000 |
---|---|---|
committer | Nicola Pero <nicola@gcc.gnu.org> | 2011-10-14 10:19:01 +0000 |
commit | 3cc2dd4b7eb3d60c24b9cebecb9cd3fb1ab94184 (patch) | |
tree | ef62f9d9f5ec292c126959f551815e48323b3a22 /gcc/objc/objc-map.c | |
parent | 5e678de8bada6db595df5addc0a2db6e7fc9cf32 (diff) | |
download | gcc-3cc2dd4b7eb3d60c24b9cebecb9cd3fb1ab94184.zip gcc-3cc2dd4b7eb3d60c24b9cebecb9cd3fb1ab94184.tar.gz gcc-3cc2dd4b7eb3d60c24b9cebecb9cd3fb1ab94184.tar.bz2 |
In gcc/: 2011-10-14 Nicola Pero <nicola.pero@meta-innovation.com>
In gcc/:
2011-10-14 Nicola Pero <nicola.pero@meta-innovation.com>
* gengtype.c (files_rules): Added rules for objc/objc-map.h and
objc/objc-map.c.
In gcc/objc/:
2011-10-14 Nicola Pero <nicola.pero@meta-innovation.com>
* objc-map.h: New file.
* objc-map.c: New file.
* config-lang.in (gtfiles): Added objc-map.h.
* Make-lang.in (OBJC_OBJS): Added objc-map.o.
(objc/objc-map.o): New rule.
(objc/objc-act.o): Depend on objc/objc-map.h.
* objc-next-runtime-abi-02.c: Added a TODO comment.
* objc-act.c: Include objc-map.h.
(nst_method_hash_list, cls_method_hash_list): Removed.
(instance_method_map, class_method_map): New.
(cls_name_hash_list, als_name_hash_list): Removed.
(class_name_map, alias_name_map): Removed.
(ivar_offset_hash_list): Removed.
(hash_class_name_enter, hash_class_name_lookup, hash_enter,
hash_lookup, hash_add_attr, add_method_to_hash_list): Removed.
(interface_hash_init): New.
(objc_init): Call interface_hash_init.
(objc_write_global_declarations): Iterate over class_method_map
and instance_method_map instead of cls_method_hash_list and
nst_method_hash_list.
(objc_declare_alias): Use alias_name_map instead of
cls_name_hash_list.
(objc_is_class_name): Use class_name_map and alias_name_map
instead of cls_name_hash_list and als_name_hash_list.
(interface_tuple, interface_htab, hash_interface, eq_interface):
Removed.
(interface_map): New.
(add_class): Renamed to add_interface. Use interface_map instead
of interface_htab.
(lookup_interface): Use interface_map instead of interface_htab.
(check_duplicates): Changed first argument to be a tree,
potentially a TREE_VEC, instead of a hash. Changed implementation
to match.
(lookup_method_in_hash_lists): Use class_method_map and
instance_method_map instead of cls_method_hash_list and
nst_method_hash_list.
(objc_build_selector_expr): Likewise.
(hash_func): Removed.
(hash_init): Create instance_method_map, class_method_map,
class_name_map, and alias_name_map. Do not create
nst_method_hash_list, cls_method_hash_list, cls_name_hash_list,
als_name_hash_list, and ivar_offset_hash_list.
(insert_method_into_method_map): New.
(objc_add_method): Use insert_method_into_method_map instead of
add_method_to_hash_list.
(start_class): Call add_interface instead of add_class.
* objc-act.h (cls_name_hash_list, als_name_hash_list,
nst_method_hash_list, cls_method_hash_list): Removed.
In gcc/objcp/:
2011-10-14 Nicola Pero <nicola.pero@meta-innovation.com>
* Make-lang.in (OBJCXX_OBJS): Added objc-map.o.
(objcp/objc-map.o): New rule.
(objcp/objcp-act.o): Depend on objc/objc-map.h.
* config-lang.in (gtfiles): Added objc-map.h.
From-SVN: r179965
Diffstat (limited to 'gcc/objc/objc-map.c')
-rw-r--r-- | gcc/objc/objc-map.c | 161 |
1 files changed, 161 insertions, 0 deletions
diff --git a/gcc/objc/objc-map.c b/gcc/objc/objc-map.c new file mode 100644 index 0000000..77a98c1 --- /dev/null +++ b/gcc/objc/objc-map.c @@ -0,0 +1,161 @@ +/* objc-map.c -- Implementation of map data structures for ObjC compiler + Copyright 2011 Free Software Foundation, Inc. + Written by Nicola Pero <nicola.pero@meta-innovation.com> + +This program is free software; you can redistribute it and/or modify it +under the terms of the GNU Lesser Public License as published by the +Free Software Foundation; either version 3, or (at your option) any +later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU Lesser Public License for more details. + +You should have received a copy of the GNU Lesser Public License +along with this program; if not, write to the Free Software +Foundation, 51 Franklin Street - Fifth Floor, +Boston, MA 02110-1301, USA. */ + +#include "config.h" +#include "system.h" +#include "coretypes.h" +#include "tree.h" +#include "ggc.h" +#include "objc-map.h" + +#define OUT_OF_MEMORY { fprintf (stderr, "Out of memory\n"); abort (); } + +static +size_t +ATTRIBUTE_PURE +next_power_of_two (size_t x) +{ + size_t result = 1; + + if (x < 2) + return 2; + + /* Avoid the long calculation if x is already a power of two. Since + we internally always increase/shrink tables by powers of 2, the + calculation should only be done once, when the table is first + set up. */ + if ((x & (x - 1)) == 0) + return x; + + /* Calculate log_2 by counting how many times we can divide by 2 + before reaching 0. */ + while (x > 0) + { + x = x >> 1; + result = result << 1; + } + return result; +} + +objc_map_t +objc_map_alloc_ggc (size_t initial_capacity) +{ + objc_map_t map = (objc_map_t) ggc_internal_cleared_vec_alloc_stat (1, sizeof (struct objc_map_private)); + if (map == NULL) + OUT_OF_MEMORY; + + initial_capacity = next_power_of_two (initial_capacity); + + map->number_of_slots = initial_capacity; + map->mask = initial_capacity - 1; + map->maximum_load_factor = 70; + map->max_number_of_non_empty_slots = (initial_capacity * map->maximum_load_factor) / 100; + + map->slots = (tree *)ggc_internal_cleared_vec_alloc_stat (initial_capacity, sizeof (tree)); + map->values = (tree *)ggc_internal_cleared_vec_alloc_stat (initial_capacity, sizeof (tree)); + + if (map->slots == NULL) + OUT_OF_MEMORY; + + if (map->values == NULL) + OUT_OF_MEMORY; + + return map; +} + +void +objc_map_set_maximum_load_factor (objc_map_t map, int number_between_zero_and_one_hundred) +{ + if (map->number_of_non_empty_slots != 0) + return; + + map->maximum_load_factor = number_between_zero_and_one_hundred; + map->max_number_of_non_empty_slots = (map->number_of_slots * number_between_zero_and_one_hundred) / 100; +} + +int +objc_map_maximum_load_factor (objc_map_t map) +{ + return map->maximum_load_factor; +} + +static void +objc_map_private_resize (objc_map_t map, size_t new_number_of_slots) +{ + tree *old_slots = map->slots; + tree *old_values = map->values; + size_t i, old_number_of_slots = map->number_of_slots; + + if (new_number_of_slots < (map->number_of_non_empty_slots)) + new_number_of_slots = 2 * map->number_of_non_empty_slots; + + new_number_of_slots = next_power_of_two (new_number_of_slots); + + map->number_of_slots = new_number_of_slots; + map->mask = map->number_of_slots - 1; + map->max_number_of_non_empty_slots = (map->number_of_slots * map->maximum_load_factor) / 100; + + + map->slots = (tree *)ggc_internal_cleared_vec_alloc_stat (map->number_of_slots, sizeof (tree)); + map->values = (tree *)ggc_internal_cleared_vec_alloc_stat (map->number_of_slots, sizeof (tree)); + + if (map->slots == NULL) + OUT_OF_MEMORY; + + if (map->values == NULL) + OUT_OF_MEMORY; + + for (i = 0; i < old_number_of_slots; i++) + if (old_slots[i] != OBJC_MAP_PRIVATE_EMPTY_SLOT) + { + size_t k = IDENTIFIER_HASH_VALUE (old_slots[i]) & map->mask; + + if (map->slots[k] == OBJC_MAP_PRIVATE_EMPTY_SLOT) + { + map->slots[k] = old_slots[i]; + map->values[k] = old_values[i]; + } + else + { + size_t j = 1; + while (1) + { + k = (k + j) & map->mask; + if (map->slots[k] == OBJC_MAP_PRIVATE_EMPTY_SLOT) + { + map->slots[k] = old_slots[i]; + map->values[k] = old_values[i]; + break; + } + j++; + } + } + } + + ggc_free (old_slots); + ggc_free (old_values); +} + +void +objc_map_private_grow (struct objc_map_private *map) +{ + objc_map_private_resize (map, map->number_of_slots * 2); +} + +#include "gt-objc-objc-map.h" |