OLD | NEW |
1 /* | 1 /* |
2 * xpath.c: XML Path Language implementation | 2 * xpath.c: XML Path Language implementation |
3 * XPath is a language for addressing parts of an XML document, | 3 * XPath is a language for addressing parts of an XML document, |
4 * designed to be used by both XSLT and XPointer | 4 * designed to be used by both XSLT and XPointer |
5 *f | 5 *f |
6 * Reference: W3C Recommendation 16 November 1999 | 6 * Reference: W3C Recommendation 16 November 1999 |
7 * http://www.w3.org/TR/1999/REC-xpath-19991116 | 7 * http://www.w3.org/TR/1999/REC-xpath-19991116 |
8 * Public reference: | 8 * Public reference: |
9 * http://www.w3.org/TR/xpath | 9 * http://www.w3.org/TR/xpath |
10 * | 10 * |
(...skipping 14714 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
14725 xmlDictReference(comp->dict); | 14725 xmlDictReference(comp->dict); |
14726 return(comp); | 14726 return(comp); |
14727 } | 14727 } |
14728 xmlFreePattern(stream); | 14728 xmlFreePattern(stream); |
14729 } | 14729 } |
14730 return(NULL); | 14730 return(NULL); |
14731 } | 14731 } |
14732 #endif /* XPATH_STREAMING */ | 14732 #endif /* XPATH_STREAMING */ |
14733 | 14733 |
14734 static void | 14734 static void |
14735 xmlXPathOptimizeExpressionInternal(xmlXPathCompExprPtr comp, xmlXPathStepOpPtr o
p) | 14735 xmlXPathOptimizeExpression(xmlXPathCompExprPtr comp, xmlXPathStepOpPtr op) |
14736 { | 14736 { |
14737 /* Already optimized? */ | |
14738 if (op->cacheURI != 0) | |
14739 return; | |
14740 | |
14741 /* | 14737 /* |
14742 * Try to rewrite "descendant-or-self::node()/foo" to an optimized | 14738 * Try to rewrite "descendant-or-self::node()/foo" to an optimized |
14743 * internal representation. | 14739 * internal representation. |
14744 */ | 14740 */ |
14745 | 14741 |
14746 if ((op->op == XPATH_OP_COLLECT /* 11 */) && | 14742 if ((op->op == XPATH_OP_COLLECT /* 11 */) && |
14747 (op->ch1 != -1) && | 14743 (op->ch1 != -1) && |
14748 (op->ch2 == -1 /* no predicate */)) | 14744 (op->ch2 == -1 /* no predicate */)) |
14749 { | 14745 { |
14750 xmlXPathStepOpPtr prevop = &comp->steps[op->ch1]; | 14746 xmlXPathStepOpPtr prevop = &comp->steps[op->ch1]; |
(...skipping 30 matching lines...) Expand all Loading... |
14781 */ | 14777 */ |
14782 op->ch1 = prevop->ch1; | 14778 op->ch1 = prevop->ch1; |
14783 op->value = AXIS_DESCENDANT_OR_SELF; | 14779 op->value = AXIS_DESCENDANT_OR_SELF; |
14784 break; | 14780 break; |
14785 default: | 14781 default: |
14786 break; | 14782 break; |
14787 } | 14783 } |
14788 } | 14784 } |
14789 } | 14785 } |
14790 | 14786 |
14791 /* Mark the node. */ | |
14792 op->cacheURI = (void*)(~0); | |
14793 | |
14794 /* Recurse */ | 14787 /* Recurse */ |
14795 if (op->ch1 != -1) | 14788 if (op->ch1 != -1) |
14796 xmlXPathOptimizeExpressionInternal(comp, &comp->steps[op->ch1]); | 14789 xmlXPathOptimizeExpression(comp, &comp->steps[op->ch1]); |
14797 if (op->ch2 != -1) | 14790 if (op->ch2 != -1) |
14798 » xmlXPathOptimizeExpressionInternal(comp, &comp->steps[op->ch2]); | 14791 » xmlXPathOptimizeExpression(comp, &comp->steps[op->ch2]); |
14799 } | |
14800 | |
14801 static void | |
14802 xmlXPathOptimizeExpression(xmlXPathCompExprPtr comp, int root) | |
14803 { | |
14804 int i; | |
14805 // The expression tree/graph traversal is linear, visiting | |
14806 // each node at most once. Mark each xmlXPathStepOp node | |
14807 // upon visiting, taking care of clearing out the marks | |
14808 // afterwards | |
14809 xmlXPathOptimizeExpressionInternal(comp, &comp->steps[root]); | |
14810 for (i = 0; i <= root; ++i) | |
14811 comp->steps[i].cacheURI = 0; | |
14812 } | 14792 } |
14813 | 14793 |
14814 /** | 14794 /** |
14815 * xmlXPathCtxtCompile: | 14795 * xmlXPathCtxtCompile: |
14816 * @ctxt: an XPath context | 14796 * @ctxt: an XPath context |
14817 * @str: the XPath expression | 14797 * @str: the XPath expression |
14818 * | 14798 * |
14819 * Compile an XPath expression | 14799 * Compile an XPath expression |
14820 * | 14800 * |
14821 * Returns the xmlXPathCompExprPtr resulting from the compilation or NULL. | 14801 * Returns the xmlXPathCompExprPtr resulting from the compilation or NULL. |
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
14860 } | 14840 } |
14861 xmlXPathFreeParserContext(pctxt); | 14841 xmlXPathFreeParserContext(pctxt); |
14862 | 14842 |
14863 if (comp != NULL) { | 14843 if (comp != NULL) { |
14864 comp->expr = xmlStrdup(str); | 14844 comp->expr = xmlStrdup(str); |
14865 #ifdef DEBUG_EVAL_COUNTS | 14845 #ifdef DEBUG_EVAL_COUNTS |
14866 comp->string = xmlStrdup(str); | 14846 comp->string = xmlStrdup(str); |
14867 comp->nb = 0; | 14847 comp->nb = 0; |
14868 #endif | 14848 #endif |
14869 if ((comp->nbStep > 1) && (comp->last >= 0)) { | 14849 if ((comp->nbStep > 1) && (comp->last >= 0)) { |
14870 » xmlXPathOptimizeExpression(comp, comp->last); | 14850 » xmlXPathOptimizeExpression(comp, &comp->steps[comp->last]); |
14871 } | 14851 } |
14872 } | 14852 } |
14873 return(comp); | 14853 return(comp); |
14874 } | 14854 } |
14875 | 14855 |
14876 /** | 14856 /** |
14877 * xmlXPathCompile: | 14857 * xmlXPathCompile: |
14878 * @str: the XPath expression | 14858 * @str: the XPath expression |
14879 * | 14859 * |
14880 * Compile an XPath expression | 14860 * Compile an XPath expression |
(...skipping 161 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
15042 while (*ctxt->cur != 0) ctxt->cur++; | 15022 while (*ctxt->cur != 0) ctxt->cur++; |
15043 } else | 15023 } else |
15044 #endif | 15024 #endif |
15045 { | 15025 { |
15046 xmlXPathCompileExpr(ctxt, 1); | 15026 xmlXPathCompileExpr(ctxt, 1); |
15047 if ((ctxt->error == XPATH_EXPRESSION_OK) && | 15027 if ((ctxt->error == XPATH_EXPRESSION_OK) && |
15048 (ctxt->comp != NULL) && | 15028 (ctxt->comp != NULL) && |
15049 (ctxt->comp->nbStep > 1) && | 15029 (ctxt->comp->nbStep > 1) && |
15050 (ctxt->comp->last >= 0)) | 15030 (ctxt->comp->last >= 0)) |
15051 { | 15031 { |
15052 » xmlXPathOptimizeExpression(ctxt->comp, ctxt->comp->last); | 15032 » xmlXPathOptimizeExpression(ctxt->comp, |
| 15033 » » &ctxt->comp->steps[ctxt->comp->last]); |
15053 } | 15034 } |
15054 } | 15035 } |
15055 CHECK_ERROR; | 15036 CHECK_ERROR; |
15056 xmlXPathRunEval(ctxt, 0); | 15037 xmlXPathRunEval(ctxt, 0); |
15057 } | 15038 } |
15058 | 15039 |
15059 /** | 15040 /** |
15060 * xmlXPathEval: | 15041 * xmlXPathEval: |
15061 * @str: the XPath expression | 15042 * @str: the XPath expression |
15062 * @ctx: the XPath context | 15043 * @ctx: the XPath context |
(...skipping 324 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
15387 xmlXPathTranslateFunction); | 15368 xmlXPathTranslateFunction); |
15388 | 15369 |
15389 xmlXPathRegisterFuncNS(ctxt, (const xmlChar *)"escape-uri", | 15370 xmlXPathRegisterFuncNS(ctxt, (const xmlChar *)"escape-uri", |
15390 (const xmlChar *)"http://www.w3.org/2002/08/xquery-functions", | 15371 (const xmlChar *)"http://www.w3.org/2002/08/xquery-functions", |
15391 xmlXPathEscapeUriFunction); | 15372 xmlXPathEscapeUriFunction); |
15392 } | 15373 } |
15393 | 15374 |
15394 #endif /* LIBXML_XPATH_ENABLED */ | 15375 #endif /* LIBXML_XPATH_ENABLED */ |
15395 #define bottom_xpath | 15376 #define bottom_xpath |
15396 #include "elfgcchack.h" | 15377 #include "elfgcchack.h" |
OLD | NEW |