From 134d2ad890e001c5fd5c87c1e9448169bc766765 Mon Sep 17 00:00:00 2001 From: Nick Wellnhofer Date: Fri, 6 Oct 2023 00:31:44 +0200 Subject: [PATCH] parser: Protect against quadratic default attribute expansion --- parser.c | 17 +++++- result/errors/quadratic-defattr.xml.ent | 3 + result/errors/quadratic-defattr.xml.err | 3 + result/errors/quadratic-defattr.xml.str | 4 ++ test/errors/quadratic-defattr.xml | 73 +++++++++++++++++++++++++ 5 files changed, 98 insertions(+), 2 deletions(-) create mode 100644 result/errors/quadratic-defattr.xml.ent create mode 100644 result/errors/quadratic-defattr.xml.err create mode 100644 result/errors/quadratic-defattr.xml.str create mode 100644 test/errors/quadratic-defattr.xml diff --git a/parser.c b/parser.c index ba196f23..3b3eae58 100644 --- a/parser.c +++ b/parser.c @@ -885,6 +885,7 @@ typedef struct { xmlHashedString value; const xmlChar *valueEnd; int external; + int expandedSize; } xmlDefAttr; typedef struct _xmlDefAttrs xmlDefAttrs; @@ -1014,7 +1015,7 @@ xmlAddDefAttrs(xmlParserCtxtPtr ctxt, const xmlChar *value) { xmlDefAttrsPtr defaults; xmlDefAttr *attr; - int len; + int len, expandedSize; xmlHashedString name; xmlHashedString prefix; xmlHashedString hvalue; @@ -1094,17 +1095,23 @@ xmlAddDefAttrs(xmlParserCtxtPtr ctxt, goto mem_error; /* intern the string and precompute the end */ - len = xmlStrlen(value); + len = strlen((const char *) value); hvalue = xmlDictLookupHashed(ctxt->dict, value, len); if (hvalue.name == NULL) goto mem_error; + expandedSize = strlen((const char *) name.name); + if (prefix.name != NULL) + expandedSize += strlen((const char *) prefix.name); + expandedSize += len; + attr = &defaults->attrs[defaults->nbAttrs++]; attr->name = name; attr->prefix = prefix; attr->value = hvalue; attr->valueEnd = hvalue.name + len; attr->external = ctxt->external; + attr->expandedSize = expandedSize; return; @@ -9801,9 +9808,13 @@ next_attr: aprefix = attr->prefix.name; if ((attname == ctxt->str_xmlns) && (aprefix == NULL)) { + xmlParserEntityCheck(ctxt, attr->expandedSize); + if (xmlParserNsPush(ctxt, NULL, &attr->value, NULL, 1) > 0) nbNs++; } else if (aprefix == ctxt->str_xmlns) { + xmlParserEntityCheck(ctxt, attr->expandedSize); + if (xmlParserNsPush(ctxt, &attr->name, &attr->value, NULL, 1) > 0) nbNs++; @@ -9951,6 +9962,8 @@ next_attr: attname, nsuri, NULL); } + xmlParserEntityCheck(ctxt, attr->expandedSize); + if ((atts == NULL) || (nbatts + 5 > maxatts)) { if (xmlCtxtGrowAttrs(ctxt, nbatts + 5) < 0) { localname = NULL; diff --git a/result/errors/quadratic-defattr.xml.ent b/result/errors/quadratic-defattr.xml.ent new file mode 100644 index 00000000..eddbe4f5 --- /dev/null +++ b/result/errors/quadratic-defattr.xml.ent @@ -0,0 +1,3 @@ +./test/errors/quadratic-defattr.xml:65: parser error : Maximum entity amplification factor exceeded, see xmlCtxtSetMaxAmplification. + + ^ diff --git a/result/errors/quadratic-defattr.xml.err b/result/errors/quadratic-defattr.xml.err new file mode 100644 index 00000000..eddbe4f5 --- /dev/null +++ b/result/errors/quadratic-defattr.xml.err @@ -0,0 +1,3 @@ +./test/errors/quadratic-defattr.xml:65: parser error : Maximum entity amplification factor exceeded, see xmlCtxtSetMaxAmplification. + + ^ diff --git a/result/errors/quadratic-defattr.xml.str b/result/errors/quadratic-defattr.xml.str new file mode 100644 index 00000000..a6b4d80b --- /dev/null +++ b/result/errors/quadratic-defattr.xml.str @@ -0,0 +1,4 @@ +./test/errors/quadratic-defattr.xml:65: parser error : Maximum entity amplification factor exceeded, see xmlCtxtSetMaxAmplification. + + ^ +./test/errors/quadratic-defattr.xml : failed to parse diff --git a/test/errors/quadratic-defattr.xml b/test/errors/quadratic-defattr.xml new file mode 100644 index 00000000..dd5189f6 --- /dev/null +++ b/test/errors/quadratic-defattr.xml @@ -0,0 +1,73 @@ + + + +]> + + + + + + + + + + + + + + + + +