+/*
+ * Copyright 2001-2003 Ximian, Inc
+ * Copyright 2003-2010 Novell, Inc.
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining
+ * a copy of this software and associated documentation files (the
+ * "Software"), to deal in the Software without restriction, including
+ * without limitation the rights to use, copy, modify, merge, publish,
+ * distribute, sublicense, and/or sell copies of the Software, and to
+ * permit persons to whom the Software is furnished to do so, subject to
+ * the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be
+ * included in all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+ * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+ * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+ * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+ * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+ * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+ */
#define PIN_STAGING_AREA_SIZE 1024
static void* pin_staging_area [PIN_STAGING_AREA_SIZE];
static int pin_staging_area_index;
+static void** pin_queue;
+static int pin_queue_size = 0;
+static int next_pin_slot = 0;
+
static void
init_pinning (void)
{
pin_staging_area_index = 0;
}
+static void
+realloc_pin_queue (void)
+{
+ int new_size = pin_queue_size? pin_queue_size + pin_queue_size/2: 1024;
+ void **new_pin = mono_sgen_alloc_internal_dynamic (sizeof (void*) * new_size, INTERNAL_MEM_PIN_QUEUE);
+ memcpy (new_pin, pin_queue, sizeof (void*) * next_pin_slot);
+ mono_sgen_free_internal_dynamic (pin_queue, sizeof (void*) * pin_queue_size, INTERNAL_MEM_PIN_QUEUE);
+ pin_queue = new_pin;
+ pin_queue_size = new_size;
+ DEBUG (4, fprintf (gc_debug_file, "Reallocated pin queue to size: %d\n", new_size));
+}
+
static void
evacuate_pin_staging_area (void)
{
g_assert (pin_staging_area_index >= 0 && pin_staging_area_index <= PIN_STAGING_AREA_SIZE);
+ if (pin_staging_area_index == 0)
+ return;
+
/*
* The pinning addresses might come from undefined memory, this is normal. Since they
* are used in lots of functions, we make the memory defined here instead of having
pin_staging_area [pin_staging_area_index++] = ptr;
}
+
+static int
+optimized_pin_queue_search (void *addr)
+{
+ int first = 0, last = next_pin_slot;
+ while (first < last) {
+ int middle = first + ((last - first) >> 1);
+ if (addr <= pin_queue [middle])
+ last = middle;
+ else
+ first = middle + 1;
+ }
+ g_assert (first == last);
+ return first;
+}
+
+void**
+mono_sgen_find_optimized_pin_queue_area (void *start, void *end, int *num)
+{
+ int first, last;
+ first = optimized_pin_queue_search (start);
+ last = optimized_pin_queue_search (end);
+ *num = last - first;
+ if (first == last)
+ return NULL;
+ return pin_queue + first;
+}
+
+void
+mono_sgen_find_section_pin_queue_start_end (GCMemSection *section)
+{
+ DEBUG (6, fprintf (gc_debug_file, "Pinning from section %p (%p-%p)\n", section, section->data, section->end_data));
+ section->pin_queue_start = mono_sgen_find_optimized_pin_queue_area (section->data, section->end_data, §ion->pin_queue_num_entries);
+ DEBUG (6, fprintf (gc_debug_file, "Found %d pinning addresses in section %p\n", section->pin_queue_num_entries, section));
+}
+
+static void
+mono_sgen_pin_queue_clear_discarded_entries (GCMemSection *section, int max_pin_slot)
+{
+ void **start = section->pin_queue_start + section->pin_queue_num_entries;
+ void **end = pin_queue + max_pin_slot;
+ void *addr;
+
+ for (; start < end; ++start) {
+ addr = *start;
+ if ((char*)addr < section->data || (char*)addr > section->end_data)
+ break;
+ *start = NULL;
+ }
+}