Commit ee01d848 authored by Heikki Linnakangas's avatar Heikki Linnakangas

In bms_add_member(), use repalloc() if the bms needs to be enlarged.

Previously bms_add_member() would palloc a whole-new copy of the existing
set, copy the words, and pfree the old one. repalloc() is potentially much
faster, and more importantly, this is less surprising if CurrentMemoryContext
is not the same as the context the old set is in. bms_add_member() still
allocates a new bitmapset in CurrentMemoryContext if NULL is passed as
argument, but that is a lot less likely to induce bugs.

Nicholas White.
parent 357f7521
...@@ -632,21 +632,20 @@ bms_add_member(Bitmapset *a, int x) ...@@ -632,21 +632,20 @@ bms_add_member(Bitmapset *a, int x)
return bms_make_singleton(x); return bms_make_singleton(x);
wordnum = WORDNUM(x); wordnum = WORDNUM(x);
bitnum = BITNUM(x); bitnum = BITNUM(x);
/* enlarge the set if necessary */
if (wordnum >= a->nwords) if (wordnum >= a->nwords)
{ {
/* Slow path: make a larger set and union the input set into it */ int oldnwords = a->nwords;
Bitmapset *result;
int nwords;
int i; int i;
result = bms_make_singleton(x); a = (Bitmapset *) repalloc(a, BITMAPSET_SIZE(wordnum + 1));
nwords = a->nwords; a->nwords = wordnum + 1;
for (i = 0; i < nwords; i++) /* zero out the enlarged portion */
result->words[i] |= a->words[i]; for (i = oldnwords; i < a->nwords; i++)
pfree(a); a->words[i] = 0;
return result;
} }
/* Fast path: x fits in existing set */
a->words[wordnum] |= ((bitmapword) 1 << bitnum); a->words[wordnum] |= ((bitmapword) 1 << bitnum);
return a; return a;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment