[libxslt] Fix certain patterns with predicates
- From: Nick Wellnhofer <nwellnhof src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [libxslt] Fix certain patterns with predicates
- Date: Sun, 4 Aug 2013 23:32:50 +0000 (UTC)
commit cd40951e8e8a00e3ea022c2956d352bc131a54ba
Author: Nick Wellnhofer <wellnhofer aevum de>
Date: Sun Aug 4 22:59:26 2013 +0200
Fix certain patterns with predicates
The optimization for predicates in patterns only supports XSLT_OP_ELEM
and XSLT_OP_ALL. This commit makes predicates on other ops fall back to
the slow direct matching code path.
Fixes bugs #531685 and #538580.
libxslt/pattern.c | 18 ++++++++++++------
tests/docs/Makefile.am | 2 ++
tests/docs/bug-181.xml | 4 ++++
tests/docs/bug-182.xml | 4 ++++
tests/general/Makefile.am | 2 ++
tests/general/bug-181.out | 5 +++++
tests/general/bug-181.xsl | 13 +++++++++++++
tests/general/bug-182.out | 2 ++
tests/general/bug-182.xsl | 19 +++++++++++++++++++
9 files changed, 63 insertions(+), 6 deletions(-)
---
diff --git a/libxslt/pattern.c b/libxslt/pattern.c
index 058917a..a97726c 100644
--- a/libxslt/pattern.c
+++ b/libxslt/pattern.c
@@ -451,11 +451,14 @@ xsltReverseCompMatch(xsltParserContextPtr ctxt, xsltCompMatchPtr comp) {
xsltCompMatchAdd(ctxt, comp, XSLT_OP_END, NULL, NULL, 0);
/*
- * detect consecutive XSLT_OP_PREDICATE indicating a direct
- * matching should be done.
+ * Detect consecutive XSLT_OP_PREDICATE and predicates on ops which
+ * haven't been optimized yet indicating a direct matching should be done.
*/
for (i = 0;i < comp->nbStep - 1;i++) {
- if ((comp->steps[i].op == XSLT_OP_PREDICATE) &&
+ xsltOp op = comp->steps[i].op;
+
+ if ((op != XSLT_OP_ELEM) &&
+ (op != XSLT_OP_ALL) &&
(comp->steps[i + 1].op == XSLT_OP_PREDICATE)) {
comp->direct = 1;
@@ -655,8 +658,10 @@ xsltTestPredicateMatch(xsltTransformContextPtr ctxt, xsltCompMatchPtr comp,
isRVT = 0;
/*
- * Depending on the last selection, one may need to
- * recompute contextSize and proximityPosition.
+ * Recompute contextSize and proximityPosition.
+ *
+ * TODO: Make this work for additional ops. Currently, only XSLT_OP_ELEM
+ * and XSLT_OP_ALL are supported.
*/
oldCS = ctxt->xpathCtxt->contextSize;
oldCP = ctxt->xpathCtxt->proximityPosition;
@@ -1128,7 +1133,8 @@ restart:
break;
case XSLT_OP_PREDICATE: {
/*
- * when there is cascading XSLT_OP_PREDICATE, then use a
+ * When there is cascading XSLT_OP_PREDICATE or a predicate
+ * after an op which hasn't been optimized yet, then use a
* direct computation approach. It's not done directly
* at the beginning of the routine to filter out as much
* as possible this costly computation.
diff --git a/tests/docs/Makefile.am b/tests/docs/Makefile.am
index c5dad4c..e3a4aef 100644
--- a/tests/docs/Makefile.am
+++ b/tests/docs/Makefile.am
@@ -179,6 +179,8 @@ EXTRA_DIST = \
bug-178.xml \
bug-179.xml \
bug-180.xml \
+ bug-181.xml \
+ bug-182.xml \
character.xml \
array.xml \
items.xml
diff --git a/tests/docs/bug-181.xml b/tests/docs/bug-181.xml
new file mode 100644
index 0000000..285b69a
--- /dev/null
+++ b/tests/docs/bug-181.xml
@@ -0,0 +1,4 @@
+<Urmel>
+ <E>1. zwei <F>drei</F> zwei eins</E>
+ <E a="b">2. zwei <F>drei</F> zwei eins</E>
+</Urmel>
diff --git a/tests/docs/bug-182.xml b/tests/docs/bug-182.xml
new file mode 100644
index 0000000..02b378d
--- /dev/null
+++ b/tests/docs/bug-182.xml
@@ -0,0 +1,4 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<root>
+<body><b> b 1 </b> text 1 <b> b 2 </b> text 2 </body>
+</root>
diff --git a/tests/general/Makefile.am b/tests/general/Makefile.am
index 0c2ef30..4802a6b 100644
--- a/tests/general/Makefile.am
+++ b/tests/general/Makefile.am
@@ -188,6 +188,8 @@ EXTRA_DIST = \
bug-178.out bug-178.xsl \
bug-179.out bug-179.xsl \
bug-180.out bug-180.xsl bug-180.err \
+ bug-181.out bug-181.xsl \
+ bug-182.out bug-182.xsl \
character.out character.xsl \
character2.out character2.xsl \
itemschoose.out itemschoose.xsl \
diff --git a/tests/general/bug-181.out b/tests/general/bug-181.out
new file mode 100644
index 0000000..730387a
--- /dev/null
+++ b/tests/general/bug-181.out
@@ -0,0 +1,5 @@
+<?xml version="1.0"?>
+<Urmel>
+ <E>1. * zwei <F>drei</F> zwei eins</E>
+ <E a="b">2. * zwei <F>drei</F> zwei eins</E>
+</Urmel>
diff --git a/tests/general/bug-181.xsl b/tests/general/bug-181.xsl
new file mode 100644
index 0000000..0c4de71
--- /dev/null
+++ b/tests/general/bug-181.xsl
@@ -0,0 +1,13 @@
+<xsl:transform version="1.0"
+ xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
+ <xsl:template match="E/text()[ 1 ]">
+ <xsl:value-of select="substring-before( . , ' ')"/>
+ <xsl:text> * </xsl:text>
+ <xsl:value-of select="substring-after( . , ' ')"/>
+ </xsl:template>
+ <xsl:template match="@*|node()">
+ <xsl:copy>
+ <xsl:apply-templates select="@*|node()"/>
+ </xsl:copy>
+ </xsl:template>
+</xsl:transform>
diff --git a/tests/general/bug-182.out b/tests/general/bug-182.out
new file mode 100644
index 0000000..93ab15e
--- /dev/null
+++ b/tests/general/bug-182.out
@@ -0,0 +1,2 @@
+<?xml version="1.0"?>
+<body><p>b[2]: b 2 </p><p>text()[2]: text 2 </p></body>
diff --git a/tests/general/bug-182.xsl b/tests/general/bug-182.xsl
new file mode 100644
index 0000000..ecce187
--- /dev/null
+++ b/tests/general/bug-182.xsl
@@ -0,0 +1,19 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
+
+<xsl:template match="node()"/>
+
+<xsl:template match="text()[2]">
+ <p>text()[2]: <xsl:value-of select="."/></p>
+</xsl:template>
+<xsl:template match="b[2]">
+ <p>b[2]: <xsl:value-of select="."/></p>
+</xsl:template>
+
+<xsl:template match="/">
+ <body>
+ <xsl:apply-templates select="/root/body/node()"/>
+ </body>
+</xsl:template>
+
+</xsl:stylesheet>
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]