diff --git a/ext/xml/tests/gh15868.phpt b/ext/xml/tests/gh15868.phpt
new file mode 100644
index 00000000000..17ed80558d7
--- /dev/null
+++ b/ext/xml/tests/gh15868.phpt
@@ -0,0 +1,46 @@
+--TEST--
+GH-15868 (Assertion failure in xml_parse_into_struct after exception)
+--EXTENSIONS--
+xml
+--FILE--
+", $values, $tags);
+} catch (Error $e) {
+ echo $e->getMessage(), "\n";
+}
+
+$parser = xml_parser_create();
+xml_set_element_handler($parser,
+ function ($parser, $name, $attrs) {
+ }, function ($parser, $name) {
+ throw new Error('stop 2');
+ }
+);
+try {
+ xml_parse_into_struct($parser, "", $values, $tags);
+} catch (Error $e) {
+ echo $e->getMessage(), "\n";
+}
+
+$parser = xml_parser_create();
+xml_set_character_data_handler($parser, function() {
+ throw new Error('stop 3');
+});
+try {
+ xml_parse_into_struct($parser, "", $values, $tags);
+} catch (Error $e) {
+ echo $e->getMessage(), "\n";
+}
+?>
+--EXPECT--
+stop 1
+stop 2
+stop 3
diff --git a/ext/xml/xml.c b/ext/xml/xml.c
index d1965b7bb53..7eca5c0795e 100644
--- a/ext/xml/xml.c
+++ b/ext/xml/xml.c
@@ -626,7 +626,7 @@ void xml_startElementHandler(void *userData, const XML_Char *name, const XML_Cha
zval_ptr_dtor(&args[2]);
}
- if (!Z_ISUNDEF(parser->data)) {
+ if (!Z_ISUNDEF(parser->data) && !EG(exception)) {
if (parser->level <= XML_MAXLEVEL) {
zval tag, atr;
int atcnt = 0;
@@ -697,7 +697,7 @@ void xml_endElementHandler(void *userData, const XML_Char *name)
zval_ptr_dtor(&args[1]);
}
- if (!Z_ISUNDEF(parser->data)) {
+ if (!Z_ISUNDEF(parser->data) && !EG(exception)) {
zval tag;
if (parser->lastwasopen) {
@@ -746,7 +746,7 @@ void xml_characterDataHandler(void *userData, const XML_Char *s, int len)
zval_ptr_dtor(&args[1]);
}
- if (Z_ISUNDEF(parser->data)) {
+ if (Z_ISUNDEF(parser->data) || EG(exception)) {
return;
}