8edb8caaf8
Fix CVE-2024-45492 integer overflow Fix CVE-2024-45491 Integer Overflow or Wraparound Fix CVE-2024-45490 Negative Length Parsing Vulnerability Resolves: RHEL-57505 Resolves: RHEL-57493 Resolves: RHEL-56751
130 lines
4.4 KiB
Diff
130 lines
4.4 KiB
Diff
commit 3c1a64705b5662c5b78f4aa5a5acc7a59c477094
|
|
Author: Tomas Korbar <tkorbar@redhat.com>
|
|
Date: Wed Sep 11 15:03:05 2024 +0200
|
|
|
|
Fix CVE-2024-45490
|
|
|
|
https://github.com/libexpat/libexpat/pull/890
|
|
|
|
diff --git a/expat/doc/reference.html b/expat/doc/reference.html
|
|
index 95c33c7..08cf9b0 100644
|
|
--- a/expat/doc/reference.html
|
|
+++ b/expat/doc/reference.html
|
|
@@ -1039,7 +1039,9 @@ containing part (or perhaps all) of the document. The number of bytes of s
|
|
that are part of the document is indicated by <code>len</code>. This means
|
|
that <code>s</code> doesn't have to be null terminated. It also means that
|
|
if <code>len</code> is larger than the number of bytes in the block of
|
|
-memory that <code>s</code> points at, then a memory fault is likely. The
|
|
+memory that <code>s</code> points at, then a memory fault is likely.
|
|
+Negative values for <code>len</code> are rejected since Expat 2.2.1.
|
|
+The
|
|
<code>isFinal</code> parameter informs the parser that this is the last
|
|
piece of the document. Frequently, the last piece is empty (i.e.
|
|
<code>len</code> is zero.)
|
|
@@ -1054,11 +1056,17 @@ XML_ParseBuffer(XML_Parser p,
|
|
int isFinal);
|
|
</pre>
|
|
<div class="fcndef">
|
|
+<p>
|
|
This is just like <code><a href= "#XML_Parse" >XML_Parse</a></code>,
|
|
except in this case Expat provides the buffer. By obtaining the
|
|
buffer from Expat with the <code><a href= "#XML_GetBuffer"
|
|
>XML_GetBuffer</a></code> function, the application can avoid double
|
|
copying of the input.
|
|
+</p>
|
|
+
|
|
+<p>
|
|
+Negative values for <code>len</code> are rejected since Expat 2.6.3.
|
|
+</p>
|
|
</div>
|
|
|
|
<pre class="fcndec" id="XML_GetBuffer">
|
|
diff --git a/expat/lib/xmlparse.c b/expat/lib/xmlparse.c
|
|
index 488f63f..c3c1af9 100644
|
|
--- a/expat/lib/xmlparse.c
|
|
+++ b/expat/lib/xmlparse.c
|
|
@@ -1981,6 +1981,12 @@ XML_ParseBuffer(XML_Parser parser, int len, int isFinal)
|
|
|
|
if (parser == NULL)
|
|
return XML_STATUS_ERROR;
|
|
+
|
|
+ if (len < 0) {
|
|
+ parser->m_errorCode = XML_ERROR_INVALID_ARGUMENT;
|
|
+ return XML_STATUS_ERROR;
|
|
+ }
|
|
+
|
|
switch (parser->m_parsingStatus.parsing) {
|
|
case XML_SUSPENDED:
|
|
parser->m_errorCode = XML_ERROR_SUSPENDED;
|
|
diff --git a/expat/tests/runtests.c b/expat/tests/runtests.c
|
|
index 486073f..6a3e09a 100644
|
|
--- a/expat/tests/runtests.c
|
|
+++ b/expat/tests/runtests.c
|
|
@@ -4083,6 +4083,57 @@ START_TEST(test_empty_parse)
|
|
}
|
|
END_TEST
|
|
|
|
+/* Test XML_Parse for len < 0 */
|
|
+START_TEST(test_negative_len_parse) {
|
|
+ const char *const doc = "<root/>";
|
|
+ for (int isFinal = 0; isFinal < 2; isFinal++) {
|
|
+ XML_Parser parser = XML_ParserCreate(NULL);
|
|
+
|
|
+ if (XML_GetErrorCode(parser) != XML_ERROR_NONE)
|
|
+ fail("There was not supposed to be any initial parse error.");
|
|
+
|
|
+ const enum XML_Status status = XML_Parse(parser, doc, -1, isFinal);
|
|
+
|
|
+ if (status != XML_STATUS_ERROR)
|
|
+ fail("Negative len was expected to fail the parse but did not.");
|
|
+
|
|
+ if (XML_GetErrorCode(parser) != XML_ERROR_INVALID_ARGUMENT)
|
|
+ fail("Parse error does not match XML_ERROR_INVALID_ARGUMENT.");
|
|
+
|
|
+ XML_ParserFree(parser);
|
|
+ }
|
|
+}
|
|
+END_TEST
|
|
+
|
|
+/* Test XML_ParseBuffer for len < 0 */
|
|
+START_TEST(test_negative_len_parse_buffer) {
|
|
+ const char *const doc = "<root/>";
|
|
+ for (int isFinal = 0; isFinal < 2; isFinal++) {
|
|
+ XML_Parser parser = XML_ParserCreate(NULL);
|
|
+
|
|
+ if (XML_GetErrorCode(parser) != XML_ERROR_NONE)
|
|
+ fail("There was not supposed to be any initial parse error.");
|
|
+
|
|
+ void *const buffer = XML_GetBuffer(parser, (int)strlen(doc));
|
|
+
|
|
+ if (buffer == NULL)
|
|
+ fail("XML_GetBuffer failed.");
|
|
+
|
|
+ memcpy(buffer, doc, strlen(doc));
|
|
+
|
|
+ const enum XML_Status status = XML_ParseBuffer(parser, -1, isFinal);
|
|
+
|
|
+ if (status != XML_STATUS_ERROR)
|
|
+ fail("Negative len was expected to fail the parse but did not.");
|
|
+
|
|
+ if (XML_GetErrorCode(parser) != XML_ERROR_INVALID_ARGUMENT)
|
|
+ fail("Parse error does not match XML_ERROR_INVALID_ARGUMENT.");
|
|
+
|
|
+ XML_ParserFree(parser);
|
|
+ }
|
|
+}
|
|
+END_TEST
|
|
+
|
|
/* Test odd corners of the XML_GetBuffer interface */
|
|
static enum XML_Status
|
|
get_feature(enum XML_FeatureEnum feature_id, long *presult)
|
|
@@ -13094,6 +13145,8 @@ make_suite(void)
|
|
tcase_add_test(tc_basic, test_user_parameters);
|
|
tcase_add_test(tc_basic, test_ext_entity_ref_parameter);
|
|
tcase_add_test(tc_basic, test_empty_parse);
|
|
+ tcase_add_test(tc_basic, test_negative_len_parse);
|
|
+ tcase_add_test(tc_basic, test_negative_len_parse_buffer);
|
|
tcase_add_test(tc_basic, test_get_buffer_1);
|
|
tcase_add_test(tc_basic, test_get_buffer_2);
|
|
#if defined(XML_CONTEXT_BYTES)
|