mirror of
https://gitlab.gnome.org/GNOME/libxml2
synced 2025-03-28 21:33:13 +00:00
parser: Fix performance regression when parsing namespaces
The namespace hash table didn't reuse deleted buckets, leading to quadratic behavior. Also ignore deleted buckets when resizing. Fixes #726.
This commit is contained in:
parent
897c73fe24
commit
122b61309f
13
parser.c
13
parser.c
@ -1525,7 +1525,7 @@ xmlParserNsStartElement(xmlParserNsData *nsdb) {
|
||||
static int
|
||||
xmlParserNsLookup(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix,
|
||||
xmlParserNsBucket **bucketPtr) {
|
||||
xmlParserNsBucket *bucket;
|
||||
xmlParserNsBucket *bucket, *tombstone;
|
||||
unsigned index, hashValue;
|
||||
|
||||
if (prefix->name == NULL)
|
||||
@ -1537,10 +1537,13 @@ xmlParserNsLookup(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix,
|
||||
hashValue = prefix->hashValue;
|
||||
index = hashValue & (ctxt->nsdb->hashSize - 1);
|
||||
bucket = &ctxt->nsdb->hash[index];
|
||||
tombstone = NULL;
|
||||
|
||||
while (bucket->hashValue) {
|
||||
if ((bucket->hashValue == hashValue) &&
|
||||
(bucket->index != INT_MAX)) {
|
||||
if (bucket->index == INT_MAX) {
|
||||
if (tombstone == NULL)
|
||||
tombstone = bucket;
|
||||
} else if (bucket->hashValue == hashValue) {
|
||||
if (ctxt->nsTab[bucket->index * 2] == prefix->name) {
|
||||
if (bucketPtr != NULL)
|
||||
*bucketPtr = bucket;
|
||||
@ -1557,7 +1560,7 @@ xmlParserNsLookup(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix,
|
||||
}
|
||||
|
||||
if (bucketPtr != NULL)
|
||||
*bucketPtr = bucket;
|
||||
*bucketPtr = tombstone ? tombstone : bucket;
|
||||
return(INT_MAX);
|
||||
}
|
||||
|
||||
@ -1798,7 +1801,7 @@ xmlParserNsPush(xmlParserCtxtPtr ctxt, const xmlHashedString *prefix,
|
||||
unsigned hv = ctxt->nsdb->hash[i].hashValue;
|
||||
unsigned newIndex;
|
||||
|
||||
if (hv == 0)
|
||||
if ((hv == 0) || (ctxt->nsdb->hash[i].index == INT_MAX))
|
||||
continue;
|
||||
newIndex = hv & (newSize - 1);
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user