[set] Use relaxed atomic ops for last_page_index

Since iterating a set from multiple threads is supported.
This commit is contained in:
Behdad Esfahbod 2022-05-27 07:06:02 -06:00
parent d61b207491
commit e2aa29907d
1 changed files with 16 additions and 16 deletions

View File

@ -56,7 +56,7 @@ struct hb_bit_set_t
{ {
successful = true; successful = true;
population = 0; population = 0;
last_page_lookup = 0; last_page_lookup.set_relaxed (0);
page_map.init (); page_map.init ();
pages.init (); pages.init ();
} }
@ -78,7 +78,7 @@ struct hb_bit_set_t
bool successful = true; /* Allocations successful */ bool successful = true; /* Allocations successful */
mutable unsigned int population = 0; mutable unsigned int population = 0;
mutable unsigned int last_page_lookup = 0; mutable hb_atomic_int_t last_page_lookup = 0;
hb_sorted_vector_t<page_map_t> page_map; hb_sorted_vector_t<page_map_t> page_map;
hb_vector_t<page_t> pages; hb_vector_t<page_t> pages;
@ -607,7 +607,7 @@ struct hb_bit_set_t
const auto* page_map_array = page_map.arrayZ; const auto* page_map_array = page_map.arrayZ;
unsigned int major = get_major (*codepoint); unsigned int major = get_major (*codepoint);
unsigned int i = last_page_lookup; unsigned int i = last_page_lookup.get_relaxed ();
if (unlikely (i >= page_map.length || page_map_array[i].major != major)) if (unlikely (i >= page_map.length || page_map_array[i].major != major))
{ {
@ -625,7 +625,7 @@ struct hb_bit_set_t
if (pages_array[current.index].next (codepoint)) if (pages_array[current.index].next (codepoint))
{ {
*codepoint += current.major * page_t::PAGE_BITS; *codepoint += current.major * page_t::PAGE_BITS;
last_page_lookup = i; last_page_lookup.set_relaxed (i);
return true; return true;
} }
i++; i++;
@ -638,11 +638,11 @@ struct hb_bit_set_t
if (m != INVALID) if (m != INVALID)
{ {
*codepoint = current.major * page_t::PAGE_BITS + m; *codepoint = current.major * page_t::PAGE_BITS + m;
last_page_lookup = i; last_page_lookup.set_relaxed (i);
return true; return true;
} }
} }
last_page_lookup = 0; last_page_lookup.set_relaxed (0);
*codepoint = INVALID; *codepoint = INVALID;
return false; return false;
} }
@ -725,7 +725,7 @@ struct hb_bit_set_t
{ {
const auto* page_map_array = page_map.arrayZ; const auto* page_map_array = page_map.arrayZ;
unsigned int major = get_major (codepoint); unsigned int major = get_major (codepoint);
unsigned int i = last_page_lookup; unsigned int i = last_page_lookup.get_relaxed ();
if (unlikely (i >= page_map.length || page_map_array[i].major != major)) if (unlikely (i >= page_map.length || page_map_array[i].major != major))
{ {
page_map.bfind (major, &i, HB_NOT_FOUND_STORE_CLOSEST); page_map.bfind (major, &i, HB_NOT_FOUND_STORE_CLOSEST);
@ -766,7 +766,7 @@ struct hb_bit_set_t
{ {
const auto* page_map_array = page_map.arrayZ; const auto* page_map_array = page_map.arrayZ;
unsigned int major = get_major (codepoint); unsigned int major = get_major (codepoint);
unsigned int i = last_page_lookup; unsigned int i = last_page_lookup.get_relaxed ();
if (unlikely (i >= page_map.length || page_map_array[i].major != major)) if (unlikely (i >= page_map.length || page_map_array[i].major != major))
{ {
page_map.bfind(major, &i, HB_NOT_FOUND_STORE_CLOSEST); page_map.bfind(major, &i, HB_NOT_FOUND_STORE_CLOSEST);
@ -893,15 +893,15 @@ struct hb_bit_set_t
/* The extra page_map length is necessary; can't just rely on vector here, /* The extra page_map length is necessary; can't just rely on vector here,
* since the next check would be tricked because a null page also has * since the next check would be tricked because a null page also has
* major==0, which we can't distinguish from an actualy major==0 page... */ * major==0, which we can't distinguish from an actualy major==0 page... */
if (likely (last_page_lookup < page_map.length)) unsigned i = last_page_lookup.get_relaxed ();
if (likely (i < page_map.length))
{ {
auto &cached_page = page_map.arrayZ[last_page_lookup]; auto &cached_page = page_map.arrayZ[i];
if (cached_page.major == major) if (cached_page.major == major)
return &pages[cached_page.index]; return &pages[cached_page.index];
} }
page_map_t map = {major, pages.length}; page_map_t map = {major, pages.length};
unsigned int i;
if (!page_map.bfind (map, &i, HB_NOT_FOUND_STORE_CLOSEST)) if (!page_map.bfind (map, &i, HB_NOT_FOUND_STORE_CLOSEST))
{ {
if (!insert) if (!insert)
@ -917,7 +917,7 @@ struct hb_bit_set_t
page_map[i] = map; page_map[i] = map;
} }
last_page_lookup = i; last_page_lookup.set_relaxed (i);
return &pages[page_map[i].index]; return &pages[page_map[i].index];
} }
const page_t *page_for (hb_codepoint_t g) const const page_t *page_for (hb_codepoint_t g) const
@ -927,19 +927,19 @@ struct hb_bit_set_t
/* The extra page_map length is necessary; can't just rely on vector here, /* The extra page_map length is necessary; can't just rely on vector here,
* since the next check would be tricked because a null page also has * since the next check would be tricked because a null page also has
* major==0, which we can't distinguish from an actualy major==0 page... */ * major==0, which we can't distinguish from an actualy major==0 page... */
if (likely (last_page_lookup < page_map.length)) unsigned i = last_page_lookup.get_relaxed ();
if (likely (i < page_map.length))
{ {
auto &cached_page = page_map.arrayZ[last_page_lookup]; auto &cached_page = page_map.arrayZ[i];
if (cached_page.major == major) if (cached_page.major == major)
return &pages[cached_page.index]; return &pages[cached_page.index];
} }
page_map_t key = {major}; page_map_t key = {major};
unsigned int i;
if (!page_map.bfind (key, &i)) if (!page_map.bfind (key, &i))
return nullptr; return nullptr;
last_page_lookup = i; last_page_lookup.set_relaxed (i);
return &pages[page_map[i].index]; return &pages[page_map[i].index];
} }
page_t &page_at (unsigned int i) { return pages[page_map[i].index]; } page_t &page_at (unsigned int i) { return pages[page_map[i].index]; }