少年修仙传客户端基础资源
hch
2024-04-01 d01413b00ef631ac20347716b23818b0b811f65f
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
/**
 * \file
 * A pointer queue that can be sorted.
 *
 * Copyright (C) 2014 Xamarin Inc
 *
 * Licensed under the MIT license. See LICENSE file in the project root for full license information.
 */
 
#ifdef HAVE_SGEN_GC
 
#include <string.h>
 
#include "mono/sgen/sgen-gc.h"
#include "mono/sgen/sgen-pointer-queue.h"
 
void
sgen_pointer_queue_clear (SgenPointerQueue *queue)
{
    queue->next_slot = 0;
}
 
void
sgen_pointer_queue_init (SgenPointerQueue *queue, int mem_type)
{
    queue->next_slot = 0;
    queue->size = 0;
    queue->data = NULL;
    queue->mem_type = mem_type;
}
 
static void
realloc_queue (SgenPointerQueue *queue)
{
    size_t new_size = queue->size ? queue->size + queue->size/2 : 1024;
    void **new_data = (void **)sgen_alloc_internal_dynamic (sizeof (void*) * new_size, queue->mem_type, TRUE);
 
    memcpy (new_data, queue->data, sizeof (void*) * queue->next_slot);
    sgen_free_internal_dynamic (queue->data, sizeof (void*) * queue->size, queue->mem_type);
    queue->data = new_data;
    queue->size = new_size;
    SGEN_LOG (4, "Reallocated pointer queue to size: %lu", new_size);
}
 
gboolean
sgen_pointer_queue_will_grow (SgenPointerQueue *queue)
{
    return queue->next_slot >= queue->size;
}
 
void
sgen_pointer_queue_add (SgenPointerQueue *queue, void *ptr)
{
    if (sgen_pointer_queue_will_grow (queue))
        realloc_queue (queue);
 
    queue->data [queue->next_slot++] = ptr;
}
 
void*
sgen_pointer_queue_pop (SgenPointerQueue *queue)
{
    g_assert (queue->next_slot);
 
    return queue->data [--queue->next_slot];
}
 
size_t
sgen_pointer_queue_search (SgenPointerQueue *queue, void *addr)
{
    size_t first = 0, last = queue->next_slot;
    while (first < last) {
        size_t middle = first + ((last - first) >> 1);
        if (addr <= queue->data [middle])
            last = middle;
        else
            first = middle + 1;
    }
    g_assert (first == last);
    return first;
}
 
/*
 * Removes all NULL pointers from the queue.
 */
void
sgen_pointer_queue_remove_nulls (SgenPointerQueue *queue)
{
    void **start, **cur, **end;
    start = cur = queue->data;
    end = queue->data + queue->next_slot;
    while (cur < end) {
        if (*cur)
            *start++ = *cur++;
        else
            ++cur;
    }
    queue->next_slot = start - queue->data;
}
 
/*
 * Sorts the pointers in the queue, then removes duplicates.
 */
void
sgen_pointer_queue_sort_uniq (SgenPointerQueue *queue)
{
    void **start, **cur, **end;
    /* sort and uniq pin_queue: we just sort and we let the rest discard multiple values */
    /* it may be better to keep ranges of pinned memory instead of individually pinning objects */
    SGEN_LOG (5, "Sorting pointer queue, size: %lu", queue->next_slot);
    if (queue->next_slot > 1)
        sgen_sort_addresses (queue->data, queue->next_slot);
    start = cur = queue->data;
    end = queue->data + queue->next_slot;
    while (cur < end) {
        *start = *cur++;
        while (cur < end && *start == *cur)
            cur++;
        start++;
    };
    queue->next_slot = start - queue->data;
    SGEN_LOG (5, "Pointer queue reduced to size: %lu", queue->next_slot);
}
 
/*
 * Does a linear search through the pointer queue to find `ptr`.  Returns the index if
 * found, otherwise (size_t)-1.
 */
size_t
sgen_pointer_queue_find (SgenPointerQueue *queue, void *ptr)
{
    size_t i;
    for (i = 0; i < queue->next_slot; ++i)
        if (queue->data [i] == ptr)
            return i;
    return (size_t)-1;
}
 
gboolean
sgen_pointer_queue_is_empty (SgenPointerQueue *queue)
{
    return !queue->next_slot;
}
 
void
sgen_pointer_queue_free (SgenPointerQueue *queue)
{
    sgen_free_internal_dynamic (queue->data, sizeof (void*) * queue->size, queue->mem_type);
}
 
#endif