summaryrefslogtreecommitdiff
path: root/devel/glib2/patches
diff options
context:
space:
mode:
authorprlw1 <prlw1@pkgsrc.org>2019-06-21 20:20:59 +0000
committerprlw1 <prlw1@pkgsrc.org>2019-06-21 20:20:59 +0000
commit78cce72a4682c94bca2bfb56409cd267aa562c5c (patch)
treeb264c4a2be5fe5601147dd3276d0a9c7b19ff54c /devel/glib2/patches
parent48c83b16fa0a46665400503690642c0102c8d773 (diff)
downloadpkgsrc-78cce72a4682c94bca2bfb56409cd267aa562c5c.tar.gz
Revert GHashTable improvements
https://gitlab.gnome.org/GNOME/glib/merge_requests/208 to fix PR pkg/54310 For the record, the patch was created with: git checkout -f 2.60.4 git revert --no-edit 86c6f7e2b..3bed8a13b git revert --no-edit 75f8ec1df9b48b0c3a13a9125f2c7d7c5adf5159 git revert --no-edit 603fb5958..d3074a748 git revert --no-edit 0b45ddc55..0600dd322 git diff 2.60.4
Diffstat (limited to 'devel/glib2/patches')
-rw-r--r--devel/glib2/patches/patch-gio_tests_gdbus-export.c70
-rw-r--r--devel/glib2/patches/patch-glib_ghash.c873
-rw-r--r--devel/glib2/patches/patch-glib_tests_hash.c60
3 files changed, 1003 insertions, 0 deletions
diff --git a/devel/glib2/patches/patch-gio_tests_gdbus-export.c b/devel/glib2/patches/patch-gio_tests_gdbus-export.c
new file mode 100644
index 00000000000..7d5d1099fd3
--- /dev/null
+++ b/devel/glib2/patches/patch-gio_tests_gdbus-export.c
@@ -0,0 +1,70 @@
+$NetBSD: patch-gio_tests_gdbus-export.c,v 1.1 2019/06/21 20:21:00 prlw1 Exp $
+
+Revert GHashTable improvements
+https://gitlab.gnome.org/GNOME/glib/merge_requests/208
+
+to fix PR pkg/54310
+
+--- gio/tests/gdbus-export.c.orig 2019-06-10 17:47:20.000000000 +0000
++++ gio/tests/gdbus-export.c
+@@ -337,22 +337,6 @@ introspect_callback (GDBusProxy *proxy
+ g_main_loop_quit (loop);
+ }
+
+-static gint
+-compare_strings (gconstpointer a,
+- gconstpointer b)
+-{
+- const gchar *sa = *(const gchar **) a;
+- const gchar *sb = *(const gchar **) b;
+-
+- /* Array terminator must sort last */
+- if (sa == NULL)
+- return 1;
+- if (sb == NULL)
+- return -1;
+-
+- return strcmp (sa, sb);
+-}
+-
+ static gchar **
+ get_nodes_at (GDBusConnection *c,
+ const gchar *object_path)
+@@ -406,9 +390,6 @@ get_nodes_at (GDBusConnection *c,
+ g_free (xml_data);
+ g_dbus_node_info_unref (node_info);
+
+- /* Nodes are semantically unordered; sort array so tests can rely on order */
+- g_ptr_array_sort (p, compare_strings);
+-
+ return (gchar **) g_ptr_array_free (p, FALSE);
+ }
+
+@@ -1259,9 +1240,9 @@ test_object_registration (void)
+ nodes = get_nodes_at (c, "/foo/dyna");
+ g_assert (nodes != NULL);
+ g_assert_cmpint (g_strv_length (nodes), ==, 3);
+- g_assert_cmpstr (nodes[0], ==, "cat");
+- g_assert_cmpstr (nodes[1], ==, "cheezburger");
+- g_assert_cmpstr (nodes[2], ==, "lol");
++ g_assert_cmpstr (nodes[0], ==, "lol");
++ g_assert_cmpstr (nodes[1], ==, "cat");
++ g_assert_cmpstr (nodes[2], ==, "cheezburger");
+ g_strfreev (nodes);
+ g_assert_cmpint (count_interfaces (c, "/foo/dyna/lol"), ==, 4);
+ g_assert_cmpint (count_interfaces (c, "/foo/dyna/cat"), ==, 4);
+@@ -1272,10 +1253,10 @@ test_object_registration (void)
+ nodes = get_nodes_at (c, "/foo/dyna");
+ g_assert (nodes != NULL);
+ g_assert_cmpint (g_strv_length (nodes), ==, 4);
+- g_assert_cmpstr (nodes[0], ==, "cat");
+- g_assert_cmpstr (nodes[1], ==, "cheezburger");
+- g_assert_cmpstr (nodes[2], ==, "dynamicallycreated");
+- g_assert_cmpstr (nodes[3], ==, "lol");
++ g_assert_cmpstr (nodes[0], ==, "lol");
++ g_assert_cmpstr (nodes[1], ==, "cat");
++ g_assert_cmpstr (nodes[2], ==, "cheezburger");
++ g_assert_cmpstr (nodes[3], ==, "dynamicallycreated");
+ g_strfreev (nodes);
+ g_assert_cmpint (count_interfaces (c, "/foo/dyna/dynamicallycreated"), ==, 4);
+
diff --git a/devel/glib2/patches/patch-glib_ghash.c b/devel/glib2/patches/patch-glib_ghash.c
new file mode 100644
index 00000000000..73a2c69ffed
--- /dev/null
+++ b/devel/glib2/patches/patch-glib_ghash.c
@@ -0,0 +1,873 @@
+$NetBSD: patch-glib_ghash.c,v 1.1 2019/06/21 20:21:00 prlw1 Exp $
+
+Revert GHashTable improvements
+https://gitlab.gnome.org/GNOME/glib/merge_requests/208
+
+to fix PR pkg/54310
+
+--- glib/ghash.c.orig 2019-06-10 17:47:20.000000000 +0000
++++ glib/ghash.c
+@@ -38,26 +38,6 @@
+ #include "gtestutils.h"
+ #include "gslice.h"
+ #include "grefcount.h"
+-#include "gvalgrind.h"
+-
+-/* The following #pragma is here so we can do this...
+- *
+- * #ifndef USE_SMALL_ARRAYS
+- * is_big = TRUE;
+- * #endif
+- * return is_big ? *(((gpointer *) a) + index) : GUINT_TO_POINTER (*(((guint *) a) + index));
+- *
+- * ...instead of this...
+- *
+- * #ifndef USE_SMALL_ARRAYS
+- * return *(((gpointer *) a) + index);
+- * #else
+- * return is_big ? *(((gpointer *) a) + index) : GUINT_TO_POINTER (*(((guint *) a) + index));
+- * #endif
+- *
+- * ...and still compile successfully when -Werror=duplicated-branches is passed. */
+-
+-#pragma GCC diagnostic ignored "-Wduplicated-branches"
+
+ /**
+ * SECTION:hash_tables
+@@ -233,18 +213,6 @@
+ #define HASH_IS_TOMBSTONE(h_) ((h_) == TOMBSTONE_HASH_VALUE)
+ #define HASH_IS_REAL(h_) ((h_) >= 2)
+
+-/* If int is smaller than void * on our arch, we start out with
+- * int-sized keys and values and resize to pointer-sized entries as
+- * needed. This saves a good amount of memory when the HT is being
+- * used with e.g. GUINT_TO_POINTER(). */
+-
+-#define BIG_ENTRY_SIZE (SIZEOF_VOID_P)
+-#define SMALL_ENTRY_SIZE (SIZEOF_INT)
+-
+-#if SMALL_ENTRY_SIZE < BIG_ENTRY_SIZE
+-# define USE_SMALL_ARRAYS
+-#endif
+-
+ struct _GHashTable
+ {
+ gint size;
+@@ -253,12 +221,9 @@ struct _GHashTable
+ gint nnodes;
+ gint noccupied; /* nnodes + tombstones */
+
+- guint have_big_keys : 1;
+- guint have_big_values : 1;
+-
+- gpointer keys;
++ gpointer *keys;
+ guint *hashes;
+- gpointer values;
++ gpointer *values;
+
+ GHashFunc hash_func;
+ GEqualFunc key_equal_func;
+@@ -332,15 +297,19 @@ static const gint prime_mod [] =
+ static void
+ g_hash_table_set_shift (GHashTable *hash_table, gint shift)
+ {
++ gint i;
++ guint mask = 0;
++
+ hash_table->size = 1 << shift;
+ hash_table->mod = prime_mod [shift];
+
+- /* hash_table->size is always a power of two, so we can calculate the mask
+- * by simply subtracting 1 from it. The leading assertion ensures that
+- * we're really dealing with a power of two. */
++ for (i = 0; i < shift; i++)
++ {
++ mask <<= 1;
++ mask |= 1;
++ }
+
+- g_assert ((hash_table->size & (hash_table->size - 1)) == 0);
+- hash_table->mask = hash_table->size - 1;
++ hash_table->mask = mask;
+ }
+
+ static gint
+@@ -365,67 +334,6 @@ g_hash_table_set_shift_from_size (GHashT
+ g_hash_table_set_shift (hash_table, shift);
+ }
+
+-static inline gpointer
+-g_hash_table_realloc_key_or_value_array (gpointer a, guint size, G_GNUC_UNUSED gboolean is_big)
+-{
+-#ifdef USE_SMALL_ARRAYS
+- return g_realloc (a, size * (is_big ? BIG_ENTRY_SIZE : SMALL_ENTRY_SIZE));
+-#else
+- return g_renew (gpointer, a, size);
+-#endif
+-}
+-
+-static inline gpointer
+-g_hash_table_fetch_key_or_value (gpointer a, guint index, gboolean is_big)
+-{
+-#ifndef USE_SMALL_ARRAYS
+- is_big = TRUE;
+-#endif
+- return is_big ? *(((gpointer *) a) + index) : GUINT_TO_POINTER (*(((guint *) a) + index));
+-}
+-
+-static inline void
+-g_hash_table_assign_key_or_value (gpointer a, guint index, gboolean is_big, gpointer v)
+-{
+-#ifndef USE_SMALL_ARRAYS
+- is_big = TRUE;
+-#endif
+- if (is_big)
+- *(((gpointer *) a) + index) = v;
+- else
+- *(((guint *) a) + index) = GPOINTER_TO_UINT (v);
+-}
+-
+-static inline gpointer
+-g_hash_table_evict_key_or_value (gpointer a, guint index, gboolean is_big, gpointer v)
+-{
+-#ifndef USE_SMALL_ARRAYS
+- is_big = TRUE;
+-#endif
+- if (is_big)
+- {
+- gpointer r = *(((gpointer *) a) + index);
+- *(((gpointer *) a) + index) = v;
+- return r;
+- }
+- else
+- {
+- gpointer r = GUINT_TO_POINTER (*(((guint *) a) + index));
+- *(((guint *) a) + index) = GPOINTER_TO_UINT (v);
+- return r;
+- }
+-}
+-
+-static inline guint
+-g_hash_table_hash_to_index (GHashTable *hash_table, guint hash)
+-{
+- /* Multiply the hash by a small prime before applying the modulo. This
+- * prevents the table from becoming densely packed, even with a poor hash
+- * function. A densely packed table would have poor performance on
+- * workloads with many failed lookups or a high degree of churn. */
+- return (hash * 11) % hash_table->mod;
+-}
+-
+ /*
+ * g_hash_table_lookup_node:
+ * @hash_table: our #GHashTable
+@@ -474,7 +382,7 @@ g_hash_table_lookup_node (GHashTable
+
+ *hash_return = hash_value;
+
+- node_index = g_hash_table_hash_to_index (hash_table, hash_value);
++ node_index = hash_value % hash_table->mod;
+ node_hash = hash_table->hashes[node_index];
+
+ while (!HASH_IS_UNUSED (node_hash))
+@@ -485,7 +393,7 @@ g_hash_table_lookup_node (GHashTable
+ */
+ if (node_hash == hash_value)
+ {
+- gpointer node_key = g_hash_table_fetch_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys);
++ gpointer node_key = hash_table->keys[node_index];
+
+ if (hash_table->key_equal_func)
+ {
+@@ -535,15 +443,15 @@ g_hash_table_remove_node (GHashTable *
+ gpointer key;
+ gpointer value;
+
+- key = g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys);
+- value = g_hash_table_fetch_key_or_value (hash_table->values, i, hash_table->have_big_values);
++ key = hash_table->keys[i];
++ value = hash_table->values[i];
+
+ /* Erect tombstone */
+ hash_table->hashes[i] = TOMBSTONE_HASH_VALUE;
+
+ /* Be GC friendly */
+- g_hash_table_assign_key_or_value (hash_table->keys, i, hash_table->have_big_keys, NULL);
+- g_hash_table_assign_key_or_value (hash_table->values, i, hash_table->have_big_values, NULL);
++ hash_table->keys[i] = NULL;
++ hash_table->values[i] = NULL;
+
+ hash_table->nnodes--;
+
+@@ -556,58 +464,15 @@ g_hash_table_remove_node (GHashTable *
+ }
+
+ /*
+- * g_hash_table_setup_storage:
+- * @hash_table: our #GHashTable
+- *
+- * Initialise the hash table size, mask, mod, and arrays.
+- */
+-static void
+-g_hash_table_setup_storage (GHashTable *hash_table)
+-{
+- gboolean small;
+-
+- /* We want to use small arrays only if:
+- * - we are running on a system where that makes sense (64 bit); and
+- * - we are not running under valgrind.
+- */
+- small = FALSE;
+-
+-#ifdef USE_SMALL_ARRAYS
+- small = TRUE;
+-
+-# ifdef ENABLE_VALGRIND
+- if (RUNNING_ON_VALGRIND)
+- small = FALSE;
+-# endif
+-#endif
+-
+- g_hash_table_set_shift (hash_table, HASH_TABLE_MIN_SHIFT);
+-
+- hash_table->have_big_keys = !small;
+- hash_table->have_big_values = !small;
+-
+- hash_table->keys = g_hash_table_realloc_key_or_value_array (NULL, hash_table->size, hash_table->have_big_keys);
+- hash_table->values = hash_table->keys;
+- hash_table->hashes = g_new0 (guint, hash_table->size);
+-}
+-
+-/*
+ * g_hash_table_remove_all_nodes:
+ * @hash_table: our #GHashTable
+ * @notify: %TRUE if the destroy notify handlers are to be called
+ *
+- * Removes all nodes from the table.
++ * Removes all nodes from the table. Since this may be a precursor to
++ * freeing the table entirely, no resize is performed.
+ *
+ * If @notify is %TRUE then the destroy notify functions are called
+ * for the key and value of the hash node.
+- *
+- * Since this may be a precursor to freeing the table entirely, we'd
+- * ideally perform no resize, and we can indeed avoid that in some
+- * cases. However: in the case that we'll be making callbacks to user
+- * code (via destroy notifies) we need to consider that the user code
+- * might call back into the table again. In this case, we setup a new
+- * set of arrays so that any callers will see an empty (but valid)
+- * table.
+ */
+ static void
+ g_hash_table_remove_all_nodes (GHashTable *hash_table,
+@@ -621,8 +486,6 @@ g_hash_table_remove_all_nodes (GHashTabl
+ gpointer *old_keys;
+ gpointer *old_values;
+ guint *old_hashes;
+- gboolean old_have_big_keys;
+- gboolean old_have_big_values;
+
+ /* If the hash table is already empty, there is nothing to be done. */
+ if (hash_table->nnodes == 0)
+@@ -631,7 +494,6 @@ g_hash_table_remove_all_nodes (GHashTabl
+ hash_table->nnodes = 0;
+ hash_table->noccupied = 0;
+
+- /* Easy case: no callbacks, so we just zero out the arrays */
+ if (!notify ||
+ (hash_table->key_destroy_func == NULL &&
+ hash_table->value_destroy_func == NULL))
+@@ -639,65 +501,49 @@ g_hash_table_remove_all_nodes (GHashTabl
+ if (!destruction)
+ {
+ memset (hash_table->hashes, 0, hash_table->size * sizeof (guint));
+-
+-#ifdef USE_SMALL_ARRAYS
+- memset (hash_table->keys, 0, hash_table->size * (hash_table->have_big_keys ? BIG_ENTRY_SIZE : SMALL_ENTRY_SIZE));
+- memset (hash_table->values, 0, hash_table->size * (hash_table->have_big_values ? BIG_ENTRY_SIZE : SMALL_ENTRY_SIZE));
+-#else
+ memset (hash_table->keys, 0, hash_table->size * sizeof (gpointer));
+ memset (hash_table->values, 0, hash_table->size * sizeof (gpointer));
+-#endif
+ }
+
+ return;
+ }
+
+- /* Hard case: we need to do user callbacks. There are two
+- * possibilities here:
+- *
+- * 1) there are no outstanding references on the table and therefore
+- * nobody should be calling into it again (destroying == true)
+- *
+- * 2) there are outstanding references, and there may be future
+- * calls into the table, either after we return, or from the destroy
+- * notifies that we're about to do (destroying == false)
+- *
+- * We handle both cases by taking the current state of the table into
+- * local variables and replacing it with something else: in the "no
+- * outstanding references" cases we replace it with a bunch of
+- * null/zero values so that any access to the table will fail. In the
+- * "may receive future calls" case, we reinitialise the struct to
+- * appear like a newly-created empty table.
+- *
+- * In both cases, we take over the references for the current state,
+- * freeing them below.
+- */
++ /* Keep the old storage space around to iterate over it. */
+ old_size = hash_table->size;
+- old_have_big_keys = hash_table->have_big_keys;
+- old_have_big_values = hash_table->have_big_values;
+- old_keys = g_steal_pointer (&hash_table->keys);
+- old_values = g_steal_pointer (&hash_table->values);
+- old_hashes = g_steal_pointer (&hash_table->hashes);
+-
++ old_keys = hash_table->keys;
++ old_values = hash_table->values;
++ old_hashes = hash_table->hashes;
++
++ /* Now create a new storage space; If the table is destroyed we can use the
++ * shortcut of not creating a new storage. This saves the allocation at the
++ * cost of not allowing any recursive access.
++ * However, the application doesn't own any reference anymore, so access
++ * is not allowed. If accesses are done, then either an assert or crash
++ * *will* happen. */
++ g_hash_table_set_shift (hash_table, HASH_TABLE_MIN_SHIFT);
+ if (!destruction)
+- /* Any accesses will see an empty table */
+- g_hash_table_setup_storage (hash_table);
++ {
++ hash_table->keys = g_new0 (gpointer, hash_table->size);
++ hash_table->values = hash_table->keys;
++ hash_table->hashes = g_new0 (guint, hash_table->size);
++ }
+ else
+- /* Will cause a quick crash on any attempted access */
+- hash_table->size = hash_table->mod = hash_table->mask = 0;
++ {
++ hash_table->keys = NULL;
++ hash_table->values = NULL;
++ hash_table->hashes = NULL;
++ }
+
+- /* Now do the actual destroy notifies */
+ for (i = 0; i < old_size; i++)
+ {
+ if (HASH_IS_REAL (old_hashes[i]))
+ {
+- key = g_hash_table_fetch_key_or_value (old_keys, i, old_have_big_keys);
+- value = g_hash_table_fetch_key_or_value (old_values, i, old_have_big_values);
++ key = old_keys[i];
++ value = old_values[i];
+
+ old_hashes[i] = UNUSED_HASH_VALUE;
+-
+- g_hash_table_assign_key_or_value (old_keys, i, old_have_big_keys, NULL);
+- g_hash_table_assign_key_or_value (old_values, i, old_have_big_values, NULL);
++ old_keys[i] = NULL;
++ old_values[i] = NULL;
+
+ if (hash_table->key_destroy_func != NULL)
+ hash_table->key_destroy_func (key);
+@@ -715,125 +561,6 @@ g_hash_table_remove_all_nodes (GHashTabl
+ g_free (old_hashes);
+ }
+
+-static void
+-realloc_arrays (GHashTable *hash_table, gboolean is_a_set)
+-{
+- hash_table->hashes = g_renew (guint, hash_table->hashes, hash_table->size);
+- hash_table->keys = g_hash_table_realloc_key_or_value_array (hash_table->keys, hash_table->size, hash_table->have_big_keys);
+-
+- if (is_a_set)
+- hash_table->values = hash_table->keys;
+- else
+- hash_table->values = g_hash_table_realloc_key_or_value_array (hash_table->values, hash_table->size, hash_table->have_big_values);
+-}
+-
+-/* When resizing the table in place, we use a temporary bit array to keep
+- * track of which entries have been assigned a proper location in the new
+- * table layout.
+- *
+- * Each bit corresponds to a bucket. A bit is set if an entry was assigned
+- * its corresponding location during the resize and thus should not be
+- * evicted. The array starts out cleared to zero. */
+-
+-static inline gboolean
+-get_status_bit (const guint32 *bitmap, guint index)
+-{
+- return (bitmap[index / 32] >> (index % 32)) & 1;
+-}
+-
+-static inline void
+-set_status_bit (guint32 *bitmap, guint index)
+-{
+- bitmap[index / 32] |= 1U << (index % 32);
+-}
+-
+-/* By calling dedicated resize functions for sets and maps, we avoid 2x
+- * test-and-branch per key in the inner loop. This yields a small
+- * performance improvement at the cost of a bit of macro gunk. */
+-
+-#define DEFINE_RESIZE_FUNC(fname) \
+-static void fname (GHashTable *hash_table, guint old_size, guint32 *reallocated_buckets_bitmap) \
+-{ \
+- guint i; \
+- \
+- for (i = 0; i < old_size; i++) \
+- { \
+- guint node_hash = hash_table->hashes[i]; \
+- gpointer key, value G_GNUC_UNUSED; \
+- \
+- if (!HASH_IS_REAL (node_hash)) \
+- { \
+- /* Clear tombstones */ \
+- hash_table->hashes[i] = UNUSED_HASH_VALUE; \
+- continue; \
+- } \
+- \
+- /* Skip entries relocated through eviction */ \
+- if (get_status_bit (reallocated_buckets_bitmap, i)) \
+- continue; \
+- \
+- hash_table->hashes[i] = UNUSED_HASH_VALUE; \
+- EVICT_KEYVAL (hash_table, i, NULL, NULL, key, value); \
+- \
+- for (;;) \
+- { \
+- guint hash_val; \
+- guint replaced_hash; \
+- guint step = 0; \
+- \
+- hash_val = g_hash_table_hash_to_index (hash_table, node_hash); \
+- \
+- while (get_status_bit (reallocated_buckets_bitmap, hash_val)) \
+- { \
+- step++; \
+- hash_val += step; \
+- hash_val &= hash_table->mask; \
+- } \
+- \
+- set_status_bit (reallocated_buckets_bitmap, hash_val); \
+- \
+- replaced_hash = hash_table->hashes[hash_val]; \
+- hash_table->hashes[hash_val] = node_hash; \
+- if (!HASH_IS_REAL (replaced_hash)) \
+- { \
+- ASSIGN_KEYVAL (hash_table, hash_val, key, value); \
+- break; \
+- } \
+- \
+- node_hash = replaced_hash; \
+- EVICT_KEYVAL (hash_table, hash_val, key, value, key, value); \
+- } \
+- } \
+-}
+-
+-#define ASSIGN_KEYVAL(ht, index, key, value) G_STMT_START{ \
+- g_hash_table_assign_key_or_value ((ht)->keys, (index), (ht)->have_big_keys, (key)); \
+- g_hash_table_assign_key_or_value ((ht)->values, (index), (ht)->have_big_values, (value)); \
+- }G_STMT_END
+-
+-#define EVICT_KEYVAL(ht, index, key, value, outkey, outvalue) G_STMT_START{ \
+- (outkey) = g_hash_table_evict_key_or_value ((ht)->keys, (index), (ht)->have_big_keys, (key)); \
+- (outvalue) = g_hash_table_evict_key_or_value ((ht)->values, (index), (ht)->have_big_values, (value)); \
+- }G_STMT_END
+-
+-DEFINE_RESIZE_FUNC (resize_map)
+-
+-#undef ASSIGN_KEYVAL
+-#undef EVICT_KEYVAL
+-
+-#define ASSIGN_KEYVAL(ht, index, key, value) G_STMT_START{ \
+- g_hash_table_assign_key_or_value ((ht)->keys, (index), (ht)->have_big_keys, (key)); \
+- }G_STMT_END
+-
+-#define EVICT_KEYVAL(ht, index, key, value, outkey, outvalue) G_STMT_START{ \
+- (outkey) = g_hash_table_evict_key_or_value ((ht)->keys, (index), (ht)->have_big_keys, (key)); \
+- }G_STMT_END
+-
+-DEFINE_RESIZE_FUNC (resize_set)
+-
+-#undef ASSIGN_KEYVAL
+-#undef EVICT_KEYVAL
+-
+ /*
+ * g_hash_table_resize:
+ * @hash_table: our #GHashTable
+@@ -850,47 +577,54 @@ DEFINE_RESIZE_FUNC (resize_set)
+ static void
+ g_hash_table_resize (GHashTable *hash_table)
+ {
+- guint32 *reallocated_buckets_bitmap;
+- guint old_size;
+- gboolean is_a_set;
++ gpointer *new_keys;
++ gpointer *new_values;
++ guint *new_hashes;
++ gint old_size;
++ gint i;
+
+ old_size = hash_table->size;
+- is_a_set = hash_table->keys == hash_table->values;
+-
+- /* The outer checks in g_hash_table_maybe_resize() will only consider
+- * cleanup/resize when the load factor goes below .25 (1/4, ignoring
+- * tombstones) or above .9375 (15/16, including tombstones).
+- *
+- * Once this happens, tombstones will always be cleaned out. If our
+- * load sans tombstones is greater than .75 (1/1.333, see below), we'll
+- * take this opportunity to grow the table too.
+- *
+- * Immediately after growing, the load factor will be in the range
+- * .375 .. .469. After shrinking, it will be exactly .5. */
++ g_hash_table_set_shift_from_size (hash_table, hash_table->nnodes * 2);
+
+- g_hash_table_set_shift_from_size (hash_table, hash_table->nnodes * 1.333);
++ new_keys = g_new0 (gpointer, hash_table->size);
++ if (hash_table->keys == hash_table->values)
++ new_values = new_keys;
++ else
++ new_values = g_new0 (gpointer, hash_table->size);
++ new_hashes = g_new0 (guint, hash_table->size);
+
+- if (hash_table->size > old_size)
++ for (i = 0; i < old_size; i++)
+ {
+- realloc_arrays (hash_table, is_a_set);
+- memset (&hash_table->hashes[old_size], 0, (hash_table->size - old_size) * sizeof (guint));
++ guint node_hash = hash_table->hashes[i];
++ guint hash_val;
++ guint step = 0;
+
+- reallocated_buckets_bitmap = g_new0 (guint32, (hash_table->size + 31) / 32);
+- }
+- else
+- {
+- reallocated_buckets_bitmap = g_new0 (guint32, (old_size + 31) / 32);
++ if (!HASH_IS_REAL (node_hash))
++ continue;
++
++ hash_val = node_hash % hash_table->mod;
++
++ while (!HASH_IS_UNUSED (new_hashes[hash_val]))
++ {
++ step++;
++ hash_val += step;
++ hash_val &= hash_table->mask;
++ }
++
++ new_hashes[hash_val] = hash_table->hashes[i];
++ new_keys[hash_val] = hash_table->keys[i];
++ new_values[hash_val] = hash_table->values[i];
+ }
+
+- if (is_a_set)
+- resize_set (hash_table, old_size, reallocated_buckets_bitmap);
+- else
+- resize_map (hash_table, old_size, reallocated_buckets_bitmap);
++ if (hash_table->keys != hash_table->values)
++ g_free (hash_table->values);
+
+- g_free (reallocated_buckets_bitmap);
++ g_free (hash_table->keys);
++ g_free (hash_table->hashes);
+
+- if (hash_table->size < old_size)
+- realloc_arrays (hash_table, is_a_set);
++ hash_table->keys = new_keys;
++ hash_table->values = new_values;
++ hash_table->hashes = new_hashes;
+
+ hash_table->noccupied = hash_table->nnodes;
+ }
+@@ -915,94 +649,6 @@ g_hash_table_maybe_resize (GHashTable *h
+ g_hash_table_resize (hash_table);
+ }
+
+-#ifdef USE_SMALL_ARRAYS
+-
+-static inline gboolean
+-entry_is_big (gpointer v)
+-{
+- return (((guintptr) v) >> ((BIG_ENTRY_SIZE - SMALL_ENTRY_SIZE) * 8)) != 0;
+-}
+-
+-static inline gboolean
+-g_hash_table_maybe_make_big_keys_or_values (gpointer *a_p, gpointer v, gint ht_size)
+-{
+- if (entry_is_big (v))
+- {
+- guint *a = (guint *) *a_p;
+- gpointer *a_new;
+- gint i;
+-
+- a_new = g_new (gpointer, ht_size);
+-
+- for (i = 0; i < ht_size; i++)
+- {
+- a_new[i] = GUINT_TO_POINTER (a[i]);
+- }
+-
+- g_free (a);
+- *a_p = a_new;
+- return TRUE;
+- }
+-
+- return FALSE;
+-}
+-
+-#endif
+-
+-static inline void
+-g_hash_table_ensure_keyval_fits (GHashTable *hash_table, gpointer key, gpointer value)
+-{
+- gboolean is_a_set = (hash_table->keys == hash_table->values);
+-
+-#ifdef USE_SMALL_ARRAYS
+-
+- /* Convert from set to map? */
+- if (is_a_set)
+- {
+- if (hash_table->have_big_keys)
+- {
+- if (key != value)
+- hash_table->values = g_memdup (hash_table->keys, sizeof (gpointer) * hash_table->size);
+- /* Keys and values are both big now, so no need for further checks */
+- return;
+- }
+- else
+- {
+- if (key != value)
+- {
+- hash_table->values = g_memdup (hash_table->keys, sizeof (guint) * hash_table->size);
+- is_a_set = FALSE;
+- }
+- }
+- }
+-
+- /* Make keys big? */
+- if (!hash_table->have_big_keys)
+- {
+- hash_table->have_big_keys = g_hash_table_maybe_make_big_keys_or_values (&hash_table->keys, key, hash_table->size);
+-
+- if (is_a_set)
+- {
+- hash_table->values = hash_table->keys;
+- hash_table->have_big_values = hash_table->have_big_keys;
+- }
+- }
+-
+- /* Make values big? */
+- if (!is_a_set && !hash_table->have_big_values)
+- {
+- hash_table->have_big_values = g_hash_table_maybe_make_big_keys_or_values (&hash_table->values, value, hash_table->size);
+- }
+-
+-#else
+-
+- /* Just split if necessary */
+- if (is_a_set && key != value)
+- hash_table->values = g_memdup (hash_table->keys, sizeof (gpointer) * hash_table->size);
+-
+-#endif
+-}
+-
+ /**
+ * g_hash_table_new:
+ * @hash_func: a function to create a hash value from a key
+@@ -1069,6 +715,7 @@ g_hash_table_new_full (GHashFunc ha
+ GHashTable *hash_table;
+
+ hash_table = g_slice_new (GHashTable);
++ g_hash_table_set_shift (hash_table, HASH_TABLE_MIN_SHIFT);
+ g_atomic_ref_count_init (&hash_table->ref_count);
+ hash_table->nnodes = 0;
+ hash_table->noccupied = 0;
+@@ -1079,8 +726,9 @@ g_hash_table_new_full (GHashFunc ha
+ #endif
+ hash_table->key_destroy_func = key_destroy_func;
+ hash_table->value_destroy_func = value_destroy_func;
+-
+- g_hash_table_setup_storage (hash_table);
++ hash_table->keys = g_new0 (gpointer, hash_table->size);
++ hash_table->values = hash_table->keys;
++ hash_table->hashes = g_new0 (guint, hash_table->size);
+
+ return hash_table;
+ }
+@@ -1164,9 +812,9 @@ g_hash_table_iter_next (GHashTableIter *
+ while (!HASH_IS_REAL (ri->hash_table->hashes[position]));
+
+ if (key != NULL)
+- *key = g_hash_table_fetch_key_or_value (ri->hash_table->keys, position, ri->hash_table->have_big_keys);
++ *key = ri->hash_table->keys[position];
+ if (value != NULL)
+- *value = g_hash_table_fetch_key_or_value (ri->hash_table->values, position, ri->hash_table->have_big_values);
++ *value = ri->hash_table->values[position];
+
+ ri->position = position;
+ return TRUE;
+@@ -1269,7 +917,6 @@ g_hash_table_insert_node (GHashTable *ha
+ gboolean already_exists;
+ guint old_hash;
+ gpointer key_to_free = NULL;
+- gpointer key_to_keep = NULL;
+ gpointer value_to_free = NULL;
+
+ old_hash = hash_table->hashes[node_index];
+@@ -1299,31 +946,31 @@ g_hash_table_insert_node (GHashTable *ha
+ * because we might change the value in the event that the two
+ * arrays are shared.
+ */
+- value_to_free = g_hash_table_fetch_key_or_value (hash_table->values, node_index, hash_table->have_big_values);
++ value_to_free = hash_table->values[node_index];
+
+ if (keep_new_key)
+ {
+- key_to_free = g_hash_table_fetch_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys);
+- key_to_keep = new_key;
++ key_to_free = hash_table->keys[node_index];
++ hash_table->keys[node_index] = new_key;
+ }
+ else
+- {
+- key_to_free = new_key;
+- key_to_keep = g_hash_table_fetch_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys);
+- }
++ key_to_free = new_key;
+ }
+ else
+ {
+ hash_table->hashes[node_index] = key_hash;
+- key_to_keep = new_key;
++ hash_table->keys[node_index] = new_key;
+ }
+
+- /* Resize key/value arrays and split table as necessary */
+- g_hash_table_ensure_keyval_fits (hash_table, key_to_keep, new_value);
+- g_hash_table_assign_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys, key_to_keep);
++ /* Step two: check if the value that we are about to write to the
++ * table is the same as the key in the same position. If it's not,
++ * split the table.
++ */
++ if (G_UNLIKELY (hash_table->keys == hash_table->values && hash_table->keys[node_index] != new_value))
++ hash_table->values = g_memdup (hash_table->keys, sizeof (gpointer) * hash_table->size);
+
+ /* Step 3: Actually do the write */
+- g_hash_table_assign_key_or_value (hash_table->values, node_index, hash_table->have_big_values, new_value);
++ hash_table->values[node_index] = new_value;
+
+ /* Now, the bookkeeping... */
+ if (!already_exists)
+@@ -1385,8 +1032,7 @@ g_hash_table_iter_replace (GHashTableIte
+ g_return_if_fail (ri->position < ri->hash_table->size);
+
+ node_hash = ri->hash_table->hashes[ri->position];
+-
+- key = g_hash_table_fetch_key_or_value (ri->hash_table->keys, ri->position, ri->hash_table->have_big_keys);
++ key = ri->hash_table->keys[ri->position];
+
+ g_hash_table_insert_node (ri->hash_table, ri->position, node_hash, key, value, TRUE, TRUE);
+
+@@ -1507,7 +1153,7 @@ g_hash_table_lookup (GHashTable *hash
+ node_index = g_hash_table_lookup_node (hash_table, key, &node_hash);
+
+ return HASH_IS_REAL (hash_table->hashes[node_index])
+- ? g_hash_table_fetch_key_or_value (hash_table->values, node_index, hash_table->have_big_values)
++ ? hash_table->values[node_index]
+ : NULL;
+ }
+
+@@ -1554,10 +1200,10 @@ g_hash_table_lookup_extended (GHashTable
+ }
+
+ if (orig_key)
+- *orig_key = g_hash_table_fetch_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys);
++ *orig_key = hash_table->keys[node_index];
+
+ if (value)
+- *value = g_hash_table_fetch_key_or_value (hash_table->values, node_index, hash_table->have_big_values);
++ *value = hash_table->values[node_index];
+
+ return TRUE;
+ }
+@@ -1828,16 +1474,10 @@ g_hash_table_steal_extended (GHashTable
+ }
+
+ if (stolen_key != NULL)
+- {
+- *stolen_key = g_hash_table_fetch_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys);
+- g_hash_table_assign_key_or_value (hash_table->keys, node_index, hash_table->have_big_keys, NULL);
+- }
++ *stolen_key = g_steal_pointer (&hash_table->keys[node_index]);
+
+ if (stolen_value != NULL)
+- {
+- *stolen_value = g_hash_table_fetch_key_or_value (hash_table->values, node_index, hash_table->have_big_values);
+- g_hash_table_assign_key_or_value (hash_table->values, node_index, hash_table->have_big_values, NULL);
+- }
++ *stolen_value = g_steal_pointer (&hash_table->values[node_index]);
+
+ g_hash_table_remove_node (hash_table, node_index, FALSE);
+ g_hash_table_maybe_resize (hash_table);
+@@ -1931,8 +1571,8 @@ g_hash_table_foreach_remove_or_steal (GH
+ for (i = 0; i < hash_table->size; i++)
+ {
+ guint node_hash = hash_table->hashes[i];
+- gpointer node_key = g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys);
+- gpointer node_value = g_hash_table_fetch_key_or_value (hash_table->values, i, hash_table->have_big_values);
++ gpointer node_key = hash_table->keys[i];
++ gpointer node_value = hash_table->values[i];
+
+ if (HASH_IS_REAL (node_hash) &&
+ (* func) (node_key, node_value, user_data))
+@@ -2047,8 +1687,8 @@ g_hash_table_foreach (GHashTable *hash_t
+ for (i = 0; i < hash_table->size; i++)
+ {
+ guint node_hash = hash_table->hashes[i];
+- gpointer node_key = g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys);
+- gpointer node_value = g_hash_table_fetch_key_or_value (hash_table->values, i, hash_table->have_big_values);
++ gpointer node_key = hash_table->keys[i];
++ gpointer node_value = hash_table->values[i];
+
+ if (HASH_IS_REAL (node_hash))
+ (* func) (node_key, node_value, user_data);
+@@ -2108,8 +1748,8 @@ g_hash_table_find (GHashTable *hash_tabl
+ for (i = 0; i < hash_table->size; i++)
+ {
+ guint node_hash = hash_table->hashes[i];
+- gpointer node_key = g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys);
+- gpointer node_value = g_hash_table_fetch_key_or_value (hash_table->values, i, hash_table->have_big_values);
++ gpointer node_key = hash_table->keys[i];
++ gpointer node_value = hash_table->values[i];
+
+ if (HASH_IS_REAL (node_hash))
+ match = predicate (node_key, node_value, user_data);
+@@ -2171,7 +1811,7 @@ g_hash_table_get_keys (GHashTable *hash_
+ for (i = 0; i < hash_table->size; i++)
+ {
+ if (HASH_IS_REAL (hash_table->hashes[i]))
+- retval = g_list_prepend (retval, g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys));
++ retval = g_list_prepend (retval, hash_table->keys[i]);
+ }
+
+ return retval;
+@@ -2216,7 +1856,7 @@ g_hash_table_get_keys_as_array (GHashTab
+ for (i = 0; i < hash_table->size; i++)
+ {
+ if (HASH_IS_REAL (hash_table->hashes[i]))
+- result[j++] = g_hash_table_fetch_key_or_value (hash_table->keys, i, hash_table->have_big_keys);
++ result[j++] = hash_table->keys[i];
+ }
+ g_assert_cmpint (j, ==, hash_table->nnodes);
+ result[j] = NULL;
+@@ -2257,7 +1897,7 @@ g_hash_table_get_values (GHashTable *has
+ for (i = 0; i < hash_table->size; i++)
+ {
+ if (HASH_IS_REAL (hash_table->hashes[i]))
+- retval = g_list_prepend (retval, g_hash_table_fetch_key_or_value (hash_table->values, i, hash_table->have_big_values));
++ retval = g_list_prepend (retval, hash_table->values[i]);
+ }
+
+ return retval;
diff --git a/devel/glib2/patches/patch-glib_tests_hash.c b/devel/glib2/patches/patch-glib_tests_hash.c
new file mode 100644
index 00000000000..fff6421e0ef
--- /dev/null
+++ b/devel/glib2/patches/patch-glib_tests_hash.c
@@ -0,0 +1,60 @@
+$NetBSD: patch-glib_tests_hash.c,v 1.1 2019/06/21 20:21:00 prlw1 Exp $
+
+Revert GHashTable improvements
+https://gitlab.gnome.org/GNOME/glib/merge_requests/208
+
+to fix PR pkg/54310
+
+--- glib/tests/hash.c.orig 2019-06-10 17:47:20.000000000 +0000
++++ glib/tests/hash.c
+@@ -1353,9 +1353,6 @@ struct _GHashTable
+ gint nnodes;
+ gint noccupied; /* nnodes + tombstones */
+
+- guint have_big_keys : 1;
+- guint have_big_values : 1;
+-
+ gpointer *keys;
+ guint *hashes;
+ gpointer *values;
+@@ -1390,23 +1387,6 @@ count_keys (GHashTable *h, gint *unused,
+ }
+ }
+
+-#define BIG_ENTRY_SIZE (SIZEOF_VOID_P)
+-#define SMALL_ENTRY_SIZE (SIZEOF_INT)
+-
+-#if SMALL_ENTRY_SIZE < BIG_ENTRY_SIZE
+-# define USE_SMALL_ARRAYS
+-#endif
+-
+-static gpointer
+-fetch_key_or_value (gpointer a, guint index, gboolean is_big)
+-{
+-#ifdef USE_SMALL_ARRAYS
+- return is_big ? *(((gpointer *) a) + index) : GUINT_TO_POINTER (*(((guint *) a) + index));
+-#else
+- return *(((gpointer *) a) + index);
+-#endif
+-}
+-
+ static void
+ check_data (GHashTable *h)
+ {
+@@ -1414,9 +1394,14 @@ check_data (GHashTable *h)
+
+ for (i = 0; i < h->size; i++)
+ {
+- if (h->hashes[i] >= 2)
++ if (h->hashes[i] < 2)
++ {
++ g_assert (h->keys[i] == NULL);
++ g_assert (h->values[i] == NULL);
++ }
++ else
+ {
+- g_assert_cmpint (h->hashes[i], ==, h->hash_func (fetch_key_or_value (h->keys, i, h->have_big_keys)));
++ g_assert_cmpint (h->hashes[i], ==, h->hash_func (h->keys[i]));
+ }
+ }
+ }