Home | History | Annotate | Download | only in libxslt
      1 /*
      2  * xslt.c: Implemetation of an XSL Transformation 1.0 engine
      3  *
      4  * Reference:
      5  *   XSLT specification
      6  *   http://www.w3.org/TR/1999/REC-xslt-19991116
      7  *
      8  *   Associating Style Sheets with XML documents
      9  *   http://www.w3.org/1999/06/REC-xml-stylesheet-19990629
     10  *
     11  * See Copyright for the status of this software.
     12  *
     13  * daniel (at) veillard.com
     14  */
     15 
     16 #define IN_LIBXSLT
     17 #include "libxslt.h"
     18 
     19 #include <string.h>
     20 
     21 #include <libxml/xmlmemory.h>
     22 #include <libxml/parser.h>
     23 #include <libxml/tree.h>
     24 #include <libxml/valid.h>
     25 #include <libxml/hash.h>
     26 #include <libxml/uri.h>
     27 #include <libxml/xmlerror.h>
     28 #include <libxml/parserInternals.h>
     29 #include <libxml/xpathInternals.h>
     30 #include <libxml/xpath.h>
     31 #include "xslt.h"
     32 #include "xsltInternals.h"
     33 #include "pattern.h"
     34 #include "variables.h"
     35 #include "namespaces.h"
     36 #include "attributes.h"
     37 #include "xsltutils.h"
     38 #include "imports.h"
     39 #include "keys.h"
     40 #include "documents.h"
     41 #include "extensions.h"
     42 #include "preproc.h"
     43 #include "extra.h"
     44 #include "security.h"
     45 
     46 #ifdef WITH_XSLT_DEBUG
     47 #define WITH_XSLT_DEBUG_PARSING
     48 /* #define WITH_XSLT_DEBUG_BLANKS */
     49 #endif
     50 
     51 const char *xsltEngineVersion = LIBXSLT_VERSION_STRING LIBXSLT_VERSION_EXTRA;
     52 const int xsltLibxsltVersion = LIBXSLT_VERSION;
     53 const int xsltLibxmlVersion = LIBXML_VERSION;
     54 
     55 #ifdef XSLT_REFACTORED
     56 
     57 const xmlChar *xsltConstNamespaceNameXSLT = (const xmlChar *) XSLT_NAMESPACE;
     58 
     59 #define XSLT_ELEMENT_CATEGORY_XSLT 0
     60 #define XSLT_ELEMENT_CATEGORY_EXTENSION 1
     61 #define XSLT_ELEMENT_CATEGORY_LRE 2
     62 
     63 /*
     64 * xsltLiteralResultMarker:
     65 * Marker for Literal result elements, in order to avoid multiple attempts
     66 * to recognize such elements in the stylesheet's tree.
     67 * This marker is set on node->psvi during the initial traversal
     68 * of a stylesheet's node tree.
     69 *
     70 const xmlChar *xsltLiteralResultMarker =
     71     (const xmlChar *) "Literal Result Element";
     72 */
     73 
     74 /*
     75 * xsltXSLTTextMarker:
     76 * Marker for xsl:text elements. Used to recognize xsl:text elements
     77 * for post-processing of the stylesheet's tree, where those
     78 * elements are removed from the tree.
     79 */
     80 const xmlChar *xsltXSLTTextMarker = (const xmlChar *) "XSLT Text Element";
     81 
     82 /*
     83 * xsltXSLTAttrMarker:
     84 * Marker for XSLT attribute on Literal Result Elements.
     85 */
     86 const xmlChar *xsltXSLTAttrMarker = (const xmlChar *) "LRE XSLT Attr";
     87 
     88 #endif
     89 
     90 #ifdef XSLT_LOCALE_WINAPI
     91 extern xmlRMutexPtr xsltLocaleMutex;
     92 #endif
     93 /*
     94  * Harmless but avoiding a problem when compiling against a
     95  * libxml <= 2.3.11 without LIBXML_DEBUG_ENABLED
     96  */
     97 #ifndef LIBXML_DEBUG_ENABLED
     98 double xmlXPathStringEvalNumber(const xmlChar *str);
     99 #endif
    100 /*
    101  * Useful macros
    102  */
    103 
    104 #ifdef  IS_BLANK
    105 #undef	IS_BLANK
    106 #endif
    107 #define IS_BLANK(c) (((c) == 0x20) || ((c) == 0x09) || ((c) == 0xA) ||	\
    108                      ((c) == 0x0D))
    109 
    110 #ifdef	IS_BLANK_NODE
    111 #undef	IS_BLANK_NODE
    112 #endif
    113 #define IS_BLANK_NODE(n)						\
    114     (((n)->type == XML_TEXT_NODE) && (xsltIsBlank((n)->content)))
    115 
    116 /**
    117  * xsltParseContentError:
    118  *
    119  * @style: the stylesheet
    120  * @node: the node where the error occured
    121  *
    122  * Compile-time error function.
    123  */
    124 static void
    125 xsltParseContentError(xsltStylesheetPtr style,
    126 		       xmlNodePtr node)
    127 {
    128     if ((style == NULL) || (node == NULL))
    129 	return;
    130 
    131     if (IS_XSLT_ELEM(node))
    132 	xsltTransformError(NULL, style, node,
    133 	    "The XSLT-element '%s' is not allowed at this position.\n",
    134 	    node->name);
    135     else
    136 	xsltTransformError(NULL, style, node,
    137 	    "The element '%s' is not allowed at this position.\n",
    138 	    node->name);
    139     style->errors++;
    140 }
    141 
    142 #ifdef XSLT_REFACTORED
    143 #else
    144 /**
    145  * exclPrefixPush:
    146  * @style: the transformation stylesheet
    147  * @value:  the excluded namespace name to push on the stack
    148  *
    149  * Push an excluded namespace name on the stack
    150  *
    151  * Returns the new index in the stack or -1 if already present or
    152  * in case of error
    153  */
    154 static int
    155 exclPrefixPush(xsltStylesheetPtr style, xmlChar * value)
    156 {
    157     int i;
    158 
    159     if (style->exclPrefixMax == 0) {
    160         style->exclPrefixMax = 4;
    161         style->exclPrefixTab =
    162             (xmlChar * *)xmlMalloc(style->exclPrefixMax *
    163                                    sizeof(style->exclPrefixTab[0]));
    164         if (style->exclPrefixTab == NULL) {
    165             xmlGenericError(xmlGenericErrorContext, "malloc failed !\n");
    166             return (-1);
    167         }
    168     }
    169     /* do not push duplicates */
    170     for (i = 0;i < style->exclPrefixNr;i++) {
    171         if (xmlStrEqual(style->exclPrefixTab[i], value))
    172 	    return(-1);
    173     }
    174     if (style->exclPrefixNr >= style->exclPrefixMax) {
    175         style->exclPrefixMax *= 2;
    176         style->exclPrefixTab =
    177             (xmlChar * *)xmlRealloc(style->exclPrefixTab,
    178                                     style->exclPrefixMax *
    179                                     sizeof(style->exclPrefixTab[0]));
    180         if (style->exclPrefixTab == NULL) {
    181             xmlGenericError(xmlGenericErrorContext, "realloc failed !\n");
    182             return (-1);
    183         }
    184     }
    185     style->exclPrefixTab[style->exclPrefixNr] = value;
    186     style->exclPrefix = value;
    187     return (style->exclPrefixNr++);
    188 }
    189 /**
    190  * exclPrefixPop:
    191  * @style: the transformation stylesheet
    192  *
    193  * Pop an excluded prefix value from the stack
    194  *
    195  * Returns the stored excluded prefix value
    196  */
    197 static xmlChar *
    198 exclPrefixPop(xsltStylesheetPtr style)
    199 {
    200     xmlChar *ret;
    201 
    202     if (style->exclPrefixNr <= 0)
    203         return (0);
    204     style->exclPrefixNr--;
    205     if (style->exclPrefixNr > 0)
    206         style->exclPrefix = style->exclPrefixTab[style->exclPrefixNr - 1];
    207     else
    208         style->exclPrefix = NULL;
    209     ret = style->exclPrefixTab[style->exclPrefixNr];
    210     style->exclPrefixTab[style->exclPrefixNr] = 0;
    211     return (ret);
    212 }
    213 #endif
    214 
    215 /************************************************************************
    216  *									*
    217  *			Helper functions				*
    218  *									*
    219  ************************************************************************/
    220 
    221 static int initialized = 0;
    222 /**
    223  * xsltInit:
    224  *
    225  * Initializes the processor (e.g. registers built-in extensions,
    226  * etc.)
    227  */
    228 void
    229 xsltInit (void) {
    230     if (initialized == 0) {
    231 	initialized = 1;
    232 #ifdef XSLT_LOCALE_WINAPI
    233 	xsltLocaleMutex = xmlNewRMutex();
    234 #endif
    235         xsltRegisterAllExtras();
    236     }
    237 }
    238 
    239 /**
    240  * xsltUninit:
    241  *
    242  * Uninitializes the processor.
    243  */
    244 void
    245 xsltUninit (void) {
    246     initialized = 0;
    247 }
    248 
    249 /**
    250  * xsltIsBlank:
    251  * @str:  a string
    252  *
    253  * Check if a string is ignorable
    254  *
    255  * Returns 1 if the string is NULL or made of blanks chars, 0 otherwise
    256  */
    257 int
    258 xsltIsBlank(xmlChar *str) {
    259     if (str == NULL)
    260 	return(1);
    261     while (*str != 0) {
    262 	if (!(IS_BLANK(*str))) return(0);
    263 	str++;
    264     }
    265     return(1);
    266 }
    267 
    268 /************************************************************************
    269  *									*
    270  *		Routines to handle XSLT data structures			*
    271  *									*
    272  ************************************************************************/
    273 static xsltDecimalFormatPtr
    274 xsltNewDecimalFormat(xmlChar *name)
    275 {
    276     xsltDecimalFormatPtr self;
    277     /* UTF-8 for 0x2030 */
    278     static const xmlChar permille[4] = {0xe2, 0x80, 0xb0, 0};
    279 
    280     self = xmlMalloc(sizeof(xsltDecimalFormat));
    281     if (self != NULL) {
    282 	self->next = NULL;
    283 	self->name = name;
    284 
    285 	/* Default values */
    286 	self->digit = xmlStrdup(BAD_CAST("#"));
    287 	self->patternSeparator = xmlStrdup(BAD_CAST(";"));
    288 	self->decimalPoint = xmlStrdup(BAD_CAST("."));
    289 	self->grouping = xmlStrdup(BAD_CAST(","));
    290 	self->percent = xmlStrdup(BAD_CAST("%"));
    291 	self->permille = xmlStrdup(BAD_CAST(permille));
    292 	self->zeroDigit = xmlStrdup(BAD_CAST("0"));
    293 	self->minusSign = xmlStrdup(BAD_CAST("-"));
    294 	self->infinity = xmlStrdup(BAD_CAST("Infinity"));
    295 	self->noNumber = xmlStrdup(BAD_CAST("NaN"));
    296     }
    297     return self;
    298 }
    299 
    300 static void
    301 xsltFreeDecimalFormat(xsltDecimalFormatPtr self)
    302 {
    303     if (self != NULL) {
    304 	if (self->digit)
    305 	    xmlFree(self->digit);
    306 	if (self->patternSeparator)
    307 	    xmlFree(self->patternSeparator);
    308 	if (self->decimalPoint)
    309 	    xmlFree(self->decimalPoint);
    310 	if (self->grouping)
    311 	    xmlFree(self->grouping);
    312 	if (self->percent)
    313 	    xmlFree(self->percent);
    314 	if (self->permille)
    315 	    xmlFree(self->permille);
    316 	if (self->zeroDigit)
    317 	    xmlFree(self->zeroDigit);
    318 	if (self->minusSign)
    319 	    xmlFree(self->minusSign);
    320 	if (self->infinity)
    321 	    xmlFree(self->infinity);
    322 	if (self->noNumber)
    323 	    xmlFree(self->noNumber);
    324 	if (self->name)
    325 	    xmlFree(self->name);
    326 	xmlFree(self);
    327     }
    328 }
    329 
    330 static void
    331 xsltFreeDecimalFormatList(xsltStylesheetPtr self)
    332 {
    333     xsltDecimalFormatPtr iter;
    334     xsltDecimalFormatPtr tmp;
    335 
    336     if (self == NULL)
    337 	return;
    338 
    339     iter = self->decimalFormat;
    340     while (iter != NULL) {
    341 	tmp = iter->next;
    342 	xsltFreeDecimalFormat(iter);
    343 	iter = tmp;
    344     }
    345 }
    346 
    347 /**
    348  * xsltDecimalFormatGetByName:
    349  * @style: the XSLT stylesheet
    350  * @name: the decimal-format name to find
    351  *
    352  * Find decimal-format by name
    353  *
    354  * Returns the xsltDecimalFormatPtr
    355  */
    356 xsltDecimalFormatPtr
    357 xsltDecimalFormatGetByName(xsltStylesheetPtr style, xmlChar *name)
    358 {
    359     xsltDecimalFormatPtr result = NULL;
    360 
    361     if (name == NULL)
    362 	return style->decimalFormat;
    363 
    364     while (style != NULL) {
    365 	for (result = style->decimalFormat->next;
    366 	     result != NULL;
    367 	     result = result->next) {
    368 	    if (xmlStrEqual(name, result->name))
    369 		return result;
    370 	}
    371 	style = xsltNextImport(style);
    372     }
    373     return result;
    374 }
    375 
    376 
    377 /**
    378  * xsltNewTemplate:
    379  *
    380  * Create a new XSLT Template
    381  *
    382  * Returns the newly allocated xsltTemplatePtr or NULL in case of error
    383  */
    384 static xsltTemplatePtr
    385 xsltNewTemplate(void) {
    386     xsltTemplatePtr cur;
    387 
    388     cur = (xsltTemplatePtr) xmlMalloc(sizeof(xsltTemplate));
    389     if (cur == NULL) {
    390 	xsltTransformError(NULL, NULL, NULL,
    391 		"xsltNewTemplate : malloc failed\n");
    392 	return(NULL);
    393     }
    394     memset(cur, 0, sizeof(xsltTemplate));
    395     cur->priority = XSLT_PAT_NO_PRIORITY;
    396     return(cur);
    397 }
    398 
    399 /**
    400  * xsltFreeTemplate:
    401  * @template:  an XSLT template
    402  *
    403  * Free up the memory allocated by @template
    404  */
    405 static void
    406 xsltFreeTemplate(xsltTemplatePtr template) {
    407     if (template == NULL)
    408 	return;
    409     if (template->match) xmlFree(template->match);
    410 /*
    411 *   NOTE: @name and @nameURI are put into the string dict now.
    412 *   if (template->name) xmlFree(template->name);
    413 *   if (template->nameURI) xmlFree(template->nameURI);
    414 */
    415 /*
    416     if (template->mode) xmlFree(template->mode);
    417     if (template->modeURI) xmlFree(template->modeURI);
    418  */
    419     if (template->inheritedNs) xmlFree(template->inheritedNs);
    420     memset(template, -1, sizeof(xsltTemplate));
    421     xmlFree(template);
    422 }
    423 
    424 /**
    425  * xsltFreeTemplateList:
    426  * @template:  an XSLT template list
    427  *
    428  * Free up the memory allocated by all the elements of @template
    429  */
    430 static void
    431 xsltFreeTemplateList(xsltTemplatePtr template) {
    432     xsltTemplatePtr cur;
    433 
    434     while (template != NULL) {
    435 	cur = template;
    436 	template = template->next;
    437 	xsltFreeTemplate(cur);
    438     }
    439 }
    440 
    441 #ifdef XSLT_REFACTORED
    442 
    443 static void
    444 xsltFreeNsAliasList(xsltNsAliasPtr item)
    445 {
    446     xsltNsAliasPtr tmp;
    447 
    448     while (item) {
    449 	tmp = item;
    450 	item = item->next;
    451 	xmlFree(tmp);
    452     }
    453     return;
    454 }
    455 
    456 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
    457 static void
    458 xsltFreeNamespaceMap(xsltNsMapPtr item)
    459 {
    460     xsltNsMapPtr tmp;
    461 
    462     while (item) {
    463 	tmp = item;
    464 	item = item->next;
    465 	xmlFree(tmp);
    466     }
    467     return;
    468 }
    469 
    470 static xsltNsMapPtr
    471 xsltNewNamespaceMapItem(xsltCompilerCtxtPtr cctxt,
    472 			xmlDocPtr doc,
    473 			xmlNsPtr ns,
    474 			xmlNodePtr elem)
    475 {
    476     xsltNsMapPtr ret;
    477 
    478     if ((cctxt == NULL) || (doc == NULL) || (ns == NULL))
    479 	return(NULL);
    480 
    481     ret = (xsltNsMapPtr) xmlMalloc(sizeof(xsltNsMap));
    482     if (ret == NULL) {
    483 	xsltTransformError(NULL, cctxt->style, elem,
    484 	    "Internal error: (xsltNewNamespaceMapItem) "
    485 	    "memory allocation failed.\n");
    486 	return(NULL);
    487     }
    488     memset(ret, 0, sizeof(xsltNsMap));
    489     ret->doc = doc;
    490     ret->ns = ns;
    491     ret->origNsName = ns->href;
    492     /*
    493     * Store the item at current stylesheet-level.
    494     */
    495     if (cctxt->psData->nsMap != NULL)
    496 	ret->next = cctxt->psData->nsMap;
    497     cctxt->psData->nsMap = ret;
    498 
    499     return(ret);
    500 }
    501 #endif /* XSLT_REFACTORED_XSLT_NSCOMP */
    502 
    503 /**
    504  * xsltCompilerVarInfoFree:
    505  * @cctxt: the compilation context
    506  *
    507  * Frees the list of information for vars/params.
    508  */
    509 static void
    510 xsltCompilerVarInfoFree(xsltCompilerCtxtPtr cctxt)
    511 {
    512     xsltVarInfoPtr ivar = cctxt->ivars, ivartmp;
    513 
    514     while (ivar) {
    515 	ivartmp = ivar;
    516 	ivar = ivar->next;
    517 	xmlFree(ivartmp);
    518     }
    519 }
    520 
    521 /**
    522  * xsltCompilerCtxtFree:
    523  *
    524  * Free an XSLT compiler context.
    525  */
    526 static void
    527 xsltCompilationCtxtFree(xsltCompilerCtxtPtr cctxt)
    528 {
    529     if (cctxt == NULL)
    530 	return;
    531 #ifdef WITH_XSLT_DEBUG_PARSING
    532     xsltGenericDebug(xsltGenericDebugContext,
    533 	"Freeing compilation context\n");
    534     xsltGenericDebug(xsltGenericDebugContext,
    535 	"### Max inodes: %d\n", cctxt->maxNodeInfos);
    536     xsltGenericDebug(xsltGenericDebugContext,
    537 	"### Max LREs  : %d\n", cctxt->maxLREs);
    538 #endif
    539     /*
    540     * Free node-infos.
    541     */
    542     if (cctxt->inodeList != NULL) {
    543 	xsltCompilerNodeInfoPtr tmp, cur = cctxt->inodeList;
    544 	while (cur != NULL) {
    545 	    tmp = cur;
    546 	    cur = cur->next;
    547 	    xmlFree(tmp);
    548 	}
    549     }
    550     if (cctxt->tmpList != NULL)
    551 	xsltPointerListFree(cctxt->tmpList);
    552 #ifdef XSLT_REFACTORED_XPATHCOMP
    553     if (cctxt->xpathCtxt != NULL)
    554 	xmlXPathFreeContext(cctxt->xpathCtxt);
    555 #endif
    556     if (cctxt->nsAliases != NULL)
    557 	xsltFreeNsAliasList(cctxt->nsAliases);
    558 
    559     if (cctxt->ivars)
    560 	xsltCompilerVarInfoFree(cctxt);
    561 
    562     xmlFree(cctxt);
    563 }
    564 
    565 /**
    566  * xsltCompilerCreate:
    567  *
    568  * Creates an XSLT compiler context.
    569  *
    570  * Returns the pointer to the created xsltCompilerCtxt or
    571  *         NULL in case of an internal error.
    572  */
    573 static xsltCompilerCtxtPtr
    574 xsltCompilationCtxtCreate(xsltStylesheetPtr style) {
    575     xsltCompilerCtxtPtr ret;
    576 
    577     ret = (xsltCompilerCtxtPtr) xmlMalloc(sizeof(xsltCompilerCtxt));
    578     if (ret == NULL) {
    579 	xsltTransformError(NULL, style, NULL,
    580 	    "xsltCompilerCreate: allocation of compiler "
    581 	    "context failed.\n");
    582 	return(NULL);
    583     }
    584     memset(ret, 0, sizeof(xsltCompilerCtxt));
    585 
    586     ret->errSeverity = XSLT_ERROR_SEVERITY_ERROR;
    587     ret->tmpList = xsltPointerListCreate(20);
    588     if (ret->tmpList == NULL) {
    589 	goto internal_err;
    590     }
    591 #ifdef XSLT_REFACTORED_XPATHCOMP
    592     /*
    593     * Create the XPath compilation context in order
    594     * to speed up precompilation of XPath expressions.
    595     */
    596     ret->xpathCtxt = xmlXPathNewContext(NULL);
    597     if (ret->xpathCtxt == NULL)
    598 	goto internal_err;
    599 #endif
    600 
    601     return(ret);
    602 
    603 internal_err:
    604     xsltCompilationCtxtFree(ret);
    605     return(NULL);
    606 }
    607 
    608 static void
    609 xsltLREEffectiveNsNodesFree(xsltEffectiveNsPtr first)
    610 {
    611     xsltEffectiveNsPtr tmp;
    612 
    613     while (first != NULL) {
    614 	tmp = first;
    615 	first = first->nextInStore;
    616 	xmlFree(tmp);
    617     }
    618 }
    619 
    620 static void
    621 xsltFreePrincipalStylesheetData(xsltPrincipalStylesheetDataPtr data)
    622 {
    623     if (data == NULL)
    624 	return;
    625 
    626     if (data->inScopeNamespaces != NULL) {
    627 	int i;
    628 	xsltNsListContainerPtr nsi;
    629 	xsltPointerListPtr list =
    630 	    (xsltPointerListPtr) data->inScopeNamespaces;
    631 
    632 	for (i = 0; i < list->number; i++) {
    633 	    /*
    634 	    * REVISIT TODO: Free info of in-scope namespaces.
    635 	    */
    636 	    nsi = (xsltNsListContainerPtr) list->items[i];
    637 	    if (nsi->list != NULL)
    638 		xmlFree(nsi->list);
    639 	    xmlFree(nsi);
    640 	}
    641 	xsltPointerListFree(list);
    642 	data->inScopeNamespaces = NULL;
    643     }
    644 
    645     if (data->exclResultNamespaces != NULL) {
    646 	int i;
    647 	xsltPointerListPtr list = (xsltPointerListPtr)
    648 	    data->exclResultNamespaces;
    649 
    650 	for (i = 0; i < list->number; i++)
    651 	    xsltPointerListFree((xsltPointerListPtr) list->items[i]);
    652 
    653 	xsltPointerListFree(list);
    654 	data->exclResultNamespaces = NULL;
    655     }
    656 
    657     if (data->extElemNamespaces != NULL) {
    658 	xsltPointerListPtr list = (xsltPointerListPtr)
    659 	    data->extElemNamespaces;
    660 	int i;
    661 
    662 	for (i = 0; i < list->number; i++)
    663 	    xsltPointerListFree((xsltPointerListPtr) list->items[i]);
    664 
    665 	xsltPointerListFree(list);
    666 	data->extElemNamespaces = NULL;
    667     }
    668     if (data->effectiveNs) {
    669 	xsltLREEffectiveNsNodesFree(data->effectiveNs);
    670 	data->effectiveNs = NULL;
    671     }
    672 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
    673     xsltFreeNamespaceMap(data->nsMap);
    674 #endif
    675     xmlFree(data);
    676 }
    677 
    678 static xsltPrincipalStylesheetDataPtr
    679 xsltNewPrincipalStylesheetData(void)
    680 {
    681     xsltPrincipalStylesheetDataPtr ret;
    682 
    683     ret = (xsltPrincipalStylesheetDataPtr)
    684 	xmlMalloc(sizeof(xsltPrincipalStylesheetData));
    685     if (ret == NULL) {
    686 	xsltTransformError(NULL, NULL, NULL,
    687 	    "xsltNewPrincipalStylesheetData: memory allocation failed.\n");
    688 	return(NULL);
    689     }
    690     memset(ret, 0, sizeof(xsltPrincipalStylesheetData));
    691 
    692     /*
    693     * Global list of in-scope namespaces.
    694     */
    695     ret->inScopeNamespaces = xsltPointerListCreate(-1);
    696     if (ret->inScopeNamespaces == NULL)
    697 	goto internal_err;
    698     /*
    699     * Global list of excluded result ns-decls.
    700     */
    701     ret->exclResultNamespaces = xsltPointerListCreate(-1);
    702     if (ret->exclResultNamespaces == NULL)
    703 	goto internal_err;
    704     /*
    705     * Global list of extension instruction namespace names.
    706     */
    707     ret->extElemNamespaces = xsltPointerListCreate(-1);
    708     if (ret->extElemNamespaces == NULL)
    709 	goto internal_err;
    710 
    711     return(ret);
    712 
    713 internal_err:
    714 
    715     return(NULL);
    716 }
    717 
    718 #endif
    719 
    720 /**
    721  * xsltNewStylesheet:
    722  *
    723  * Create a new XSLT Stylesheet
    724  *
    725  * Returns the newly allocated xsltStylesheetPtr or NULL in case of error
    726  */
    727 xsltStylesheetPtr
    728 xsltNewStylesheet(void) {
    729     xsltStylesheetPtr ret = NULL;
    730 
    731     ret = (xsltStylesheetPtr) xmlMalloc(sizeof(xsltStylesheet));
    732     if (ret == NULL) {
    733 	xsltTransformError(NULL, NULL, NULL,
    734 		"xsltNewStylesheet : malloc failed\n");
    735 	goto internal_err;
    736     }
    737     memset(ret, 0, sizeof(xsltStylesheet));
    738 
    739     ret->omitXmlDeclaration = -1;
    740     ret->standalone = -1;
    741     ret->decimalFormat = xsltNewDecimalFormat(NULL);
    742     ret->indent = -1;
    743     ret->errors = 0;
    744     ret->warnings = 0;
    745     ret->exclPrefixNr = 0;
    746     ret->exclPrefixMax = 0;
    747     ret->exclPrefixTab = NULL;
    748     ret->extInfos = NULL;
    749     ret->extrasNr = 0;
    750     ret->internalized = 1;
    751     ret->literal_result = 0;
    752     ret->dict = xmlDictCreate();
    753 #ifdef WITH_XSLT_DEBUG
    754     xsltGenericDebug(xsltGenericDebugContext,
    755 	"creating dictionary for stylesheet\n");
    756 #endif
    757 
    758     xsltInit();
    759 
    760     return(ret);
    761 
    762 internal_err:
    763     if (ret != NULL)
    764 	xsltFreeStylesheet(ret);
    765     return(NULL);
    766 }
    767 
    768 /**
    769  * xsltAllocateExtra:
    770  * @style:  an XSLT stylesheet
    771  *
    772  * Allocate an extra runtime information slot statically while compiling
    773  * the stylesheet and return its number
    774  *
    775  * Returns the number of the slot
    776  */
    777 int
    778 xsltAllocateExtra(xsltStylesheetPtr style)
    779 {
    780     return(style->extrasNr++);
    781 }
    782 
    783 /**
    784  * xsltAllocateExtraCtxt:
    785  * @ctxt:  an XSLT transformation context
    786  *
    787  * Allocate an extra runtime information slot at run-time
    788  * and return its number
    789  * This make sure there is a slot ready in the transformation context
    790  *
    791  * Returns the number of the slot
    792  */
    793 int
    794 xsltAllocateExtraCtxt(xsltTransformContextPtr ctxt)
    795 {
    796     if (ctxt->extrasNr >= ctxt->extrasMax) {
    797 	int i;
    798 	if (ctxt->extrasNr == 0) {
    799 	    ctxt->extrasMax = 20;
    800 	    ctxt->extras = (xsltRuntimeExtraPtr)
    801 		xmlMalloc(ctxt->extrasMax * sizeof(xsltRuntimeExtra));
    802 	    if (ctxt->extras == NULL) {
    803 		xmlGenericError(xmlGenericErrorContext,
    804 			"xsltAllocateExtraCtxt: out of memory\n");
    805 		ctxt->state = XSLT_STATE_ERROR;
    806 		return(0);
    807 	    }
    808 	    for (i = 0;i < ctxt->extrasMax;i++) {
    809 		ctxt->extras[i].info = NULL;
    810 		ctxt->extras[i].deallocate = NULL;
    811 		ctxt->extras[i].val.ptr = NULL;
    812 	    }
    813 
    814 	} else {
    815 	    xsltRuntimeExtraPtr tmp;
    816 
    817 	    ctxt->extrasMax += 100;
    818 	    tmp = (xsltRuntimeExtraPtr) xmlRealloc(ctxt->extras,
    819 		            ctxt->extrasMax * sizeof(xsltRuntimeExtra));
    820 	    if (tmp == NULL) {
    821 		xmlGenericError(xmlGenericErrorContext,
    822 			"xsltAllocateExtraCtxt: out of memory\n");
    823 		ctxt->state = XSLT_STATE_ERROR;
    824 		return(0);
    825 	    }
    826 	    ctxt->extras = tmp;
    827 	    for (i = ctxt->extrasNr;i < ctxt->extrasMax;i++) {
    828 		ctxt->extras[i].info = NULL;
    829 		ctxt->extras[i].deallocate = NULL;
    830 		ctxt->extras[i].val.ptr = NULL;
    831 	    }
    832 	}
    833     }
    834     return(ctxt->extrasNr++);
    835 }
    836 
    837 /**
    838  * xsltFreeStylesheetList:
    839  * @style:  an XSLT stylesheet list
    840  *
    841  * Free up the memory allocated by the list @style
    842  */
    843 static void
    844 xsltFreeStylesheetList(xsltStylesheetPtr style) {
    845     xsltStylesheetPtr next;
    846 
    847     while (style != NULL) {
    848 	next = style->next;
    849 	xsltFreeStylesheet(style);
    850 	style = next;
    851     }
    852 }
    853 
    854 /**
    855  * xsltCleanupStylesheetTree:
    856  *
    857  * @doc: the document-node
    858  * @node: the element where the stylesheet is rooted at
    859  *
    860  * Actually @node need not be the document-element, but
    861  * currently Libxslt does not support embedded stylesheets.
    862  *
    863  * Returns 0 if OK, -1 on API or internal errors.
    864  */
    865 static int
    866 xsltCleanupStylesheetTree(xmlDocPtr doc ATTRIBUTE_UNUSED,
    867 			  xmlNodePtr rootElem ATTRIBUTE_UNUSED)
    868 {
    869 #if 0 /* TODO: Currently disabled, since probably not needed. */
    870     xmlNodePtr cur;
    871 
    872     if ((doc == NULL) || (rootElem == NULL) ||
    873 	(rootElem->type != XML_ELEMENT_NODE) ||
    874 	(doc != rootElem->doc))
    875 	return(-1);
    876 
    877     /*
    878     * Cleanup was suggested by Aleksey Sanin:
    879     * Clear the PSVI field to avoid problems if the
    880     * node-tree of the stylesheet is intended to be used for
    881     * further processing by the user (e.g. for compiling it
    882     * once again - although not recommended).
    883     */
    884 
    885     cur = rootElem;
    886     while (cur != NULL) {
    887 	if (cur->type == XML_ELEMENT_NODE) {
    888 	    /*
    889 	    * Clear the PSVI field.
    890 	    */
    891 	    cur->psvi = NULL;
    892 	    if (cur->children) {
    893 		cur = cur->children;
    894 		continue;
    895 	    }
    896 	}
    897 
    898 leave_node:
    899 	if (cur == rootElem)
    900 	    break;
    901 	if (cur->next != NULL)
    902 	    cur = cur->next;
    903 	else {
    904 	    cur = cur->parent;
    905 	    if (cur == NULL)
    906 		break;
    907 	    goto leave_node;
    908 	}
    909     }
    910 #endif /* #if 0 */
    911     return(0);
    912 }
    913 
    914 /**
    915  * xsltFreeStylesheet:
    916  * @style:  an XSLT stylesheet
    917  *
    918  * Free up the memory allocated by @style
    919  */
    920 void
    921 xsltFreeStylesheet(xsltStylesheetPtr style)
    922 {
    923     if (style == NULL)
    924         return;
    925 
    926 #ifdef XSLT_REFACTORED
    927     /*
    928     * Start with a cleanup of the main stylesheet's doc.
    929     */
    930     if ((style->principal == style) && (style->doc))
    931 	xsltCleanupStylesheetTree(style->doc,
    932 	    xmlDocGetRootElement(style->doc));
    933 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
    934     /*
    935     * Restore changed ns-decls before freeing the document.
    936     */
    937     if ((style->doc != NULL) &&
    938 	XSLT_HAS_INTERNAL_NSMAP(style))
    939     {
    940 	xsltRestoreDocumentNamespaces(XSLT_GET_INTERNAL_NSMAP(style),
    941 	    style->doc);
    942     }
    943 #endif /* XSLT_REFACTORED_XSLT_NSCOMP */
    944 #else
    945     /*
    946     * Start with a cleanup of the main stylesheet's doc.
    947     */
    948     if ((style->parent == NULL) && (style->doc))
    949 	xsltCleanupStylesheetTree(style->doc,
    950 	    xmlDocGetRootElement(style->doc));
    951 #endif /* XSLT_REFACTORED */
    952 
    953     xsltFreeKeys(style);
    954     xsltFreeExts(style);
    955     xsltFreeTemplateHashes(style);
    956     xsltFreeDecimalFormatList(style);
    957     xsltFreeTemplateList(style->templates);
    958     xsltFreeAttributeSetsHashes(style);
    959     xsltFreeNamespaceAliasHashes(style);
    960     xsltFreeStylePreComps(style);
    961     /*
    962     * Free documents of all included stylsheet modules of this
    963     * stylesheet level.
    964     */
    965     xsltFreeStyleDocuments(style);
    966     /*
    967     * TODO: Best time to shutdown extension stuff?
    968     */
    969     xsltShutdownExts(style);
    970 
    971     if (style->variables != NULL)
    972         xsltFreeStackElemList(style->variables);
    973     if (style->cdataSection != NULL)
    974         xmlHashFree(style->cdataSection, NULL);
    975     if (style->stripSpaces != NULL)
    976         xmlHashFree(style->stripSpaces, NULL);
    977     if (style->nsHash != NULL)
    978         xmlHashFree(style->nsHash, NULL);
    979     if (style->exclPrefixTab != NULL)
    980         xmlFree(style->exclPrefixTab);
    981     if (style->method != NULL)
    982         xmlFree(style->method);
    983     if (style->methodURI != NULL)
    984         xmlFree(style->methodURI);
    985     if (style->version != NULL)
    986         xmlFree(style->version);
    987     if (style->encoding != NULL)
    988         xmlFree(style->encoding);
    989     if (style->doctypePublic != NULL)
    990         xmlFree(style->doctypePublic);
    991     if (style->doctypeSystem != NULL)
    992         xmlFree(style->doctypeSystem);
    993     if (style->mediaType != NULL)
    994         xmlFree(style->mediaType);
    995     if (style->attVTs)
    996         xsltFreeAVTList(style->attVTs);
    997     if (style->imports != NULL)
    998         xsltFreeStylesheetList(style->imports);
    999 
   1000 #ifdef XSLT_REFACTORED
   1001     /*
   1002     * If this is the principal stylesheet, then
   1003     * free its internal data.
   1004     */
   1005     if (style->principal == style) {
   1006 	if (style->principalData) {
   1007 	    xsltFreePrincipalStylesheetData(style->principalData);
   1008 	    style->principalData = NULL;
   1009 	}
   1010     }
   1011 #endif
   1012     /*
   1013     * Better to free the main document of this stylesheet level
   1014     * at the end - so here.
   1015     */
   1016     if (style->doc != NULL) {
   1017         xmlFreeDoc(style->doc);
   1018     }
   1019 
   1020 #ifdef WITH_XSLT_DEBUG
   1021     xsltGenericDebug(xsltGenericDebugContext,
   1022                      "freeing dictionary from stylesheet\n");
   1023 #endif
   1024     xmlDictFree(style->dict);
   1025 
   1026     memset(style, -1, sizeof(xsltStylesheet));
   1027     xmlFree(style);
   1028 }
   1029 
   1030 /************************************************************************
   1031  *									*
   1032  *		Parsing of an XSLT Stylesheet				*
   1033  *									*
   1034  ************************************************************************/
   1035 
   1036 #ifdef XSLT_REFACTORED
   1037     /*
   1038     * This is now performed in an optimized way in xsltParseXSLTTemplate.
   1039     */
   1040 #else
   1041 /**
   1042  * xsltGetInheritedNsList:
   1043  * @style:  the stylesheet
   1044  * @template: the template
   1045  * @node:  the current node
   1046  *
   1047  * Search all the namespace applying to a given element except the ones
   1048  * from excluded output prefixes currently in scope. Initialize the
   1049  * template inheritedNs list with it.
   1050  *
   1051  * Returns the number of entries found
   1052  */
   1053 static int
   1054 xsltGetInheritedNsList(xsltStylesheetPtr style,
   1055 	               xsltTemplatePtr template,
   1056 	               xmlNodePtr node)
   1057 {
   1058     xmlNsPtr cur;
   1059     xmlNsPtr *ret = NULL;
   1060     int nbns = 0;
   1061     int maxns = 10;
   1062     int i;
   1063 
   1064     if ((style == NULL) || (template == NULL) || (node == NULL) ||
   1065 	(template->inheritedNsNr != 0) || (template->inheritedNs != NULL))
   1066 	return(0);
   1067     while (node != NULL) {
   1068         if (node->type == XML_ELEMENT_NODE) {
   1069             cur = node->nsDef;
   1070             while (cur != NULL) {
   1071 		if (xmlStrEqual(cur->href, XSLT_NAMESPACE))
   1072 		    goto skip_ns;
   1073 
   1074 		if ((cur->prefix != NULL) &&
   1075 		    (xsltCheckExtPrefix(style, cur->prefix)))
   1076 		    goto skip_ns;
   1077 		/*
   1078 		* Check if this namespace was excluded.
   1079 		* Note that at this point only the exclusions defined
   1080 		* on the topmost stylesheet element are in the exclusion-list.
   1081 		*/
   1082 		for (i = 0;i < style->exclPrefixNr;i++) {
   1083 		    if (xmlStrEqual(cur->href, style->exclPrefixTab[i]))
   1084 			goto skip_ns;
   1085 		}
   1086                 if (ret == NULL) {
   1087                     ret =
   1088                         (xmlNsPtr *) xmlMalloc((maxns + 1) *
   1089                                                sizeof(xmlNsPtr));
   1090                     if (ret == NULL) {
   1091                         xmlGenericError(xmlGenericErrorContext,
   1092                                         "xsltGetInheritedNsList : out of memory!\n");
   1093                         return(0);
   1094                     }
   1095                     ret[nbns] = NULL;
   1096                 }
   1097 		/*
   1098 		* Skip shadowed namespace bindings.
   1099 		*/
   1100                 for (i = 0; i < nbns; i++) {
   1101                     if ((cur->prefix == ret[i]->prefix) ||
   1102                         (xmlStrEqual(cur->prefix, ret[i]->prefix)))
   1103                         break;
   1104                 }
   1105                 if (i >= nbns) {
   1106                     if (nbns >= maxns) {
   1107                         maxns *= 2;
   1108                         ret = (xmlNsPtr *) xmlRealloc(ret,
   1109                                                       (maxns +
   1110                                                        1) *
   1111                                                       sizeof(xmlNsPtr));
   1112                         if (ret == NULL) {
   1113                             xmlGenericError(xmlGenericErrorContext,
   1114                                             "xsltGetInheritedNsList : realloc failed!\n");
   1115                             return(0);
   1116                         }
   1117                     }
   1118                     ret[nbns++] = cur;
   1119                     ret[nbns] = NULL;
   1120                 }
   1121 skip_ns:
   1122                 cur = cur->next;
   1123             }
   1124         }
   1125         node = node->parent;
   1126     }
   1127     if (nbns != 0) {
   1128 #ifdef WITH_XSLT_DEBUG_PARSING
   1129         xsltGenericDebug(xsltGenericDebugContext,
   1130                          "template has %d inherited namespaces\n", nbns);
   1131 #endif
   1132 	template->inheritedNsNr = nbns;
   1133 	template->inheritedNs = ret;
   1134     }
   1135     return (nbns);
   1136 }
   1137 #endif /* else of XSLT_REFACTORED */
   1138 
   1139 /**
   1140  * xsltParseStylesheetOutput:
   1141  * @style:  the XSLT stylesheet
   1142  * @cur:  the "output" element
   1143  *
   1144  * parse an XSLT stylesheet output element and record
   1145  * information related to the stylesheet output
   1146  */
   1147 
   1148 void
   1149 xsltParseStylesheetOutput(xsltStylesheetPtr style, xmlNodePtr cur)
   1150 {
   1151     xmlChar *elements,
   1152      *prop;
   1153     xmlChar *element,
   1154      *end;
   1155 
   1156     if ((cur == NULL) || (style == NULL))
   1157         return;
   1158 
   1159     prop = xmlGetNsProp(cur, (const xmlChar *) "version", NULL);
   1160     if (prop != NULL) {
   1161         if (style->version != NULL)
   1162             xmlFree(style->version);
   1163         style->version = prop;
   1164     }
   1165 
   1166     prop = xmlGetNsProp(cur, (const xmlChar *) "encoding", NULL);
   1167     if (prop != NULL) {
   1168         if (style->encoding != NULL)
   1169             xmlFree(style->encoding);
   1170         style->encoding = prop;
   1171     }
   1172 
   1173     /* relaxed to support xt:document
   1174     * TODO KB: What does "relaxed to support xt:document" mean?
   1175     */
   1176     prop = xmlGetNsProp(cur, (const xmlChar *) "method", NULL);
   1177     if (prop != NULL) {
   1178         const xmlChar *URI;
   1179 
   1180         if (style->method != NULL)
   1181             xmlFree(style->method);
   1182         style->method = NULL;
   1183         if (style->methodURI != NULL)
   1184             xmlFree(style->methodURI);
   1185         style->methodURI = NULL;
   1186 
   1187 	/*
   1188 	* TODO: Don't use xsltGetQNameURI().
   1189 	*/
   1190 	URI = xsltGetQNameURI(cur, &prop);
   1191 	if (prop == NULL) {
   1192 	    if (style != NULL) style->errors++;
   1193 	} else if (URI == NULL) {
   1194             if ((xmlStrEqual(prop, (const xmlChar *) "xml")) ||
   1195                 (xmlStrEqual(prop, (const xmlChar *) "html")) ||
   1196                 (xmlStrEqual(prop, (const xmlChar *) "text"))) {
   1197                 style->method = prop;
   1198             } else {
   1199 		xsltTransformError(NULL, style, cur,
   1200                                  "invalid value for method: %s\n", prop);
   1201                 if (style != NULL) style->warnings++;
   1202             }
   1203 	} else {
   1204 	    style->method = prop;
   1205 	    style->methodURI = xmlStrdup(URI);
   1206 	}
   1207     }
   1208 
   1209     prop = xmlGetNsProp(cur, (const xmlChar *) "doctype-system", NULL);
   1210     if (prop != NULL) {
   1211         if (style->doctypeSystem != NULL)
   1212             xmlFree(style->doctypeSystem);
   1213         style->doctypeSystem = prop;
   1214     }
   1215 
   1216     prop = xmlGetNsProp(cur, (const xmlChar *) "doctype-public", NULL);
   1217     if (prop != NULL) {
   1218         if (style->doctypePublic != NULL)
   1219             xmlFree(style->doctypePublic);
   1220         style->doctypePublic = prop;
   1221     }
   1222 
   1223     prop = xmlGetNsProp(cur, (const xmlChar *) "standalone", NULL);
   1224     if (prop != NULL) {
   1225         if (xmlStrEqual(prop, (const xmlChar *) "yes")) {
   1226             style->standalone = 1;
   1227         } else if (xmlStrEqual(prop, (const xmlChar *) "no")) {
   1228             style->standalone = 0;
   1229         } else {
   1230 	    xsltTransformError(NULL, style, cur,
   1231                              "invalid value for standalone: %s\n", prop);
   1232             style->errors++;
   1233         }
   1234         xmlFree(prop);
   1235     }
   1236 
   1237     prop = xmlGetNsProp(cur, (const xmlChar *) "indent", NULL);
   1238     if (prop != NULL) {
   1239         if (xmlStrEqual(prop, (const xmlChar *) "yes")) {
   1240             style->indent = 1;
   1241         } else if (xmlStrEqual(prop, (const xmlChar *) "no")) {
   1242             style->indent = 0;
   1243         } else {
   1244 	    xsltTransformError(NULL, style, cur,
   1245                              "invalid value for indent: %s\n", prop);
   1246             style->errors++;
   1247         }
   1248         xmlFree(prop);
   1249     }
   1250 
   1251     prop = xmlGetNsProp(cur, (const xmlChar *) "omit-xml-declaration", NULL);
   1252     if (prop != NULL) {
   1253         if (xmlStrEqual(prop, (const xmlChar *) "yes")) {
   1254             style->omitXmlDeclaration = 1;
   1255         } else if (xmlStrEqual(prop, (const xmlChar *) "no")) {
   1256             style->omitXmlDeclaration = 0;
   1257         } else {
   1258 	    xsltTransformError(NULL, style, cur,
   1259                              "invalid value for omit-xml-declaration: %s\n",
   1260                              prop);
   1261             style->errors++;
   1262         }
   1263         xmlFree(prop);
   1264     }
   1265 
   1266     elements = xmlGetNsProp(cur, (const xmlChar *) "cdata-section-elements",
   1267 	NULL);
   1268     if (elements != NULL) {
   1269         if (style->cdataSection == NULL)
   1270             style->cdataSection = xmlHashCreate(10);
   1271         if (style->cdataSection == NULL)
   1272             return;
   1273 
   1274         element = elements;
   1275         while (*element != 0) {
   1276             while (IS_BLANK(*element))
   1277                 element++;
   1278             if (*element == 0)
   1279                 break;
   1280             end = element;
   1281             while ((*end != 0) && (!IS_BLANK(*end)))
   1282                 end++;
   1283             element = xmlStrndup(element, end - element);
   1284             if (element) {
   1285 #ifdef WITH_XSLT_DEBUG_PARSING
   1286                 xsltGenericDebug(xsltGenericDebugContext,
   1287                                  "add cdata section output element %s\n",
   1288                                  element);
   1289 #endif
   1290 		if (xmlValidateQName(BAD_CAST element, 0) != 0) {
   1291 		    xsltTransformError(NULL, style, cur,
   1292 			"Attribute 'cdata-section-elements': The value "
   1293 			"'%s' is not a valid QName.\n", element);
   1294 		    xmlFree(element);
   1295 		    style->errors++;
   1296 		} else {
   1297 		    const xmlChar *URI;
   1298 
   1299 		    /*
   1300 		    * TODO: Don't use xsltGetQNameURI().
   1301 		    */
   1302 		    URI = xsltGetQNameURI(cur, &element);
   1303 		    if (element == NULL) {
   1304 			/*
   1305 			* TODO: We'll report additionally an error
   1306 			*  via the stylesheet's error handling.
   1307 			*/
   1308 			xsltTransformError(NULL, style, cur,
   1309 			    "Attribute 'cdata-section-elements': The value "
   1310 			    "'%s' is not a valid QName.\n", element);
   1311 			style->errors++;
   1312 		    } else {
   1313 			xmlNsPtr ns;
   1314 
   1315 			/*
   1316 			* XSLT-1.0 "Each QName is expanded into an
   1317 			*  expanded-name using the namespace declarations in
   1318 			*  effect on the xsl:output element in which the QName
   1319 			*  occurs; if there is a default namespace, it is used
   1320 			*  for QNames that do not have a prefix"
   1321 			* NOTE: Fix of bug #339570.
   1322 			*/
   1323 			if (URI == NULL) {
   1324 			    ns = xmlSearchNs(style->doc, cur, NULL);
   1325 			    if (ns != NULL)
   1326 				URI = ns->href;
   1327 			}
   1328 			xmlHashAddEntry2(style->cdataSection, element, URI,
   1329 			    (void *) "cdata");
   1330 			xmlFree(element);
   1331 		    }
   1332 		}
   1333             }
   1334             element = end;
   1335         }
   1336         xmlFree(elements);
   1337     }
   1338 
   1339     prop = xmlGetNsProp(cur, (const xmlChar *) "media-type", NULL);
   1340     if (prop != NULL) {
   1341 	if (style->mediaType)
   1342 	    xmlFree(style->mediaType);
   1343 	style->mediaType = prop;
   1344     }
   1345     if (cur->children != NULL) {
   1346 	xsltParseContentError(style, cur->children);
   1347     }
   1348 }
   1349 
   1350 /**
   1351  * xsltParseStylesheetDecimalFormat:
   1352  * @style:  the XSLT stylesheet
   1353  * @cur:  the "decimal-format" element
   1354  *
   1355  * <!-- Category: top-level-element -->
   1356  * <xsl:decimal-format
   1357  *   name = qname, decimal-separator = char, grouping-separator = char,
   1358  *   infinity = string, minus-sign = char, NaN = string, percent = char
   1359  *   per-mille = char, zero-digit = char, digit = char,
   1360  * pattern-separator = char />
   1361  *
   1362  * parse an XSLT stylesheet decimal-format element and
   1363  * and record the formatting characteristics
   1364  */
   1365 static void
   1366 xsltParseStylesheetDecimalFormat(xsltStylesheetPtr style, xmlNodePtr cur)
   1367 {
   1368     xmlChar *prop;
   1369     xsltDecimalFormatPtr format;
   1370     xsltDecimalFormatPtr iter;
   1371 
   1372     if ((cur == NULL) || (style == NULL))
   1373 	return;
   1374 
   1375     format = style->decimalFormat;
   1376 
   1377     prop = xmlGetNsProp(cur, BAD_CAST("name"), NULL);
   1378     if (prop != NULL) {
   1379 	format = xsltDecimalFormatGetByName(style, prop);
   1380 	if (format != NULL) {
   1381 	    xsltTransformError(NULL, style, cur,
   1382 	 "xsltParseStylestyleDecimalFormat: %s already exists\n", prop);
   1383 	    if (style != NULL) style->warnings++;
   1384 	    return;
   1385 	}
   1386 	format = xsltNewDecimalFormat(prop);
   1387 	if (format == NULL) {
   1388 	    xsltTransformError(NULL, style, cur,
   1389      "xsltParseStylestyleDecimalFormat: failed creating new decimal-format\n");
   1390 	    if (style != NULL) style->errors++;
   1391 	    return;
   1392 	}
   1393 	/* Append new decimal-format structure */
   1394 	for (iter = style->decimalFormat; iter->next; iter = iter->next)
   1395 	    ;
   1396 	if (iter)
   1397 	    iter->next = format;
   1398     }
   1399 
   1400     prop = xmlGetNsProp(cur, (const xmlChar *)"decimal-separator", NULL);
   1401     if (prop != NULL) {
   1402 	if (format->decimalPoint != NULL) xmlFree(format->decimalPoint);
   1403 	format->decimalPoint  = prop;
   1404     }
   1405 
   1406     prop = xmlGetNsProp(cur, (const xmlChar *)"grouping-separator", NULL);
   1407     if (prop != NULL) {
   1408 	if (format->grouping != NULL) xmlFree(format->grouping);
   1409 	format->grouping  = prop;
   1410     }
   1411 
   1412     prop = xmlGetNsProp(cur, (const xmlChar *)"infinity", NULL);
   1413     if (prop != NULL) {
   1414 	if (format->infinity != NULL) xmlFree(format->infinity);
   1415 	format->infinity  = prop;
   1416     }
   1417 
   1418     prop = xmlGetNsProp(cur, (const xmlChar *)"minus-sign", NULL);
   1419     if (prop != NULL) {
   1420 	if (format->minusSign != NULL) xmlFree(format->minusSign);
   1421 	format->minusSign  = prop;
   1422     }
   1423 
   1424     prop = xmlGetNsProp(cur, (const xmlChar *)"NaN", NULL);
   1425     if (prop != NULL) {
   1426 	if (format->noNumber != NULL) xmlFree(format->noNumber);
   1427 	format->noNumber  = prop;
   1428     }
   1429 
   1430     prop = xmlGetNsProp(cur, (const xmlChar *)"percent", NULL);
   1431     if (prop != NULL) {
   1432 	if (format->percent != NULL) xmlFree(format->percent);
   1433 	format->percent  = prop;
   1434     }
   1435 
   1436     prop = xmlGetNsProp(cur, (const xmlChar *)"per-mille", NULL);
   1437     if (prop != NULL) {
   1438 	if (format->permille != NULL) xmlFree(format->permille);
   1439 	format->permille  = prop;
   1440     }
   1441 
   1442     prop = xmlGetNsProp(cur, (const xmlChar *)"zero-digit", NULL);
   1443     if (prop != NULL) {
   1444 	if (format->zeroDigit != NULL) xmlFree(format->zeroDigit);
   1445 	format->zeroDigit  = prop;
   1446     }
   1447 
   1448     prop = xmlGetNsProp(cur, (const xmlChar *)"digit", NULL);
   1449     if (prop != NULL) {
   1450 	if (format->digit != NULL) xmlFree(format->digit);
   1451 	format->digit  = prop;
   1452     }
   1453 
   1454     prop = xmlGetNsProp(cur, (const xmlChar *)"pattern-separator", NULL);
   1455     if (prop != NULL) {
   1456 	if (format->patternSeparator != NULL) xmlFree(format->patternSeparator);
   1457 	format->patternSeparator  = prop;
   1458     }
   1459     if (cur->children != NULL) {
   1460 	xsltParseContentError(style, cur->children);
   1461     }
   1462 }
   1463 
   1464 /**
   1465  * xsltParseStylesheetPreserveSpace:
   1466  * @style:  the XSLT stylesheet
   1467  * @cur:  the "preserve-space" element
   1468  *
   1469  * parse an XSLT stylesheet preserve-space element and record
   1470  * elements needing preserving
   1471  */
   1472 
   1473 static void
   1474 xsltParseStylesheetPreserveSpace(xsltStylesheetPtr style, xmlNodePtr cur) {
   1475     xmlChar *elements;
   1476     xmlChar *element, *end;
   1477 
   1478     if ((cur == NULL) || (style == NULL))
   1479 	return;
   1480 
   1481     elements = xmlGetNsProp(cur, (const xmlChar *)"elements", NULL);
   1482     if (elements == NULL) {
   1483 	xsltTransformError(NULL, style, cur,
   1484 	    "xsltParseStylesheetPreserveSpace: missing elements attribute\n");
   1485 	if (style != NULL) style->warnings++;
   1486 	return;
   1487     }
   1488 
   1489     if (style->stripSpaces == NULL)
   1490 	style->stripSpaces = xmlHashCreate(10);
   1491     if (style->stripSpaces == NULL)
   1492 	return;
   1493 
   1494     element = elements;
   1495     while (*element != 0) {
   1496 	while (IS_BLANK(*element)) element++;
   1497 	if (*element == 0)
   1498 	    break;
   1499         end = element;
   1500 	while ((*end != 0) && (!IS_BLANK(*end))) end++;
   1501 	element = xmlStrndup(element, end - element);
   1502 	if (element) {
   1503 #ifdef WITH_XSLT_DEBUG_PARSING
   1504 	    xsltGenericDebug(xsltGenericDebugContext,
   1505 		"add preserved space element %s\n", element);
   1506 #endif
   1507 	    if (xmlStrEqual(element, (const xmlChar *)"*")) {
   1508 		style->stripAll = -1;
   1509 	    } else {
   1510 		const xmlChar *URI;
   1511 
   1512 		/*
   1513 		* TODO: Don't use xsltGetQNameURI().
   1514 		*/
   1515                 URI = xsltGetQNameURI(cur, &element);
   1516 
   1517 		xmlHashAddEntry2(style->stripSpaces, element, URI,
   1518 				(xmlChar *) "preserve");
   1519 	    }
   1520 	    xmlFree(element);
   1521 	}
   1522 	element = end;
   1523     }
   1524     xmlFree(elements);
   1525     if (cur->children != NULL) {
   1526 	xsltParseContentError(style, cur->children);
   1527     }
   1528 }
   1529 
   1530 #ifdef XSLT_REFACTORED
   1531 #else
   1532 /**
   1533  * xsltParseStylesheetExtPrefix:
   1534  * @style:  the XSLT stylesheet
   1535  * @template:  the "extension-element-prefixes" prefix
   1536  *
   1537  * parse an XSLT stylesheet's "extension-element-prefix" attribute value
   1538  * and register the namespaces of extension instruction.
   1539  * SPEC "A namespace is designated as an extension namespace by using
   1540  *   an extension-element-prefixes attribute on:
   1541  *   1) an xsl:stylesheet element
   1542  *   2) an xsl:extension-element-prefixes attribute on a
   1543  *      literal result element
   1544  *   3) an extension instruction."
   1545  */
   1546 static void
   1547 xsltParseStylesheetExtPrefix(xsltStylesheetPtr style, xmlNodePtr cur,
   1548 			     int isXsltElem) {
   1549     xmlChar *prefixes;
   1550     xmlChar *prefix, *end;
   1551 
   1552     if ((cur == NULL) || (style == NULL))
   1553 	return;
   1554 
   1555     if (isXsltElem) {
   1556 	/* For xsl:stylesheet/xsl:transform. */
   1557 	prefixes = xmlGetNsProp(cur,
   1558 	    (const xmlChar *)"extension-element-prefixes", NULL);
   1559     } else {
   1560 	/* For literal result elements and extension instructions. */
   1561 	prefixes = xmlGetNsProp(cur,
   1562 	    (const xmlChar *)"extension-element-prefixes", XSLT_NAMESPACE);
   1563     }
   1564     if (prefixes == NULL) {
   1565 	return;
   1566     }
   1567 
   1568     prefix = prefixes;
   1569     while (*prefix != 0) {
   1570 	while (IS_BLANK(*prefix)) prefix++;
   1571 	if (*prefix == 0)
   1572 	    break;
   1573         end = prefix;
   1574 	while ((*end != 0) && (!IS_BLANK(*end))) end++;
   1575 	prefix = xmlStrndup(prefix, end - prefix);
   1576 	if (prefix) {
   1577 	    xmlNsPtr ns;
   1578 
   1579 	    if (xmlStrEqual(prefix, (const xmlChar *)"#default"))
   1580 		ns = xmlSearchNs(style->doc, cur, NULL);
   1581 	    else
   1582 		ns = xmlSearchNs(style->doc, cur, prefix);
   1583 	    if (ns == NULL) {
   1584 		xsltTransformError(NULL, style, cur,
   1585 	    "xsl:extension-element-prefix : undefined namespace %s\n",
   1586 	                         prefix);
   1587 		if (style != NULL) style->warnings++;
   1588 	    } else {
   1589 #ifdef WITH_XSLT_DEBUG_PARSING
   1590 		xsltGenericDebug(xsltGenericDebugContext,
   1591 		    "add extension prefix %s\n", prefix);
   1592 #endif
   1593 		xsltRegisterExtPrefix(style, prefix, ns->href);
   1594 	    }
   1595 	    xmlFree(prefix);
   1596 	}
   1597 	prefix = end;
   1598     }
   1599     xmlFree(prefixes);
   1600 }
   1601 #endif /* else of XSLT_REFACTORED */
   1602 
   1603 /**
   1604  * xsltParseStylesheetStripSpace:
   1605  * @style:  the XSLT stylesheet
   1606  * @cur:  the "strip-space" element
   1607  *
   1608  * parse an XSLT stylesheet's strip-space element and record
   1609  * the elements needing stripping
   1610  */
   1611 
   1612 static void
   1613 xsltParseStylesheetStripSpace(xsltStylesheetPtr style, xmlNodePtr cur) {
   1614     xmlChar *elements;
   1615     xmlChar *element, *end;
   1616 
   1617     if ((cur == NULL) || (style == NULL))
   1618 	return;
   1619 
   1620     elements = xmlGetNsProp(cur, (const xmlChar *)"elements", NULL);
   1621     if (elements == NULL) {
   1622 	xsltTransformError(NULL, style, cur,
   1623 	    "xsltParseStylesheetStripSpace: missing elements attribute\n");
   1624 	if (style != NULL) style->warnings++;
   1625 	return;
   1626     }
   1627 
   1628     if (style->stripSpaces == NULL)
   1629 	style->stripSpaces = xmlHashCreate(10);
   1630     if (style->stripSpaces == NULL)
   1631 	return;
   1632 
   1633     element = elements;
   1634     while (*element != 0) {
   1635 	while (IS_BLANK(*element)) element++;
   1636 	if (*element == 0)
   1637 	    break;
   1638         end = element;
   1639 	while ((*end != 0) && (!IS_BLANK(*end))) end++;
   1640 	element = xmlStrndup(element, end - element);
   1641 	if (element) {
   1642 #ifdef WITH_XSLT_DEBUG_PARSING
   1643 	    xsltGenericDebug(xsltGenericDebugContext,
   1644 		"add stripped space element %s\n", element);
   1645 #endif
   1646 	    if (xmlStrEqual(element, (const xmlChar *)"*")) {
   1647 		style->stripAll = 1;
   1648 	    } else {
   1649 		const xmlChar *URI;
   1650 
   1651 		/*
   1652 		* TODO: Don't use xsltGetQNameURI().
   1653 		*/
   1654                 URI = xsltGetQNameURI(cur, &element);
   1655 
   1656 		xmlHashAddEntry2(style->stripSpaces, element, URI,
   1657 			        (xmlChar *) "strip");
   1658 	    }
   1659 	    xmlFree(element);
   1660 	}
   1661 	element = end;
   1662     }
   1663     xmlFree(elements);
   1664     if (cur->children != NULL) {
   1665 	xsltParseContentError(style, cur->children);
   1666     }
   1667 }
   1668 
   1669 #ifdef XSLT_REFACTORED
   1670 #else
   1671 /**
   1672  * xsltParseStylesheetExcludePrefix:
   1673  * @style:  the XSLT stylesheet
   1674  * @cur:  the current point in the stylesheet
   1675  *
   1676  * parse an XSLT stylesheet exclude prefix and record
   1677  * namespaces needing stripping
   1678  *
   1679  * Returns the number of Excluded prefixes added at that level
   1680  */
   1681 
   1682 static int
   1683 xsltParseStylesheetExcludePrefix(xsltStylesheetPtr style, xmlNodePtr cur,
   1684 				 int isXsltElem)
   1685 {
   1686     int nb = 0;
   1687     xmlChar *prefixes;
   1688     xmlChar *prefix, *end;
   1689 
   1690     if ((cur == NULL) || (style == NULL))
   1691 	return(0);
   1692 
   1693     if (isXsltElem)
   1694 	prefixes = xmlGetNsProp(cur,
   1695 	    (const xmlChar *)"exclude-result-prefixes", NULL);
   1696     else
   1697 	prefixes = xmlGetNsProp(cur,
   1698 	    (const xmlChar *)"exclude-result-prefixes", XSLT_NAMESPACE);
   1699 
   1700     if (prefixes == NULL) {
   1701 	return(0);
   1702     }
   1703 
   1704     prefix = prefixes;
   1705     while (*prefix != 0) {
   1706 	while (IS_BLANK(*prefix)) prefix++;
   1707 	if (*prefix == 0)
   1708 	    break;
   1709         end = prefix;
   1710 	while ((*end != 0) && (!IS_BLANK(*end))) end++;
   1711 	prefix = xmlStrndup(prefix, end - prefix);
   1712 	if (prefix) {
   1713 	    xmlNsPtr ns;
   1714 
   1715 	    if (xmlStrEqual(prefix, (const xmlChar *)"#default"))
   1716 		ns = xmlSearchNs(style->doc, cur, NULL);
   1717 	    else
   1718 		ns = xmlSearchNs(style->doc, cur, prefix);
   1719 	    if (ns == NULL) {
   1720 		xsltTransformError(NULL, style, cur,
   1721 	    "xsl:exclude-result-prefixes : undefined namespace %s\n",
   1722 	                         prefix);
   1723 		if (style != NULL) style->warnings++;
   1724 	    } else {
   1725 		if (exclPrefixPush(style, (xmlChar *) ns->href) >= 0) {
   1726 #ifdef WITH_XSLT_DEBUG_PARSING
   1727 		    xsltGenericDebug(xsltGenericDebugContext,
   1728 			"exclude result prefix %s\n", prefix);
   1729 #endif
   1730 		    nb++;
   1731 		}
   1732 	    }
   1733 	    xmlFree(prefix);
   1734 	}
   1735 	prefix = end;
   1736     }
   1737     xmlFree(prefixes);
   1738     return(nb);
   1739 }
   1740 #endif /* else of XSLT_REFACTORED */
   1741 
   1742 #ifdef XSLT_REFACTORED
   1743 
   1744 /*
   1745 * xsltTreeEnsureXMLDecl:
   1746 * @doc: the doc
   1747 *
   1748 * BIG NOTE:
   1749 *  This was copy&pasted from Libxml2's xmlTreeEnsureXMLDecl() in "tree.c".
   1750 * Ensures that there is an XML namespace declaration on the doc.
   1751 *
   1752 * Returns the XML ns-struct or NULL on API and internal errors.
   1753 */
   1754 static xmlNsPtr
   1755 xsltTreeEnsureXMLDecl(xmlDocPtr doc)
   1756 {
   1757     if (doc == NULL)
   1758 	return (NULL);
   1759     if (doc->oldNs != NULL)
   1760 	return (doc->oldNs);
   1761     {
   1762 	xmlNsPtr ns;
   1763 	ns = (xmlNsPtr) xmlMalloc(sizeof(xmlNs));
   1764 	if (ns == NULL) {
   1765 	    xmlGenericError(xmlGenericErrorContext,
   1766 		"xsltTreeEnsureXMLDecl: Failed to allocate "
   1767 		"the XML namespace.\n");
   1768 	    return (NULL);
   1769 	}
   1770 	memset(ns, 0, sizeof(xmlNs));
   1771 	ns->type = XML_LOCAL_NAMESPACE;
   1772 	/*
   1773 	* URGENT TODO: revisit this.
   1774 	*/
   1775 #ifdef LIBXML_NAMESPACE_DICT
   1776 	if (doc->dict)
   1777 	    ns->href = xmlDictLookup(doc->dict, XML_XML_NAMESPACE, -1);
   1778 	else
   1779 	    ns->href = xmlStrdup(XML_XML_NAMESPACE);
   1780 #else
   1781 	ns->href = xmlStrdup(XML_XML_NAMESPACE);
   1782 #endif
   1783 	ns->prefix = xmlStrdup((const xmlChar *)"xml");
   1784 	doc->oldNs = ns;
   1785 	return (ns);
   1786     }
   1787 }
   1788 
   1789 /*
   1790 * xsltTreeAcquireStoredNs:
   1791 * @doc: the doc
   1792 * @nsName: the namespace name
   1793 * @prefix: the prefix
   1794 *
   1795 * BIG NOTE:
   1796 *  This was copy&pasted from Libxml2's xmlDOMWrapStoreNs() in "tree.c".
   1797 * Creates or reuses an xmlNs struct on doc->oldNs with
   1798 * the given prefix and namespace name.
   1799 *
   1800 * Returns the aquired ns struct or NULL in case of an API
   1801 *         or internal error.
   1802 */
   1803 static xmlNsPtr
   1804 xsltTreeAcquireStoredNs(xmlDocPtr doc,
   1805 			const xmlChar *nsName,
   1806 			const xmlChar *prefix)
   1807 {
   1808     xmlNsPtr ns;
   1809 
   1810     if (doc == NULL)
   1811 	return (NULL);
   1812     if (doc->oldNs != NULL)
   1813 	ns = doc->oldNs;
   1814     else
   1815 	ns = xsltTreeEnsureXMLDecl(doc);
   1816     if (ns == NULL)
   1817 	return (NULL);
   1818     if (ns->next != NULL) {
   1819 	/* Reuse. */
   1820 	ns = ns->next;
   1821 	while (ns != NULL) {
   1822 	    if ((ns->prefix == NULL) != (prefix == NULL)) {
   1823 		/* NOP */
   1824 	    } else if (prefix == NULL) {
   1825 		if (xmlStrEqual(ns->href, nsName))
   1826 		    return (ns);
   1827 	    } else {
   1828 		if ((ns->prefix[0] == prefix[0]) &&
   1829 		     xmlStrEqual(ns->prefix, prefix) &&
   1830 		     xmlStrEqual(ns->href, nsName))
   1831 		    return (ns);
   1832 
   1833 	    }
   1834 	    if (ns->next == NULL)
   1835 		break;
   1836 	    ns = ns->next;
   1837 	}
   1838     }
   1839     /* Create. */
   1840     ns->next = xmlNewNs(NULL, nsName, prefix);
   1841     return (ns->next);
   1842 }
   1843 
   1844 /**
   1845  * xsltLREBuildEffectiveNs:
   1846  *
   1847  * Apply ns-aliasing on the namespace of the given @elem and
   1848  * its attributes.
   1849  */
   1850 static int
   1851 xsltLREBuildEffectiveNs(xsltCompilerCtxtPtr cctxt,
   1852 			xmlNodePtr elem)
   1853 {
   1854     xmlNsPtr ns;
   1855     xsltNsAliasPtr alias;
   1856 
   1857     if ((cctxt == NULL) || (elem == NULL))
   1858 	return(-1);
   1859     if ((cctxt->nsAliases == NULL) || (! cctxt->hasNsAliases))
   1860 	return(0);
   1861 
   1862     alias = cctxt->nsAliases;
   1863     while (alias != NULL) {
   1864 	if ( /* If both namespaces are NULL... */
   1865 	    ( (elem->ns == NULL) &&
   1866 	    ((alias->literalNs == NULL) ||
   1867 	    (alias->literalNs->href == NULL)) ) ||
   1868 	    /* ... or both namespace are equal */
   1869 	    ( (elem->ns != NULL) &&
   1870 	    (alias->literalNs != NULL) &&
   1871 	    xmlStrEqual(elem->ns->href, alias->literalNs->href) ) )
   1872 	{
   1873 	    if ((alias->targetNs != NULL) &&
   1874 		(alias->targetNs->href != NULL))
   1875 	    {
   1876 		/*
   1877 		* Convert namespace.
   1878 		*/
   1879 		if (elem->doc == alias->docOfTargetNs) {
   1880 		    /*
   1881 		    * This is the nice case: same docs.
   1882 		    * This will eventually assign a ns-decl which
   1883 		    * is shadowed, but this has no negative effect on
   1884 		    * the generation of the result tree.
   1885 		    */
   1886 		    elem->ns = alias->targetNs;
   1887 		} else {
   1888 		    /*
   1889 		    * This target xmlNs originates from a different
   1890 		    * stylesheet tree. Try to locate it in the
   1891 		    * in-scope namespaces.
   1892 		    * OPTIMIZE TODO: Use the compiler-node-info inScopeNs.
   1893 		    */
   1894 		    ns = xmlSearchNs(elem->doc, elem,
   1895 			alias->targetNs->prefix);
   1896 		    /*
   1897 		    * If no matching ns-decl found, then assign a
   1898 		    * ns-decl stored in xmlDoc.
   1899 		    */
   1900 		    if ((ns == NULL) ||
   1901 			(! xmlStrEqual(ns->href, alias->targetNs->href)))
   1902 		    {
   1903 			/*
   1904 			* BIG NOTE: The use of xsltTreeAcquireStoredNs()
   1905 			*  is not very efficient, but currently I don't
   1906 			*  see an other way of *safely* changing a node's
   1907 			*  namespace, since the xmlNs struct in
   1908 			*  alias->targetNs might come from an other
   1909 			*  stylesheet tree. So we need to anchor it in the
   1910 			*  current document, without adding it to the tree,
   1911 			*  which would otherwise change the in-scope-ns
   1912 			*  semantic of the tree.
   1913 			*/
   1914 			ns = xsltTreeAcquireStoredNs(elem->doc,
   1915 			    alias->targetNs->href,
   1916 			    alias->targetNs->prefix);
   1917 
   1918 			if (ns == NULL) {
   1919 			    xsltTransformError(NULL, cctxt->style, elem,
   1920 				"Internal error in "
   1921 				"xsltLREBuildEffectiveNs(): "
   1922 				"failed to acquire a stored "
   1923 				"ns-declaration.\n");
   1924 			    cctxt->style->errors++;
   1925 			    return(-1);
   1926 
   1927 			}
   1928 		    }
   1929 		    elem->ns = ns;
   1930 		}
   1931 	    } else {
   1932 		/*
   1933 		* Move into or leave in the NULL namespace.
   1934 		*/
   1935 		elem->ns = NULL;
   1936 	    }
   1937 	    break;
   1938 	}
   1939 	alias = alias->next;
   1940     }
   1941     /*
   1942     * Same with attributes of literal result elements.
   1943     */
   1944     if (elem->properties != NULL) {
   1945 	xmlAttrPtr attr = elem->properties;
   1946 
   1947 	while (attr != NULL) {
   1948 	    if (attr->ns == NULL) {
   1949 		attr = attr->next;
   1950 		continue;
   1951 	    }
   1952 	    alias = cctxt->nsAliases;
   1953 	    while (alias != NULL) {
   1954 		if ( /* If both namespaces are NULL... */
   1955 		    ( (elem->ns == NULL) &&
   1956 		    ((alias->literalNs == NULL) ||
   1957 		    (alias->literalNs->href == NULL)) ) ||
   1958 		    /* ... or both namespace are equal */
   1959 		    ( (elem->ns != NULL) &&
   1960 		    (alias->literalNs != NULL) &&
   1961 		    xmlStrEqual(elem->ns->href, alias->literalNs->href) ) )
   1962 		{
   1963 		    if ((alias->targetNs != NULL) &&
   1964 			(alias->targetNs->href != NULL))
   1965 		    {
   1966 			if (elem->doc == alias->docOfTargetNs) {
   1967 			    elem->ns = alias->targetNs;
   1968 			} else {
   1969 			    ns = xmlSearchNs(elem->doc, elem,
   1970 				alias->targetNs->prefix);
   1971 			    if ((ns == NULL) ||
   1972 				(! xmlStrEqual(ns->href, alias->targetNs->href)))
   1973 			    {
   1974 				ns = xsltTreeAcquireStoredNs(elem->doc,
   1975 				    alias->targetNs->href,
   1976 				    alias->targetNs->prefix);
   1977 
   1978 				if (ns == NULL) {
   1979 				    xsltTransformError(NULL, cctxt->style, elem,
   1980 					"Internal error in "
   1981 					"xsltLREBuildEffectiveNs(): "
   1982 					"failed to acquire a stored "
   1983 					"ns-declaration.\n");
   1984 				    cctxt->style->errors++;
   1985 				    return(-1);
   1986 
   1987 				}
   1988 			    }
   1989 			    elem->ns = ns;
   1990 			}
   1991 		    } else {
   1992 		    /*
   1993 		    * Move into or leave in the NULL namespace.
   1994 			*/
   1995 			elem->ns = NULL;
   1996 		    }
   1997 		    break;
   1998 		}
   1999 		alias = alias->next;
   2000 	    }
   2001 
   2002 	    attr = attr->next;
   2003 	}
   2004     }
   2005     return(0);
   2006 }
   2007 
   2008 /**
   2009  * xsltLREBuildEffectiveNsNodes:
   2010  *
   2011  * Computes the effective namespaces nodes for a literal result
   2012  * element.
   2013  * @effectiveNs is the set of effective ns-nodes
   2014  *  on the literal result element, which will be added to the result
   2015  *  element if not already existing in the result tree.
   2016  *  This means that excluded namespaces (via exclude-result-prefixes,
   2017  *  extension-element-prefixes and the XSLT namespace) not added
   2018  *  to the set.
   2019  *  Namespace-aliasing was applied on the @effectiveNs.
   2020  */
   2021 static int
   2022 xsltLREBuildEffectiveNsNodes(xsltCompilerCtxtPtr cctxt,
   2023 			     xsltStyleItemLRElementInfoPtr item,
   2024 			     xmlNodePtr elem,
   2025 			     int isLRE)
   2026 {
   2027     xmlNsPtr ns, tmpns;
   2028     xsltEffectiveNsPtr effNs, lastEffNs = NULL;
   2029     int i, j, holdByElem;
   2030     xsltPointerListPtr extElemNs = cctxt->inode->extElemNs;
   2031     xsltPointerListPtr exclResultNs = cctxt->inode->exclResultNs;
   2032 
   2033     if ((cctxt == NULL) || (cctxt->inode == NULL) || (elem == NULL) ||
   2034 	(item == NULL) || (item->effectiveNs != NULL))
   2035 	return(-1);
   2036 
   2037     if (item->inScopeNs == NULL)
   2038 	return(0);
   2039 
   2040     extElemNs = cctxt->inode->extElemNs;
   2041     exclResultNs = cctxt->inode->exclResultNs;
   2042 
   2043     for (i = 0; i < item->inScopeNs->totalNumber; i++) {
   2044 	ns = item->inScopeNs->list[i];
   2045 	/*
   2046 	* Skip namespaces designated as excluded namespaces
   2047 	* -------------------------------------------------
   2048 	*
   2049 	* XSLT-20 TODO: In XSLT 2.0 we need to keep namespaces
   2050 	*  which are target namespaces of namespace-aliases
   2051 	*  regardless if designated as excluded.
   2052 	*
   2053 	* Exclude the XSLT namespace.
   2054 	*/
   2055 	if (xmlStrEqual(ns->href, XSLT_NAMESPACE))
   2056 	    goto skip_ns;
   2057 
   2058 	/*
   2059 	* Apply namespace aliasing
   2060 	* ------------------------
   2061 	*
   2062 	* SPEC XSLT 2.0
   2063 	*  "- A namespace node whose string value is a literal namespace
   2064 	*     URI is not copied to the result tree.
   2065 	*   - A namespace node whose string value is a target namespace URI
   2066 	*     is copied to the result tree, whether or not the URI
   2067 	*     identifies an excluded namespace."
   2068 	*
   2069 	* NOTE: The ns-aliasing machanism is non-cascading.
   2070 	*  (checked with Saxon, Xalan and MSXML .NET).
   2071 	* URGENT TODO: is style->nsAliases the effective list of
   2072 	*  ns-aliases, or do we need to lookup the whole
   2073 	*  import-tree?
   2074 	* TODO: Get rid of import-tree lookup.
   2075 	*/
   2076 	if (cctxt->hasNsAliases) {
   2077 	    xsltNsAliasPtr alias;
   2078 	    /*
   2079 	    * First check for being a target namespace.
   2080 	    */
   2081 	    alias = cctxt->nsAliases;
   2082 	    do {
   2083 		/*
   2084 		* TODO: Is xmlns="" handled already?
   2085 		*/
   2086 		if ((alias->targetNs != NULL) &&
   2087 		    (xmlStrEqual(alias->targetNs->href, ns->href)))
   2088 		{
   2089 		    /*
   2090 		    * Recognized as a target namespace; use it regardless
   2091 		    * if excluded otherwise.
   2092 		    */
   2093 		    goto add_effective_ns;
   2094 		}
   2095 		alias = alias->next;
   2096 	    } while (alias != NULL);
   2097 
   2098 	    alias = cctxt->nsAliases;
   2099 	    do {
   2100 		/*
   2101 		* TODO: Is xmlns="" handled already?
   2102 		*/
   2103 		if ((alias->literalNs != NULL) &&
   2104 		    (xmlStrEqual(alias->literalNs->href, ns->href)))
   2105 		{
   2106 		    /*
   2107 		    * Recognized as an namespace alias; do not use it.
   2108 		    */
   2109 		    goto skip_ns;
   2110 		}
   2111 		alias = alias->next;
   2112 	    } while (alias != NULL);
   2113 	}
   2114 
   2115 	/*
   2116 	* Exclude excluded result namespaces.
   2117 	*/
   2118 	if (exclResultNs) {
   2119 	    for (j = 0; j < exclResultNs->number; j++)
   2120 		if (xmlStrEqual(ns->href, BAD_CAST exclResultNs->items[j]))
   2121 		    goto skip_ns;
   2122 	}
   2123 	/*
   2124 	* Exclude extension-element namespaces.
   2125 	*/
   2126 	if (extElemNs) {
   2127 	    for (j = 0; j < extElemNs->number; j++)
   2128 		if (xmlStrEqual(ns->href, BAD_CAST extElemNs->items[j]))
   2129 		    goto skip_ns;
   2130 	}
   2131 
   2132 add_effective_ns:
   2133 	/*
   2134 	* OPTIMIZE TODO: This information may not be needed.
   2135 	*/
   2136 	if (isLRE && (elem->nsDef != NULL)) {
   2137 	    holdByElem = 0;
   2138 	    tmpns = elem->nsDef;
   2139 	    do {
   2140 		if (tmpns == ns) {
   2141 		    holdByElem = 1;
   2142 		    break;
   2143 		}
   2144 		tmpns = tmpns->next;
   2145 	    } while (tmpns != NULL);
   2146 	} else
   2147 	    holdByElem = 0;
   2148 
   2149 
   2150 	/*
   2151 	* Add the effective namespace declaration.
   2152 	*/
   2153 	effNs = (xsltEffectiveNsPtr) xmlMalloc(sizeof(xsltEffectiveNs));
   2154 	if (effNs == NULL) {
   2155 	    xsltTransformError(NULL, cctxt->style, elem,
   2156 		"Internal error in xsltLREBuildEffectiveNs(): "
   2157 		"failed to allocate memory.\n");
   2158 	    cctxt->style->errors++;
   2159 	    return(-1);
   2160 	}
   2161 	if (cctxt->psData->effectiveNs == NULL) {
   2162 	    cctxt->psData->effectiveNs = effNs;
   2163 	    effNs->nextInStore = NULL;
   2164 	} else {
   2165 	    effNs->nextInStore = cctxt->psData->effectiveNs;
   2166 	    cctxt->psData->effectiveNs = effNs;
   2167 	}
   2168 
   2169 	effNs->next = NULL;
   2170 	effNs->prefix = ns->prefix;
   2171 	effNs->nsName = ns->href;
   2172 	effNs->holdByElem = holdByElem;
   2173 
   2174 	if (lastEffNs == NULL)
   2175 	    item->effectiveNs = effNs;
   2176 	else
   2177 	    lastEffNs->next = effNs;
   2178 	lastEffNs = effNs;
   2179 
   2180 skip_ns:
   2181 	{}
   2182     }
   2183     return(0);
   2184 }
   2185 
   2186 
   2187 /**
   2188  * xsltLREInfoCreate:
   2189  *
   2190  * @isLRE: indicates if the given @elem is a literal result element
   2191  *
   2192  * Creates a new info for a literal result element.
   2193  */
   2194 static int
   2195 xsltLREInfoCreate(xsltCompilerCtxtPtr cctxt,
   2196 		  xmlNodePtr elem,
   2197 		  int isLRE)
   2198 {
   2199     xsltStyleItemLRElementInfoPtr item;
   2200 
   2201     if ((cctxt == NULL) || (cctxt->inode == NULL))
   2202 	return(-1);
   2203 
   2204     item = (xsltStyleItemLRElementInfoPtr)
   2205 	xmlMalloc(sizeof(xsltStyleItemLRElementInfo));
   2206     if (item == NULL) {
   2207 	xsltTransformError(NULL, cctxt->style, NULL,
   2208 	    "Internal error in xsltLREInfoCreate(): "
   2209 	    "memory allocation failed.\n");
   2210 	cctxt->style->errors++;
   2211 	return(-1);
   2212     }
   2213     memset(item, 0, sizeof(xsltStyleItemLRElementInfo));
   2214     item->type = XSLT_FUNC_LITERAL_RESULT_ELEMENT;
   2215     /*
   2216     * Store it in the stylesheet.
   2217     */
   2218     item->next = cctxt->style->preComps;
   2219     cctxt->style->preComps = (xsltElemPreCompPtr) item;
   2220     /*
   2221     * @inScopeNs are used for execution of XPath expressions
   2222     *  in AVTs.
   2223     */
   2224     item->inScopeNs = cctxt->inode->inScopeNs;
   2225 
   2226     if (elem)
   2227 	xsltLREBuildEffectiveNsNodes(cctxt, item, elem, isLRE);
   2228 
   2229     cctxt->inode->litResElemInfo = item;
   2230     cctxt->inode->nsChanged = 0;
   2231     cctxt->maxLREs++;
   2232     return(0);
   2233 }
   2234 
   2235 /**
   2236  * xsltCompilerVarInfoPush:
   2237  * @cctxt: the compilation context
   2238  *
   2239  * Pushes a new var/param info onto the stack.
   2240  *
   2241  * Returns the acquired variable info.
   2242  */
   2243 static xsltVarInfoPtr
   2244 xsltCompilerVarInfoPush(xsltCompilerCtxtPtr cctxt,
   2245 				  xmlNodePtr inst,
   2246 				  const xmlChar *name,
   2247 				  const xmlChar *nsName)
   2248 {
   2249     xsltVarInfoPtr ivar;
   2250 
   2251     if ((cctxt->ivar != NULL) && (cctxt->ivar->next != NULL)) {
   2252 	ivar = cctxt->ivar->next;
   2253     } else if ((cctxt->ivar == NULL) && (cctxt->ivars != NULL)) {
   2254 	ivar = cctxt->ivars;
   2255     } else {
   2256 	ivar = (xsltVarInfoPtr) xmlMalloc(sizeof(xsltVarInfo));
   2257 	if (ivar == NULL) {
   2258 	    xsltTransformError(NULL, cctxt->style, inst,
   2259 		"xsltParseInScopeVarPush: xmlMalloc() failed!\n");
   2260 	    cctxt->style->errors++;
   2261 	    return(NULL);
   2262 	}
   2263 	/* memset(retVar, 0, sizeof(xsltInScopeVar)); */
   2264 	if (cctxt->ivars == NULL) {
   2265 	    cctxt->ivars = ivar;
   2266 	    ivar->prev = NULL;
   2267 	} else {
   2268 	    cctxt->ivar->next = ivar;
   2269 	    ivar->prev = cctxt->ivar;
   2270 	}
   2271 	cctxt->ivar = ivar;
   2272 	ivar->next = NULL;
   2273     }
   2274     ivar->depth = cctxt->depth;
   2275     ivar->name = name;
   2276     ivar->nsName = nsName;
   2277     return(ivar);
   2278 }
   2279 
   2280 /**
   2281  * xsltCompilerVarInfoPop:
   2282  * @cctxt: the compilation context
   2283  *
   2284  * Pops all var/param infos from the stack, which
   2285  * have the current depth.
   2286  */
   2287 static void
   2288 xsltCompilerVarInfoPop(xsltCompilerCtxtPtr cctxt)
   2289 {
   2290 
   2291     while ((cctxt->ivar != NULL) &&
   2292 	(cctxt->ivar->depth > cctxt->depth))
   2293     {
   2294 	cctxt->ivar = cctxt->ivar->prev;
   2295     }
   2296 }
   2297 
   2298 /*
   2299 * xsltCompilerNodePush:
   2300 *
   2301 * @cctxt: the compilation context
   2302 * @node: the node to be pushed (this can also be the doc-node)
   2303 *
   2304 *
   2305 *
   2306 * Returns the current node info structure or
   2307 *         NULL in case of an internal error.
   2308 */
   2309 static xsltCompilerNodeInfoPtr
   2310 xsltCompilerNodePush(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   2311 {
   2312     xsltCompilerNodeInfoPtr inode, iprev;
   2313 
   2314     if ((cctxt->inode != NULL) && (cctxt->inode->next != NULL)) {
   2315 	inode = cctxt->inode->next;
   2316     } else if ((cctxt->inode == NULL) && (cctxt->inodeList != NULL)) {
   2317 	inode = cctxt->inodeList;
   2318     } else {
   2319 	/*
   2320 	* Create a new node-info.
   2321 	*/
   2322 	inode = (xsltCompilerNodeInfoPtr)
   2323 	    xmlMalloc(sizeof(xsltCompilerNodeInfo));
   2324 	if (inode == NULL) {
   2325 	    xsltTransformError(NULL, cctxt->style, NULL,
   2326 		"xsltCompilerNodePush: malloc failed.\n");
   2327 	    return(NULL);
   2328 	}
   2329 	memset(inode, 0, sizeof(xsltCompilerNodeInfo));
   2330 	if (cctxt->inodeList == NULL)
   2331 	    cctxt->inodeList = inode;
   2332 	else {
   2333 	    cctxt->inodeLast->next = inode;
   2334 	    inode->prev = cctxt->inodeLast;
   2335 	}
   2336 	cctxt->inodeLast = inode;
   2337 	cctxt->maxNodeInfos++;
   2338 	if (cctxt->inode == NULL) {
   2339 	    cctxt->inode = inode;
   2340 	    /*
   2341 	    * Create an initial literal result element info for
   2342 	    * the root of the stylesheet.
   2343 	    */
   2344 	    xsltLREInfoCreate(cctxt, NULL, 0);
   2345 	}
   2346     }
   2347     cctxt->depth++;
   2348     cctxt->inode = inode;
   2349     /*
   2350     * REVISIT TODO: Keep the reset always complete.
   2351     * NOTE: Be carefull with the @node, since it might be
   2352     *  a doc-node.
   2353     */
   2354     inode->node = node;
   2355     inode->depth = cctxt->depth;
   2356     inode->templ = NULL;
   2357     inode->category = XSLT_ELEMENT_CATEGORY_XSLT;
   2358     inode->type = 0;
   2359     inode->item = NULL;
   2360     inode->curChildType = 0;
   2361     inode->extContentHandled = 0;
   2362     inode->isRoot = 0;
   2363 
   2364     if (inode->prev != NULL) {
   2365 	iprev = inode->prev;
   2366 	/*
   2367 	* Inherit the following information:
   2368 	* ---------------------------------
   2369 	*
   2370 	* In-scope namespaces
   2371 	*/
   2372 	inode->inScopeNs = iprev->inScopeNs;
   2373 	/*
   2374 	* Info for literal result elements
   2375 	*/
   2376 	inode->litResElemInfo = iprev->litResElemInfo;
   2377 	inode->nsChanged = iprev->nsChanged;
   2378 	/*
   2379 	* Excluded result namespaces
   2380 	*/
   2381 	inode->exclResultNs = iprev->exclResultNs;
   2382 	/*
   2383 	* Extension instruction namespaces
   2384 	*/
   2385 	inode->extElemNs = iprev->extElemNs;
   2386 	/*
   2387 	* Whitespace preservation
   2388 	*/
   2389 	inode->preserveWhitespace = iprev->preserveWhitespace;
   2390 	/*
   2391 	* Forwards-compatible mode
   2392 	*/
   2393 	inode->forwardsCompat = iprev->forwardsCompat;
   2394     } else {
   2395 	inode->inScopeNs = NULL;
   2396 	inode->exclResultNs = NULL;
   2397 	inode->extElemNs = NULL;
   2398 	inode->preserveWhitespace = 0;
   2399 	inode->forwardsCompat = 0;
   2400     }
   2401 
   2402     return(inode);
   2403 }
   2404 
   2405 /*
   2406 * xsltCompilerNodePop:
   2407 *
   2408 * @cctxt: the compilation context
   2409 * @node: the node to be pushed (this can also be the doc-node)
   2410 *
   2411 * Pops the current node info.
   2412 */
   2413 static void
   2414 xsltCompilerNodePop(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   2415 {
   2416     if (cctxt->inode == NULL) {
   2417 	xmlGenericError(xmlGenericErrorContext,
   2418 	    "xsltCompilerNodePop: Top-node mismatch.\n");
   2419 	return;
   2420     }
   2421     /*
   2422     * NOTE: Be carefull with the @node, since it might be
   2423     *  a doc-node.
   2424     */
   2425     if (cctxt->inode->node != node) {
   2426 	xmlGenericError(xmlGenericErrorContext,
   2427 	"xsltCompilerNodePop: Node mismatch.\n");
   2428 	goto mismatch;
   2429     }
   2430     if (cctxt->inode->depth != cctxt->depth) {
   2431 	xmlGenericError(xmlGenericErrorContext,
   2432 	"xsltCompilerNodePop: Depth mismatch.\n");
   2433 	goto mismatch;
   2434     }
   2435     /*
   2436     * Pop information of variables.
   2437     */
   2438     if ((cctxt->ivar) && (cctxt->ivar->depth > cctxt->depth))
   2439 	xsltCompilerVarInfoPop(cctxt);
   2440 
   2441     cctxt->depth--;
   2442     cctxt->inode = cctxt->inode->prev;
   2443     if (cctxt->inode != NULL)
   2444 	cctxt->inode->curChildType = 0;
   2445     return;
   2446 
   2447 mismatch:
   2448     {
   2449 	const xmlChar *nsName = NULL, *name = NULL;
   2450 	const xmlChar *infnsName = NULL, *infname = NULL;
   2451 
   2452 	if (node) {
   2453 	    if (node->type == XML_ELEMENT_NODE) {
   2454 		name = node->name;
   2455 		if (node->ns != NULL)
   2456 		    nsName = node->ns->href;
   2457 		else
   2458 		    nsName = BAD_CAST "";
   2459 	    } else {
   2460 		name = BAD_CAST "#document";
   2461 		nsName = BAD_CAST "";
   2462 	    }
   2463 	} else
   2464 	    name = BAD_CAST "Not given";
   2465 
   2466 	if (cctxt->inode->node) {
   2467 	    if (node->type == XML_ELEMENT_NODE) {
   2468 		infname = cctxt->inode->node->name;
   2469 		if (cctxt->inode->node->ns != NULL)
   2470 		    infnsName = cctxt->inode->node->ns->href;
   2471 		else
   2472 		    infnsName = BAD_CAST "";
   2473 	    } else {
   2474 		infname = BAD_CAST "#document";
   2475 		infnsName = BAD_CAST "";
   2476 	    }
   2477 	} else
   2478 	    infname = BAD_CAST "Not given";
   2479 
   2480 
   2481 	xmlGenericError(xmlGenericErrorContext,
   2482 	    "xsltCompilerNodePop: Given   : '%s' URI '%s'\n",
   2483 	    name, nsName);
   2484 	xmlGenericError(xmlGenericErrorContext,
   2485 	    "xsltCompilerNodePop: Expected: '%s' URI '%s'\n",
   2486 	    infname, infnsName);
   2487     }
   2488 }
   2489 
   2490 /*
   2491 * xsltCompilerBuildInScopeNsList:
   2492 *
   2493 * Create and store the list of in-scope namespaces for the given
   2494 * node in the stylesheet. If there are no changes in the in-scope
   2495 * namespaces then the last ns-info of the ancestor axis will be returned.
   2496 * Compilation-time only.
   2497 *
   2498 * Returns the ns-info or NULL if there are no namespaces in scope.
   2499 */
   2500 static xsltNsListContainerPtr
   2501 xsltCompilerBuildInScopeNsList(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   2502 {
   2503     xsltNsListContainerPtr nsi = NULL;
   2504     xmlNsPtr *list = NULL, ns;
   2505     int i, maxns = 5;
   2506     /*
   2507     * Create a new ns-list for this position in the node-tree.
   2508     * xmlGetNsList() will return NULL, if there are no ns-decls in the
   2509     * tree. Note that the ns-decl for the XML namespace is not added
   2510     * to the resulting list; the XPath module handles the XML namespace
   2511     * internally.
   2512     */
   2513     while (node != NULL) {
   2514         if (node->type == XML_ELEMENT_NODE) {
   2515             ns = node->nsDef;
   2516             while (ns != NULL) {
   2517                 if (nsi == NULL) {
   2518 		    nsi = (xsltNsListContainerPtr)
   2519 			xmlMalloc(sizeof(xsltNsListContainer));
   2520 		    if (nsi == NULL) {
   2521 			xsltTransformError(NULL, cctxt->style, NULL,
   2522 			    "xsltCompilerBuildInScopeNsList: "
   2523 			    "malloc failed!\n");
   2524 			goto internal_err;
   2525 		    }
   2526 		    memset(nsi, 0, sizeof(xsltNsListContainer));
   2527                     nsi->list =
   2528                         (xmlNsPtr *) xmlMalloc(maxns * sizeof(xmlNsPtr));
   2529                     if (nsi->list == NULL) {
   2530 			xsltTransformError(NULL, cctxt->style, NULL,
   2531 			    "xsltCompilerBuildInScopeNsList: "
   2532 			    "malloc failed!\n");
   2533 			goto internal_err;
   2534                     }
   2535                     nsi->list[0] = NULL;
   2536                 }
   2537 		/*
   2538 		* Skip shadowed namespace bindings.
   2539 		*/
   2540                 for (i = 0; i < nsi->totalNumber; i++) {
   2541                     if ((ns->prefix == nsi->list[i]->prefix) ||
   2542                         (xmlStrEqual(ns->prefix, nsi->list[i]->prefix)))
   2543 		    break;
   2544                 }
   2545                 if (i >= nsi->totalNumber) {
   2546                     if (nsi->totalNumber +1 >= maxns) {
   2547                         maxns *= 2;
   2548 			nsi->list =
   2549 			    (xmlNsPtr *) xmlRealloc(nsi->list,
   2550 				maxns * sizeof(xmlNsPtr));
   2551                         if (nsi->list == NULL) {
   2552                             xsltTransformError(NULL, cctxt->style, NULL,
   2553 				"xsltCompilerBuildInScopeNsList: "
   2554 				"realloc failed!\n");
   2555 				goto internal_err;
   2556                         }
   2557                     }
   2558                     nsi->list[nsi->totalNumber++] = ns;
   2559                     nsi->list[nsi->totalNumber] = NULL;
   2560                 }
   2561 
   2562                 ns = ns->next;
   2563             }
   2564         }
   2565         node = node->parent;
   2566     }
   2567     if (nsi == NULL)
   2568 	return(NULL);
   2569     /*
   2570     * Move the default namespace to last position.
   2571     */
   2572     nsi->xpathNumber = nsi->totalNumber;
   2573     for (i = 0; i < nsi->totalNumber; i++) {
   2574 	if (nsi->list[i]->prefix == NULL) {
   2575 	    ns = nsi->list[i];
   2576 	    nsi->list[i] = nsi->list[nsi->totalNumber-1];
   2577 	    nsi->list[nsi->totalNumber-1] = ns;
   2578 	    nsi->xpathNumber--;
   2579 	    break;
   2580 	}
   2581     }
   2582     /*
   2583     * Store the ns-list in the stylesheet.
   2584     */
   2585     if (xsltPointerListAddSize(
   2586 	(xsltPointerListPtr)cctxt->psData->inScopeNamespaces,
   2587 	(void *) nsi, 5) == -1)
   2588     {
   2589 	xmlFree(nsi);
   2590 	nsi = NULL;
   2591 	xsltTransformError(NULL, cctxt->style, NULL,
   2592 	    "xsltCompilerBuildInScopeNsList: failed to add ns-info.\n");
   2593 	goto internal_err;
   2594     }
   2595     /*
   2596     * Notify of change in status wrt namespaces.
   2597     */
   2598     if (cctxt->inode != NULL)
   2599 	cctxt->inode->nsChanged = 1;
   2600 
   2601     return(nsi);
   2602 
   2603 internal_err:
   2604     if (list != NULL)
   2605 	xmlFree(list);
   2606     cctxt->style->errors++;
   2607     return(NULL);
   2608 }
   2609 
   2610 static int
   2611 xsltParseNsPrefixList(xsltCompilerCtxtPtr cctxt,
   2612 		      xsltPointerListPtr list,
   2613 		      xmlNodePtr node,
   2614 		      const xmlChar *value)
   2615 {
   2616     xmlChar *cur, *end;
   2617     xmlNsPtr ns;
   2618 
   2619     if ((cctxt == NULL) || (value == NULL) || (list == NULL))
   2620 	return(-1);
   2621 
   2622     list->number = 0;
   2623 
   2624     cur = (xmlChar *) value;
   2625     while (*cur != 0) {
   2626 	while (IS_BLANK(*cur)) cur++;
   2627 	if (*cur == 0)
   2628 	    break;
   2629 	end = cur;
   2630 	while ((*end != 0) && (!IS_BLANK(*end))) end++;
   2631 	cur = xmlStrndup(cur, end - cur);
   2632 	if (cur == NULL) {
   2633 	    cur = end;
   2634 	    continue;
   2635 	}
   2636 	/*
   2637 	* TODO: Export and use xmlSearchNsByPrefixStrict()
   2638 	*   in Libxml2, tree.c, since xmlSearchNs() is in most
   2639 	*   cases not efficient and in some cases not correct.
   2640 	*
   2641 	* XSLT-2 TODO: XSLT 2.0 allows an additional "#all" value.
   2642 	*/
   2643 	if ((cur[0] == '#') &&
   2644 	    xmlStrEqual(cur, (const xmlChar *)"#default"))
   2645 	    ns = xmlSearchNs(cctxt->style->doc, node, NULL);
   2646 	else
   2647 	    ns = xmlSearchNs(cctxt->style->doc, node, cur);
   2648 
   2649 	if (ns == NULL) {
   2650 	    /*
   2651 	    * TODO: Better to report the attr-node, otherwise
   2652 	    *  the user won't know which attribute was invalid.
   2653 	    */
   2654 	    xsltTransformError(NULL, cctxt->style, node,
   2655 		"No namespace binding in scope for prefix '%s'.\n", cur);
   2656 	    /*
   2657 	    * XSLT-1.0: "It is an error if there is no namespace
   2658 	    *  bound to the prefix on the element bearing the
   2659 	    *  exclude-result-prefixes or xsl:exclude-result-prefixes
   2660 	    *  attribute."
   2661 	    */
   2662 	    cctxt->style->errors++;
   2663 	} else {
   2664 #ifdef WITH_XSLT_DEBUG_PARSING
   2665 	    xsltGenericDebug(xsltGenericDebugContext,
   2666 		"resolved prefix '%s'\n", cur);
   2667 #endif
   2668 	    /*
   2669 	    * Note that we put the namespace name into the dict.
   2670 	    */
   2671 	    if (xsltPointerListAddSize(list,
   2672 		(void *) xmlDictLookup(cctxt->style->dict,
   2673 		ns->href, -1), 5) == -1)
   2674 	    {
   2675 		xmlFree(cur);
   2676 		goto internal_err;
   2677 	    }
   2678 	}
   2679 	xmlFree(cur);
   2680 
   2681 	cur = end;
   2682     }
   2683     return(0);
   2684 
   2685 internal_err:
   2686     cctxt->style->errors++;
   2687     return(-1);
   2688 }
   2689 
   2690 /**
   2691  * xsltCompilerUtilsCreateMergedList:
   2692  * @dest: the destination list (optional)
   2693  * @first: the first list
   2694  * @second: the second list (optional)
   2695  *
   2696  * Appends the content of @second to @first into @destination.
   2697  * If @destination is NULL a new list will be created.
   2698  *
   2699  * Returns the merged list of items or NULL if there's nothing to merge.
   2700  */
   2701 static xsltPointerListPtr
   2702 xsltCompilerUtilsCreateMergedList(xsltPointerListPtr first,
   2703 			    xsltPointerListPtr second)
   2704 {
   2705     xsltPointerListPtr ret;
   2706     size_t num;
   2707 
   2708     if (first)
   2709 	num = first->number;
   2710     else
   2711 	num = 0;
   2712     if (second)
   2713 	num += second->number;
   2714     if (num == 0)
   2715 	return(NULL);
   2716     ret = xsltPointerListCreate(num);
   2717     if (ret == NULL)
   2718 	return(NULL);
   2719     /*
   2720     * Copy contents.
   2721     */
   2722     if ((first != NULL) &&  (first->number != 0)) {
   2723 	memcpy(ret->items, first->items,
   2724 	    first->number * sizeof(void *));
   2725 	if ((second != NULL) && (second->number != 0))
   2726 	    memcpy(ret->items + first->number, second->items,
   2727 		second->number * sizeof(void *));
   2728     } else if ((second != NULL) && (second->number != 0))
   2729 	memcpy(ret->items, (void *) second->items,
   2730 	    second->number * sizeof(void *));
   2731     ret->number = num;
   2732     return(ret);
   2733 }
   2734 
   2735 /*
   2736 * xsltParseExclResultPrefixes:
   2737 *
   2738 * Create and store the list of in-scope namespaces for the given
   2739 * node in the stylesheet. If there are no changes in the in-scope
   2740 * namespaces then the last ns-info of the ancestor axis will be returned.
   2741 * Compilation-time only.
   2742 *
   2743 * Returns the ns-info or NULL if there are no namespaces in scope.
   2744 */
   2745 static xsltPointerListPtr
   2746 xsltParseExclResultPrefixes(xsltCompilerCtxtPtr cctxt, xmlNodePtr node,
   2747 			    xsltPointerListPtr def,
   2748 			    int instrCategory)
   2749 {
   2750     xsltPointerListPtr list = NULL;
   2751     xmlChar *value;
   2752     xmlAttrPtr attr;
   2753 
   2754     if ((cctxt == NULL) || (node == NULL))
   2755 	return(NULL);
   2756 
   2757     if (instrCategory == XSLT_ELEMENT_CATEGORY_XSLT)
   2758 	attr = xmlHasNsProp(node, BAD_CAST "exclude-result-prefixes", NULL);
   2759     else
   2760 	attr = xmlHasNsProp(node, BAD_CAST "exclude-result-prefixes",
   2761 	    XSLT_NAMESPACE);
   2762     if (attr == NULL)
   2763 	return(def);
   2764 
   2765     if (attr && (instrCategory == XSLT_ELEMENT_CATEGORY_LRE)) {
   2766 	/*
   2767 	* Mark the XSLT attr.
   2768 	*/
   2769 	attr->psvi = (void *) xsltXSLTAttrMarker;
   2770     }
   2771 
   2772     if ((attr->children != NULL) &&
   2773 	(attr->children->content != NULL))
   2774 	value = attr->children->content;
   2775     else {
   2776 	xsltTransformError(NULL, cctxt->style, node,
   2777 	    "Attribute 'exclude-result-prefixes': Invalid value.\n");
   2778 	cctxt->style->errors++;
   2779 	return(def);
   2780     }
   2781 
   2782     if (xsltParseNsPrefixList(cctxt, cctxt->tmpList, node,
   2783 	BAD_CAST value) != 0)
   2784 	goto exit;
   2785     if (cctxt->tmpList->number == 0)
   2786 	goto exit;
   2787     /*
   2788     * Merge the list with the inherited list.
   2789     */
   2790     list = xsltCompilerUtilsCreateMergedList(def, cctxt->tmpList);
   2791     if (list == NULL)
   2792 	goto exit;
   2793     /*
   2794     * Store the list in the stylesheet/compiler context.
   2795     */
   2796     if (xsltPointerListAddSize(
   2797 	cctxt->psData->exclResultNamespaces, list, 5) == -1)
   2798     {
   2799 	xsltPointerListFree(list);
   2800 	list = NULL;
   2801 	goto exit;
   2802     }
   2803     /*
   2804     * Notify of change in status wrt namespaces.
   2805     */
   2806     if (cctxt->inode != NULL)
   2807 	cctxt->inode->nsChanged = 1;
   2808 
   2809 exit:
   2810     if (list != NULL)
   2811 	return(list);
   2812     else
   2813 	return(def);
   2814 }
   2815 
   2816 /*
   2817 * xsltParseExtElemPrefixes:
   2818 *
   2819 * Create and store the list of in-scope namespaces for the given
   2820 * node in the stylesheet. If there are no changes in the in-scope
   2821 * namespaces then the last ns-info of the ancestor axis will be returned.
   2822 * Compilation-time only.
   2823 *
   2824 * Returns the ns-info or NULL if there are no namespaces in scope.
   2825 */
   2826 static xsltPointerListPtr
   2827 xsltParseExtElemPrefixes(xsltCompilerCtxtPtr cctxt, xmlNodePtr node,
   2828 			 xsltPointerListPtr def,
   2829 			 int instrCategory)
   2830 {
   2831     xsltPointerListPtr list = NULL;
   2832     xmlAttrPtr attr;
   2833     xmlChar *value;
   2834     int i;
   2835 
   2836     if ((cctxt == NULL) || (node == NULL))
   2837 	return(NULL);
   2838 
   2839     if (instrCategory == XSLT_ELEMENT_CATEGORY_XSLT)
   2840 	attr = xmlHasNsProp(node, BAD_CAST "extension-element-prefixes", NULL);
   2841     else
   2842 	attr = xmlHasNsProp(node, BAD_CAST "extension-element-prefixes",
   2843 	    XSLT_NAMESPACE);
   2844     if (attr == NULL)
   2845 	return(def);
   2846 
   2847     if (attr && (instrCategory == XSLT_ELEMENT_CATEGORY_LRE)) {
   2848 	/*
   2849 	* Mark the XSLT attr.
   2850 	*/
   2851 	attr->psvi = (void *) xsltXSLTAttrMarker;
   2852     }
   2853 
   2854     if ((attr->children != NULL) &&
   2855 	(attr->children->content != NULL))
   2856 	value = attr->children->content;
   2857     else {
   2858 	xsltTransformError(NULL, cctxt->style, node,
   2859 	    "Attribute 'extension-element-prefixes': Invalid value.\n");
   2860 	cctxt->style->errors++;
   2861 	return(def);
   2862     }
   2863 
   2864 
   2865     if (xsltParseNsPrefixList(cctxt, cctxt->tmpList, node,
   2866 	BAD_CAST value) != 0)
   2867 	goto exit;
   2868 
   2869     if (cctxt->tmpList->number == 0)
   2870 	goto exit;
   2871     /*
   2872     * REVISIT: Register the extension namespaces.
   2873     */
   2874     for (i = 0; i < cctxt->tmpList->number; i++)
   2875 	xsltRegisterExtPrefix(cctxt->style, NULL,
   2876 	BAD_CAST cctxt->tmpList->items[i]);
   2877     /*
   2878     * Merge the list with the inherited list.
   2879     */
   2880     list = xsltCompilerUtilsCreateMergedList(def, cctxt->tmpList);
   2881     if (list == NULL)
   2882 	goto exit;
   2883     /*
   2884     * Store the list in the stylesheet.
   2885     */
   2886     if (xsltPointerListAddSize(
   2887 	cctxt->psData->extElemNamespaces, list, 5) == -1)
   2888     {
   2889 	xsltPointerListFree(list);
   2890 	list = NULL;
   2891 	goto exit;
   2892     }
   2893     /*
   2894     * Notify of change in status wrt namespaces.
   2895     */
   2896     if (cctxt->inode != NULL)
   2897 	cctxt->inode->nsChanged = 1;
   2898 
   2899 exit:
   2900     if (list != NULL)
   2901 	return(list);
   2902     else
   2903 	return(def);
   2904 }
   2905 
   2906 /*
   2907 * xsltParseAttrXSLTVersion:
   2908 *
   2909 * @cctxt: the compilation context
   2910 * @node: the element-node
   2911 * @isXsltElem: whether this is an XSLT element
   2912 *
   2913 * Parses the attribute xsl:version.
   2914 *
   2915 * Returns 1 if there was such an attribute, 0 if not and
   2916 *         -1 if an internal or API error occured.
   2917 */
   2918 static int
   2919 xsltParseAttrXSLTVersion(xsltCompilerCtxtPtr cctxt, xmlNodePtr node,
   2920 			 int instrCategory)
   2921 {
   2922     xmlChar *value;
   2923     xmlAttrPtr attr;
   2924 
   2925     if ((cctxt == NULL) || (node == NULL))
   2926 	return(-1);
   2927 
   2928     if (instrCategory == XSLT_ELEMENT_CATEGORY_XSLT)
   2929 	attr = xmlHasNsProp(node, BAD_CAST "version", NULL);
   2930     else
   2931 	attr = xmlHasNsProp(node, BAD_CAST "version", XSLT_NAMESPACE);
   2932 
   2933     if (attr == NULL)
   2934 	return(0);
   2935 
   2936     attr->psvi = (void *) xsltXSLTAttrMarker;
   2937 
   2938     if ((attr->children != NULL) &&
   2939 	(attr->children->content != NULL))
   2940 	value = attr->children->content;
   2941     else {
   2942 	xsltTransformError(NULL, cctxt->style, node,
   2943 	    "Attribute 'version': Invalid value.\n");
   2944 	cctxt->style->errors++;
   2945 	return(1);
   2946     }
   2947 
   2948     if (! xmlStrEqual(value, (const xmlChar *)"1.0")) {
   2949 	cctxt->inode->forwardsCompat = 1;
   2950 	/*
   2951 	* TODO: To what extent do we support the
   2952 	*  forwards-compatible mode?
   2953 	*/
   2954 	/*
   2955 	* Report this only once per compilation episode.
   2956 	*/
   2957 	if (! cctxt->hasForwardsCompat) {
   2958 	    cctxt->hasForwardsCompat = 1;
   2959 	    cctxt->errSeverity = XSLT_ERROR_SEVERITY_WARNING;
   2960 	    xsltTransformError(NULL, cctxt->style, node,
   2961 		"Warning: the attribute xsl:version specifies a value "
   2962 		"different from '1.0'. Switching to forwards-compatible "
   2963 		"mode. Only features of XSLT 1.0 are supported by this "
   2964 		"processor.\n");
   2965 	    cctxt->style->warnings++;
   2966 	    cctxt->errSeverity = XSLT_ERROR_SEVERITY_ERROR;
   2967 	}
   2968     } else {
   2969 	cctxt->inode->forwardsCompat = 0;
   2970     }
   2971 
   2972     if (attr && (instrCategory == XSLT_ELEMENT_CATEGORY_LRE)) {
   2973 	/*
   2974 	* Set a marker on XSLT attributes.
   2975 	*/
   2976 	attr->psvi = (void *) xsltXSLTAttrMarker;
   2977     }
   2978     return(1);
   2979 }
   2980 
   2981 static int
   2982 xsltParsePreprocessStylesheetTree(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   2983 {
   2984     xmlNodePtr deleteNode, cur, txt, textNode = NULL;
   2985     xmlDocPtr doc;
   2986     xsltStylesheetPtr style;
   2987     int internalize = 0, findSpaceAttr;
   2988     int xsltStylesheetElemDepth;
   2989     xmlAttrPtr attr;
   2990     xmlChar *value;
   2991     const xmlChar *name, *nsNameXSLT = NULL;
   2992     int strictWhitespace, inXSLText = 0;
   2993 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
   2994     xsltNsMapPtr nsMapItem;
   2995 #endif
   2996 
   2997     if ((cctxt == NULL) || (cctxt->style == NULL) ||
   2998 	(node == NULL) || (node->type != XML_ELEMENT_NODE))
   2999         return(-1);
   3000 
   3001     doc = node->doc;
   3002     if (doc == NULL)
   3003 	goto internal_err;
   3004 
   3005     style = cctxt->style;
   3006     if ((style->dict != NULL) && (doc->dict == style->dict))
   3007 	internalize = 1;
   3008     else
   3009         style->internalized = 0;
   3010 
   3011     /*
   3012     * Init value of xml:space. Since this might be an embedded
   3013     * stylesheet, this is needed to be performed on the element
   3014     * where the stylesheet is rooted at, taking xml:space of
   3015     * ancestors into account.
   3016     */
   3017     if (! cctxt->simplified)
   3018 	xsltStylesheetElemDepth = cctxt->depth +1;
   3019     else
   3020 	xsltStylesheetElemDepth = 0;
   3021 
   3022     if (xmlNodeGetSpacePreserve(node) != 1)
   3023 	cctxt->inode->preserveWhitespace = 0;
   3024     else
   3025 	cctxt->inode->preserveWhitespace = 1;
   3026 
   3027     /*
   3028     * Eval if we should keep the old incorrect behaviour.
   3029     */
   3030     strictWhitespace = (cctxt->strict != 0) ? 1 : 0;
   3031 
   3032     nsNameXSLT = xsltConstNamespaceNameXSLT;
   3033 
   3034     deleteNode = NULL;
   3035     cur = node;
   3036     while (cur != NULL) {
   3037 	if (deleteNode != NULL)	{
   3038 
   3039 #ifdef WITH_XSLT_DEBUG_BLANKS
   3040 	    xsltGenericDebug(xsltGenericDebugContext,
   3041 	     "xsltParsePreprocessStylesheetTree: removing node\n");
   3042 #endif
   3043 	    xmlUnlinkNode(deleteNode);
   3044 	    xmlFreeNode(deleteNode);
   3045 	    deleteNode = NULL;
   3046 	}
   3047 	if (cur->type == XML_ELEMENT_NODE) {
   3048 
   3049 	    /*
   3050 	    * Clear the PSVI field.
   3051 	    */
   3052 	    cur->psvi = NULL;
   3053 
   3054 	    xsltCompilerNodePush(cctxt, cur);
   3055 
   3056 	    inXSLText = 0;
   3057 	    textNode = NULL;
   3058 	    findSpaceAttr = 1;
   3059 	    cctxt->inode->stripWhitespace = 0;
   3060 	    /*
   3061 	    * TODO: I'd love to use a string pointer comparison here :-/
   3062 	    */
   3063 	    if (IS_XSLT_ELEM(cur)) {
   3064 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
   3065 		if (cur->ns->href != nsNameXSLT) {
   3066 		    nsMapItem = xsltNewNamespaceMapItem(cctxt,
   3067 			doc, cur->ns, cur);
   3068 		    if (nsMapItem == NULL)
   3069 			goto internal_err;
   3070 		    cur->ns->href = nsNameXSLT;
   3071 		}
   3072 #endif
   3073 
   3074 		if (cur->name == NULL)
   3075 		    goto process_attributes;
   3076 		/*
   3077 		* Mark the XSLT element for later recognition.
   3078 		* TODO: Using the marker is still too dangerous, since if
   3079 		*   the parsing mechanism leaves out an XSLT element, then
   3080 		*   this might hit the transformation-mechanism, which
   3081 		*   will break if it doesn't expect such a marker.
   3082 		*/
   3083 		/* cur->psvi = (void *) xsltXSLTElemMarker; */
   3084 
   3085 		/*
   3086 		* XSLT 2.0: "Any whitespace text node whose parent is
   3087 		* one of the following elements is removed from the "
   3088 		* tree, regardless of any xml:space attributes:..."
   3089 		* xsl:apply-imports,
   3090 		* xsl:apply-templates,
   3091 		* xsl:attribute-set,
   3092 		* xsl:call-template,
   3093 		* xsl:choose,
   3094 		* xsl:stylesheet, xsl:transform.
   3095 		* XSLT 2.0: xsl:analyze-string,
   3096 		*           xsl:character-map,
   3097 		*           xsl:next-match
   3098 		*
   3099 		* TODO: I'd love to use a string pointer comparison here :-/
   3100 		*/
   3101 		name = cur->name;
   3102 		switch (*name) {
   3103 		    case 't':
   3104 			if ((name[0] == 't') && (name[1] == 'e') &&
   3105 			    (name[2] == 'x') && (name[3] == 't') &&
   3106 			    (name[4] == 0))
   3107 			{
   3108 			    /*
   3109 			    * Process the xsl:text element.
   3110 			    * ----------------------------
   3111 			    * Mark it for later recognition.
   3112 			    */
   3113 			    cur->psvi = (void *) xsltXSLTTextMarker;
   3114 			    /*
   3115 			    * For stylesheets, the set of
   3116 			    * whitespace-preserving element names
   3117 			    * consists of just xsl:text.
   3118 			    */
   3119 			    findSpaceAttr = 0;
   3120 			    cctxt->inode->preserveWhitespace = 1;
   3121 			    inXSLText = 1;
   3122 			}
   3123 			break;
   3124 		    case 'c':
   3125 			if (xmlStrEqual(name, BAD_CAST "choose") ||
   3126 			    xmlStrEqual(name, BAD_CAST "call-template"))
   3127 			    cctxt->inode->stripWhitespace = 1;
   3128 			break;
   3129 		    case 'a':
   3130 			if (xmlStrEqual(name, BAD_CAST "apply-templates") ||
   3131 			    xmlStrEqual(name, BAD_CAST "apply-imports") ||
   3132 			    xmlStrEqual(name, BAD_CAST "attribute-set"))
   3133 
   3134 			    cctxt->inode->stripWhitespace = 1;
   3135 			break;
   3136 		    default:
   3137 			if (xsltStylesheetElemDepth == cctxt->depth) {
   3138 			    /*
   3139 			    * This is a xsl:stylesheet/xsl:transform.
   3140 			    */
   3141 			    cctxt->inode->stripWhitespace = 1;
   3142 			    break;
   3143 			}
   3144 
   3145 			if ((cur->prev != NULL) &&
   3146 			    (cur->prev->type == XML_TEXT_NODE))
   3147 			{
   3148 			    /*
   3149 			    * XSLT 2.0 : "Any whitespace text node whose
   3150 			    *  following-sibling node is an xsl:param or
   3151 			    *  xsl:sort element is removed from the tree,
   3152 			    *  regardless of any xml:space attributes."
   3153 			    */
   3154 			    if (((*name == 'p') || (*name == 's')) &&
   3155 				(xmlStrEqual(name, BAD_CAST "param") ||
   3156 				 xmlStrEqual(name, BAD_CAST "sort")))
   3157 			    {
   3158 				do {
   3159 				    if (IS_BLANK_NODE(cur->prev)) {
   3160 					txt = cur->prev;
   3161 					xmlUnlinkNode(txt);
   3162 					xmlFreeNode(txt);
   3163 				    } else {
   3164 					/*
   3165 					* This will result in a content
   3166 					* error, when hitting the parsing
   3167 					* functions.
   3168 					*/
   3169 					break;
   3170 				    }
   3171 				} while (cur->prev);
   3172 			    }
   3173 			}
   3174 			break;
   3175 		}
   3176 	    }
   3177 
   3178 process_attributes:
   3179 	    /*
   3180 	    * Process attributes.
   3181 	    * ------------------
   3182 	    */
   3183 	    if (cur->properties != NULL) {
   3184 		if (cur->children == NULL)
   3185 		    findSpaceAttr = 0;
   3186 		attr = cur->properties;
   3187 		do {
   3188 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
   3189 		    if ((attr->ns) && (attr->ns->href != nsNameXSLT) &&
   3190 			xmlStrEqual(attr->ns->href, nsNameXSLT))
   3191 		    {
   3192 			nsMapItem = xsltNewNamespaceMapItem(cctxt,
   3193 			    doc, attr->ns, cur);
   3194 			if (nsMapItem == NULL)
   3195 			    goto internal_err;
   3196 			attr->ns->href = nsNameXSLT;
   3197 		    }
   3198 #endif
   3199 		    if (internalize) {
   3200 			/*
   3201 			* Internalize the attribute's value; the goal is to
   3202 			* speed up operations and minimize used space by
   3203 			* compiled stylesheets.
   3204 			*/
   3205 			txt = attr->children;
   3206 			/*
   3207 			* NOTE that this assumes only one
   3208 			*  text-node in the attribute's content.
   3209 			*/
   3210 			if ((txt != NULL) && (txt->content != NULL) &&
   3211 			    (!xmlDictOwns(style->dict, txt->content)))
   3212 			{
   3213 			    value = (xmlChar *) xmlDictLookup(style->dict,
   3214 				txt->content, -1);
   3215 			    xmlNodeSetContent(txt, NULL);
   3216 			    txt->content = value;
   3217 			}
   3218 		    }
   3219 		    /*
   3220 		    * Process xml:space attributes.
   3221 		    * ----------------------------
   3222 		    */
   3223 		    if ((findSpaceAttr != 0) &&
   3224 			(attr->ns != NULL) &&
   3225 			(attr->name != NULL) &&
   3226 			(attr->name[0] == 's') &&
   3227 			(attr->ns->prefix != NULL) &&
   3228 			(attr->ns->prefix[0] == 'x') &&
   3229 			(attr->ns->prefix[1] == 'm') &&
   3230 			(attr->ns->prefix[2] == 'l') &&
   3231 			(attr->ns->prefix[3] == 0))
   3232 		    {
   3233 			value = xmlGetNsProp(cur, BAD_CAST "space",
   3234 			    XML_XML_NAMESPACE);
   3235 			if (value != NULL) {
   3236 			    if (xmlStrEqual(value, BAD_CAST "preserve")) {
   3237 				cctxt->inode->preserveWhitespace = 1;
   3238 			    } else if (xmlStrEqual(value, BAD_CAST "default")) {
   3239 				cctxt->inode->preserveWhitespace = 0;
   3240 			    } else {
   3241 				/* Invalid value for xml:space. */
   3242 				xsltTransformError(NULL, style, cur,
   3243 				    "Attribute xml:space: Invalid value.\n");
   3244 				cctxt->style->warnings++;
   3245 			    }
   3246 			    findSpaceAttr = 0;
   3247 			    xmlFree(value);
   3248 			}
   3249 
   3250 		    }
   3251 		    attr = attr->next;
   3252 		} while (attr != NULL);
   3253 	    }
   3254 	    /*
   3255 	    * We'll descend into the children of element nodes only.
   3256 	    */
   3257 	    if (cur->children != NULL) {
   3258 		cur = cur->children;
   3259 		continue;
   3260 	    }
   3261 	} else if ((cur->type == XML_TEXT_NODE) ||
   3262 		(cur->type == XML_CDATA_SECTION_NODE))
   3263 	{
   3264 	    /*
   3265 	    * Merge adjacent text/CDATA-section-nodes
   3266 	    * ---------------------------------------
   3267 	    * In order to avoid breaking of existing stylesheets,
   3268 	    * if the old behaviour is wanted (strictWhitespace == 0),
   3269 	    * then we *won't* merge adjacent text-nodes
   3270 	    * (except in xsl:text); this will ensure that whitespace-only
   3271 	    * text nodes are (incorrectly) not stripped in some cases.
   3272 	    *
   3273 	    * Example:               : <foo>  <!-- bar -->zoo</foo>
   3274 	    * Corrent (strict) result: <foo>  zoo</foo>
   3275 	    * Incorrect (old) result : <foo>zoo</foo>
   3276 	    *
   3277 	    * NOTE that we *will* merge adjacent text-nodes if
   3278 	    * they are in xsl:text.
   3279 	    * Example, the following:
   3280 	    * <xsl:text>  <!-- bar -->zoo<xsl:text>
   3281 	    * will result in both cases in:
   3282 	    * <xsl:text>  zoo<xsl:text>
   3283 	    */
   3284 	    cur->type = XML_TEXT_NODE;
   3285 	    if ((strictWhitespace != 0) || (inXSLText != 0)) {
   3286 		/*
   3287 		* New behaviour; merge nodes.
   3288 		*/
   3289 		if (textNode == NULL)
   3290 		    textNode = cur;
   3291 		else {
   3292 		    if (cur->content != NULL)
   3293 			xmlNodeAddContent(textNode, cur->content);
   3294 		    deleteNode = cur;
   3295 		}
   3296 		if ((cur->next == NULL) ||
   3297 		    (cur->next->type == XML_ELEMENT_NODE))
   3298 		    goto end_of_text;
   3299 		else
   3300 		    goto next_sibling;
   3301 	    } else {
   3302 		/*
   3303 		* Old behaviour.
   3304 		*/
   3305 		if (textNode == NULL)
   3306 		    textNode = cur;
   3307 		goto end_of_text;
   3308 	    }
   3309 	} else if ((cur->type == XML_COMMENT_NODE) ||
   3310 	    (cur->type == XML_PI_NODE))
   3311 	{
   3312 	    /*
   3313 	    * Remove processing instructions and comments.
   3314 	    */
   3315 	    deleteNode = cur;
   3316 	    if ((cur->next == NULL) ||
   3317 		(cur->next->type == XML_ELEMENT_NODE))
   3318 		goto end_of_text;
   3319 	    else
   3320 		goto next_sibling;
   3321 	} else {
   3322 	    textNode = NULL;
   3323 	    /*
   3324 	    * Invalid node-type for this data-model.
   3325 	    */
   3326 	    xsltTransformError(NULL, style, cur,
   3327 		"Invalid type of node for the XSLT data model.\n");
   3328 	    cctxt->style->errors++;
   3329 	    goto next_sibling;
   3330 	}
   3331 
   3332 end_of_text:
   3333 	if (textNode) {
   3334 	    value = textNode->content;
   3335 	    /*
   3336 	    * At this point all adjacent text/CDATA-section nodes
   3337 	    * have been merged.
   3338 	    *
   3339 	    * Strip whitespace-only text-nodes.
   3340 	    * (cctxt->inode->stripWhitespace)
   3341 	    */
   3342 	    if ((value == NULL) || (*value == 0) ||
   3343 		(((cctxt->inode->stripWhitespace) ||
   3344 		  (! cctxt->inode->preserveWhitespace)) &&
   3345 		 IS_BLANK(*value) &&
   3346 		 xsltIsBlank(value)))
   3347 	    {
   3348 		if (textNode != cur) {
   3349 		    xmlUnlinkNode(textNode);
   3350 		    xmlFreeNode(textNode);
   3351 		} else
   3352 		    deleteNode = textNode;
   3353 		textNode = NULL;
   3354 		goto next_sibling;
   3355 	    }
   3356 	    /*
   3357 	    * Convert CDATA-section nodes to text-nodes.
   3358 	    * TODO: Can this produce problems?
   3359 	    */
   3360 	    if (textNode->type != XML_TEXT_NODE) {
   3361 		textNode->type = XML_TEXT_NODE;
   3362 		textNode->name = xmlStringText;
   3363 	    }
   3364 	    if (internalize &&
   3365 		(textNode->content != NULL) &&
   3366 		(!xmlDictOwns(style->dict, textNode->content)))
   3367 	    {
   3368 		/*
   3369 		* Internalize the string.
   3370 		*/
   3371 		value = (xmlChar *) xmlDictLookup(style->dict,
   3372 		    textNode->content, -1);
   3373 		xmlNodeSetContent(textNode, NULL);
   3374 		textNode->content = value;
   3375 	    }
   3376 	    textNode = NULL;
   3377 	    /*
   3378 	    * Note that "disable-output-escaping" of the xsl:text
   3379 	    * element will be applied at a later level, when
   3380 	    * XSLT elements are processed.
   3381 	    */
   3382 	}
   3383 
   3384 next_sibling:
   3385 	if (cur->type == XML_ELEMENT_NODE) {
   3386 	    xsltCompilerNodePop(cctxt, cur);
   3387 	}
   3388 	if (cur == node)
   3389 	    break;
   3390 	if (cur->next != NULL) {
   3391 	    cur = cur->next;
   3392 	} else {
   3393 	    cur = cur->parent;
   3394 	    inXSLText = 0;
   3395 	    goto next_sibling;
   3396 	};
   3397     }
   3398     if (deleteNode != NULL) {
   3399 #ifdef WITH_XSLT_DEBUG_PARSING
   3400 	xsltGenericDebug(xsltGenericDebugContext,
   3401 	 "xsltParsePreprocessStylesheetTree: removing node\n");
   3402 #endif
   3403 	xmlUnlinkNode(deleteNode);
   3404 	xmlFreeNode(deleteNode);
   3405     }
   3406     return(0);
   3407 
   3408 internal_err:
   3409     return(-1);
   3410 }
   3411 
   3412 #endif /* XSLT_REFACTORED */
   3413 
   3414 #ifdef XSLT_REFACTORED
   3415 #else
   3416 static void
   3417 xsltPrecomputeStylesheet(xsltStylesheetPtr style, xmlNodePtr cur)
   3418 {
   3419     xmlNodePtr deleteNode, styleelem;
   3420     int internalize = 0;
   3421 
   3422     if ((style == NULL) || (cur == NULL))
   3423         return;
   3424 
   3425     if ((cur->doc != NULL) && (style->dict != NULL) &&
   3426         (cur->doc->dict == style->dict))
   3427 	internalize = 1;
   3428     else
   3429         style->internalized = 0;
   3430 
   3431     if ((cur != NULL) && (IS_XSLT_ELEM(cur)) &&
   3432         (IS_XSLT_NAME(cur, "stylesheet"))) {
   3433 	styleelem = cur;
   3434     } else {
   3435         styleelem = NULL;
   3436     }
   3437 
   3438     /*
   3439      * This content comes from the stylesheet
   3440      * For stylesheets, the set of whitespace-preserving
   3441      * element names consists of just xsl:text.
   3442      */
   3443     deleteNode = NULL;
   3444     while (cur != NULL) {
   3445 	if (deleteNode != NULL) {
   3446 #ifdef WITH_XSLT_DEBUG_BLANKS
   3447 	    xsltGenericDebug(xsltGenericDebugContext,
   3448 	     "xsltPrecomputeStylesheet: removing ignorable blank node\n");
   3449 #endif
   3450 	    xmlUnlinkNode(deleteNode);
   3451 	    xmlFreeNode(deleteNode);
   3452 	    deleteNode = NULL;
   3453 	}
   3454 	if (cur->type == XML_ELEMENT_NODE) {
   3455 	    int exclPrefixes;
   3456 	    /*
   3457 	     * Internalize attributes values.
   3458 	     */
   3459 	    if ((internalize) && (cur->properties != NULL)) {
   3460 	        xmlAttrPtr attr = cur->properties;
   3461 		xmlNodePtr txt;
   3462 
   3463 		while (attr != NULL) {
   3464 		    txt = attr->children;
   3465 		    if ((txt != NULL) && (txt->type == XML_TEXT_NODE) &&
   3466 		        (txt->content != NULL) &&
   3467 			(!xmlDictOwns(style->dict, txt->content)))
   3468 		    {
   3469 			xmlChar *tmp;
   3470 
   3471 			/*
   3472 			 * internalize the text string, goal is to speed
   3473 			 * up operations and minimize used space by compiled
   3474 			 * stylesheets.
   3475 			 */
   3476 			tmp = (xmlChar *) xmlDictLookup(style->dict,
   3477 			                                txt->content, -1);
   3478 			if (tmp != txt->content) {
   3479 			    xmlNodeSetContent(txt, NULL);
   3480 			    txt->content = tmp;
   3481 			}
   3482 		    }
   3483 		    attr = attr->next;
   3484 		}
   3485 	    }
   3486 	    if (IS_XSLT_ELEM(cur)) {
   3487 		exclPrefixes = 0;
   3488 		xsltStylePreCompute(style, cur);
   3489 		if (IS_XSLT_NAME(cur, "text")) {
   3490 		    for (;exclPrefixes > 0;exclPrefixes--)
   3491 			exclPrefixPop(style);
   3492 		    goto skip_children;
   3493 		}
   3494 	    } else {
   3495 		exclPrefixes = xsltParseStylesheetExcludePrefix(style, cur, 0);
   3496 	    }
   3497 
   3498 	    if ((cur->nsDef != NULL) && (style->exclPrefixNr > 0)) {
   3499 		xmlNsPtr ns = cur->nsDef, prev = NULL, next;
   3500 		xmlNodePtr root = NULL;
   3501 		int i, moved;
   3502 
   3503 		root = xmlDocGetRootElement(cur->doc);
   3504 		if ((root != NULL) && (root != cur)) {
   3505 		    while (ns != NULL) {
   3506 			moved = 0;
   3507 			next = ns->next;
   3508 			for (i = 0;i < style->exclPrefixNr;i++) {
   3509 			    if ((ns->prefix != NULL) &&
   3510 			        (xmlStrEqual(ns->href,
   3511 					     style->exclPrefixTab[i]))) {
   3512 				/*
   3513 				 * Move the namespace definition on the root
   3514 				 * element to avoid duplicating it without
   3515 				 * loosing it.
   3516 				 */
   3517 				if (prev == NULL) {
   3518 				    cur->nsDef = ns->next;
   3519 				} else {
   3520 				    prev->next = ns->next;
   3521 				}
   3522 				ns->next = root->nsDef;
   3523 				root->nsDef = ns;
   3524 				moved = 1;
   3525 				break;
   3526 			    }
   3527 			}
   3528 			if (moved == 0)
   3529 			    prev = ns;
   3530 			ns = next;
   3531 		    }
   3532 		}
   3533 	    }
   3534 	    /*
   3535 	     * If we have prefixes locally, recurse and pop them up when
   3536 	     * going back
   3537 	     */
   3538 	    if (exclPrefixes > 0) {
   3539 		xsltPrecomputeStylesheet(style, cur->children);
   3540 		for (;exclPrefixes > 0;exclPrefixes--)
   3541 		    exclPrefixPop(style);
   3542 		goto skip_children;
   3543 	    }
   3544 	} else if (cur->type == XML_TEXT_NODE) {
   3545 	    if (IS_BLANK_NODE(cur)) {
   3546 		if (xmlNodeGetSpacePreserve(cur) != 1) {
   3547 		    deleteNode = cur;
   3548 		}
   3549 	    } else if ((cur->content != NULL) && (internalize) &&
   3550 	               (!xmlDictOwns(style->dict, cur->content))) {
   3551 		xmlChar *tmp;
   3552 
   3553 		/*
   3554 		 * internalize the text string, goal is to speed
   3555 		 * up operations and minimize used space by compiled
   3556 		 * stylesheets.
   3557 		 */
   3558 		tmp = (xmlChar *) xmlDictLookup(style->dict, cur->content, -1);
   3559 		xmlNodeSetContent(cur, NULL);
   3560 		cur->content = tmp;
   3561 	    }
   3562 	} else if ((cur->type != XML_ELEMENT_NODE) &&
   3563 		   (cur->type != XML_CDATA_SECTION_NODE)) {
   3564 	    deleteNode = cur;
   3565 	    goto skip_children;
   3566 	}
   3567 
   3568 	/*
   3569 	 * Skip to next node. In case of a namespaced element children of
   3570 	 * the stylesheet and not in the XSLT namespace and not an extension
   3571 	 * element, ignore its content.
   3572 	 */
   3573 	if ((cur->type == XML_ELEMENT_NODE) && (cur->ns != NULL) &&
   3574 	    (styleelem != NULL) && (cur->parent == styleelem) &&
   3575 	    (!xmlStrEqual(cur->ns->href, XSLT_NAMESPACE)) &&
   3576 	    (!xsltCheckExtURI(style, cur->ns->href))) {
   3577 	    goto skip_children;
   3578 	} else if (cur->children != NULL) {
   3579 	    if ((cur->children->type != XML_ENTITY_DECL) &&
   3580 		(cur->children->type != XML_ENTITY_REF_NODE) &&
   3581 		(cur->children->type != XML_ENTITY_NODE)) {
   3582 		cur = cur->children;
   3583 		continue;
   3584 	    }
   3585 	}
   3586 
   3587 skip_children:
   3588 	if (cur->next != NULL) {
   3589 	    cur = cur->next;
   3590 	    continue;
   3591 	}
   3592 	do {
   3593 
   3594 	    cur = cur->parent;
   3595 	    if (cur == NULL)
   3596 		break;
   3597 	    if (cur == (xmlNodePtr) style->doc) {
   3598 		cur = NULL;
   3599 		break;
   3600 	    }
   3601 	    if (cur->next != NULL) {
   3602 		cur = cur->next;
   3603 		break;
   3604 	    }
   3605 	} while (cur != NULL);
   3606     }
   3607     if (deleteNode != NULL) {
   3608 #ifdef WITH_XSLT_DEBUG_PARSING
   3609 	xsltGenericDebug(xsltGenericDebugContext,
   3610 	 "xsltPrecomputeStylesheet: removing ignorable blank node\n");
   3611 #endif
   3612 	xmlUnlinkNode(deleteNode);
   3613 	xmlFreeNode(deleteNode);
   3614     }
   3615 }
   3616 #endif /* end of else XSLT_REFACTORED */
   3617 
   3618 /**
   3619  * xsltGatherNamespaces:
   3620  * @style:  the XSLT stylesheet
   3621  *
   3622  * Browse the stylesheet and build the namspace hash table which
   3623  * will be used for XPath interpretation. If needed do a bit of normalization
   3624  */
   3625 
   3626 static void
   3627 xsltGatherNamespaces(xsltStylesheetPtr style) {
   3628     xmlNodePtr cur;
   3629     const xmlChar *URI;
   3630 
   3631     if (style == NULL)
   3632         return;
   3633     /*
   3634      * TODO: basically if the stylesheet uses the same prefix for different
   3635      *       patterns, well they may be in problem, hopefully they will get
   3636      *       a warning first.
   3637      */
   3638     /*
   3639     * TODO: Eliminate the use of the hash for XPath expressions.
   3640     *   An expression should be evaluated in the context of the in-scope
   3641     *   namespaces; eliminate the restriction of an XML document to contain
   3642     *   no duplicate prefixes for different namespace names.
   3643     *
   3644     */
   3645     cur = xmlDocGetRootElement(style->doc);
   3646     while (cur != NULL) {
   3647 	if (cur->type == XML_ELEMENT_NODE) {
   3648 	    xmlNsPtr ns = cur->nsDef;
   3649 	    while (ns != NULL) {
   3650 		if (ns->prefix != NULL) {
   3651 		    if (style->nsHash == NULL) {
   3652 			style->nsHash = xmlHashCreate(10);
   3653 			if (style->nsHash == NULL) {
   3654 			    xsltTransformError(NULL, style, cur,
   3655 		 "xsltGatherNamespaces: failed to create hash table\n");
   3656 			    style->errors++;
   3657 			    return;
   3658 			}
   3659 		    }
   3660 		    URI = xmlHashLookup(style->nsHash, ns->prefix);
   3661 		    if ((URI != NULL) && (!xmlStrEqual(URI, ns->href))) {
   3662 			xsltTransformError(NULL, style, cur,
   3663 	     "Namespaces prefix %s used for multiple namespaces\n",ns->prefix);
   3664 			style->warnings++;
   3665 		    } else if (URI == NULL) {
   3666 			xmlHashUpdateEntry(style->nsHash, ns->prefix,
   3667 			    (void *) ns->href, (xmlHashDeallocator)xmlFree);
   3668 
   3669 #ifdef WITH_XSLT_DEBUG_PARSING
   3670 			xsltGenericDebug(xsltGenericDebugContext,
   3671 		 "Added namespace: %s mapped to %s\n", ns->prefix, ns->href);
   3672 #endif
   3673 		    }
   3674 		}
   3675 		ns = ns->next;
   3676 	    }
   3677 	}
   3678 
   3679 	/*
   3680 	 * Skip to next node
   3681 	 */
   3682 	if (cur->children != NULL) {
   3683 	    if (cur->children->type != XML_ENTITY_DECL) {
   3684 		cur = cur->children;
   3685 		continue;
   3686 	    }
   3687 	}
   3688 	if (cur->next != NULL) {
   3689 	    cur = cur->next;
   3690 	    continue;
   3691 	}
   3692 
   3693 	do {
   3694 	    cur = cur->parent;
   3695 	    if (cur == NULL)
   3696 		break;
   3697 	    if (cur == (xmlNodePtr) style->doc) {
   3698 		cur = NULL;
   3699 		break;
   3700 	    }
   3701 	    if (cur->next != NULL) {
   3702 		cur = cur->next;
   3703 		break;
   3704 	    }
   3705 	} while (cur != NULL);
   3706     }
   3707 }
   3708 
   3709 #ifdef XSLT_REFACTORED
   3710 
   3711 static xsltStyleType
   3712 xsltGetXSLTElementTypeByNode(xsltCompilerCtxtPtr cctxt,
   3713 			     xmlNodePtr node)
   3714 {
   3715     if ((node == NULL) || (node->type != XML_ELEMENT_NODE) ||
   3716 	(node->name == NULL))
   3717 	return(0);
   3718 
   3719     if (node->name[0] == 'a') {
   3720 	if (IS_XSLT_NAME(node, "apply-templates"))
   3721 	    return(XSLT_FUNC_APPLYTEMPLATES);
   3722 	else if (IS_XSLT_NAME(node, "attribute"))
   3723 	    return(XSLT_FUNC_ATTRIBUTE);
   3724 	else if (IS_XSLT_NAME(node, "apply-imports"))
   3725 	    return(XSLT_FUNC_APPLYIMPORTS);
   3726 	else if (IS_XSLT_NAME(node, "attribute-set"))
   3727 	    return(0);
   3728 
   3729     } else if (node->name[0] == 'c') {
   3730 	if (IS_XSLT_NAME(node, "choose"))
   3731 	    return(XSLT_FUNC_CHOOSE);
   3732 	else if (IS_XSLT_NAME(node, "copy"))
   3733 	    return(XSLT_FUNC_COPY);
   3734 	else if (IS_XSLT_NAME(node, "copy-of"))
   3735 	    return(XSLT_FUNC_COPYOF);
   3736 	else if (IS_XSLT_NAME(node, "call-template"))
   3737 	    return(XSLT_FUNC_CALLTEMPLATE);
   3738 	else if (IS_XSLT_NAME(node, "comment"))
   3739 	    return(XSLT_FUNC_COMMENT);
   3740 
   3741     } else if (node->name[0] == 'd') {
   3742 	if (IS_XSLT_NAME(node, "document"))
   3743 	    return(XSLT_FUNC_DOCUMENT);
   3744 	else if (IS_XSLT_NAME(node, "decimal-format"))
   3745 	    return(0);
   3746 
   3747     } else if (node->name[0] == 'e') {
   3748 	if (IS_XSLT_NAME(node, "element"))
   3749 	    return(XSLT_FUNC_ELEMENT);
   3750 
   3751     } else if (node->name[0] == 'f') {
   3752 	if (IS_XSLT_NAME(node, "for-each"))
   3753 	    return(XSLT_FUNC_FOREACH);
   3754 	else if (IS_XSLT_NAME(node, "fallback"))
   3755 	    return(XSLT_FUNC_FALLBACK);
   3756 
   3757     } else if (*(node->name) == 'i') {
   3758 	if (IS_XSLT_NAME(node, "if"))
   3759 	    return(XSLT_FUNC_IF);
   3760 	else if (IS_XSLT_NAME(node, "include"))
   3761 	    return(0);
   3762 	else if (IS_XSLT_NAME(node, "import"))
   3763 	    return(0);
   3764 
   3765     } else if (*(node->name) == 'k') {
   3766 	if (IS_XSLT_NAME(node, "key"))
   3767 	    return(0);
   3768 
   3769     } else if (*(node->name) == 'm') {
   3770 	if (IS_XSLT_NAME(node, "message"))
   3771 	    return(XSLT_FUNC_MESSAGE);
   3772 
   3773     } else if (*(node->name) == 'n') {
   3774 	if (IS_XSLT_NAME(node, "number"))
   3775 	    return(XSLT_FUNC_NUMBER);
   3776 	else if (IS_XSLT_NAME(node, "namespace-alias"))
   3777 	    return(0);
   3778 
   3779     } else if (*(node->name) == 'o') {
   3780 	if (IS_XSLT_NAME(node, "otherwise"))
   3781 	    return(XSLT_FUNC_OTHERWISE);
   3782 	else if (IS_XSLT_NAME(node, "output"))
   3783 	    return(0);
   3784 
   3785     } else if (*(node->name) == 'p') {
   3786 	if (IS_XSLT_NAME(node, "param"))
   3787 	    return(XSLT_FUNC_PARAM);
   3788 	else if (IS_XSLT_NAME(node, "processing-instruction"))
   3789 	    return(XSLT_FUNC_PI);
   3790 	else if (IS_XSLT_NAME(node, "preserve-space"))
   3791 	    return(0);
   3792 
   3793     } else if (*(node->name) == 's') {
   3794 	if (IS_XSLT_NAME(node, "sort"))
   3795 	    return(XSLT_FUNC_SORT);
   3796 	else if (IS_XSLT_NAME(node, "strip-space"))
   3797 	    return(0);
   3798 	else if (IS_XSLT_NAME(node, "stylesheet"))
   3799 	    return(0);
   3800 
   3801     } else if (node->name[0] == 't') {
   3802 	if (IS_XSLT_NAME(node, "text"))
   3803 	    return(XSLT_FUNC_TEXT);
   3804 	else if (IS_XSLT_NAME(node, "template"))
   3805 	    return(0);
   3806 	else if (IS_XSLT_NAME(node, "transform"))
   3807 	    return(0);
   3808 
   3809     } else if (*(node->name) == 'v') {
   3810 	if (IS_XSLT_NAME(node, "value-of"))
   3811 	    return(XSLT_FUNC_VALUEOF);
   3812 	else if (IS_XSLT_NAME(node, "variable"))
   3813 	    return(XSLT_FUNC_VARIABLE);
   3814 
   3815     } else if (*(node->name) == 'w') {
   3816 	if (IS_XSLT_NAME(node, "when"))
   3817 	    return(XSLT_FUNC_WHEN);
   3818 	if (IS_XSLT_NAME(node, "with-param"))
   3819 	    return(XSLT_FUNC_WITHPARAM);
   3820     }
   3821     return(0);
   3822 }
   3823 
   3824 /**
   3825  * xsltParseAnyXSLTElem:
   3826  *
   3827  * @cctxt: the compilation context
   3828  * @elem: the element node of the XSLT instruction
   3829  *
   3830  * Parses, validates the content models and compiles XSLT instructions.
   3831  *
   3832  * Returns 0 if everything's fine;
   3833  *         -1 on API or internal errors.
   3834  */
   3835 int
   3836 xsltParseAnyXSLTElem(xsltCompilerCtxtPtr cctxt, xmlNodePtr elem)
   3837 {
   3838     if ((cctxt == NULL) || (elem == NULL) ||
   3839 	(elem->type != XML_ELEMENT_NODE))
   3840 	return(-1);
   3841 
   3842     elem->psvi = NULL;
   3843 
   3844     if (! (IS_XSLT_ELEM_FAST(elem)))
   3845 	return(-1);
   3846     /*
   3847     * Detection of handled content of extension instructions.
   3848     */
   3849     if (cctxt->inode->category == XSLT_ELEMENT_CATEGORY_EXTENSION) {
   3850 	cctxt->inode->extContentHandled = 1;
   3851     }
   3852 
   3853     xsltCompilerNodePush(cctxt, elem);
   3854     /*
   3855     * URGENT TODO: Find a way to speed up this annoying redundant
   3856     *  textual node-name and namespace comparison.
   3857     */
   3858     if (cctxt->inode->prev->curChildType != 0)
   3859 	cctxt->inode->type = cctxt->inode->prev->curChildType;
   3860     else
   3861 	cctxt->inode->type = xsltGetXSLTElementTypeByNode(cctxt, elem);
   3862     /*
   3863     * Update the in-scope namespaces if needed.
   3864     */
   3865     if (elem->nsDef != NULL)
   3866 	cctxt->inode->inScopeNs =
   3867 	    xsltCompilerBuildInScopeNsList(cctxt, elem);
   3868     /*
   3869     * xsltStylePreCompute():
   3870     *  This will compile the information found on the current
   3871     *  element's attributes. NOTE that this won't process the
   3872     *  children of the instruction.
   3873     */
   3874     xsltStylePreCompute(cctxt->style, elem);
   3875     /*
   3876     * TODO: How to react on errors in xsltStylePreCompute() ?
   3877     */
   3878 
   3879     /*
   3880     * Validate the content model of the XSLT-element.
   3881     */
   3882     switch (cctxt->inode->type) {
   3883 	case XSLT_FUNC_APPLYIMPORTS:
   3884 	    /* EMPTY */
   3885 	    goto empty_content;
   3886 	case XSLT_FUNC_APPLYTEMPLATES:
   3887 	    /* <!-- Content: (xsl:sort | xsl:with-param)* --> */
   3888 	    goto apply_templates;
   3889 	case XSLT_FUNC_ATTRIBUTE:
   3890 	    /* <!-- Content: template --> */
   3891 	    goto sequence_constructor;
   3892 	case XSLT_FUNC_CALLTEMPLATE:
   3893 	    /* <!-- Content: xsl:with-param* --> */
   3894 	    goto call_template;
   3895 	case XSLT_FUNC_CHOOSE:
   3896 	    /* <!-- Content: (xsl:when+, xsl:otherwise?) --> */
   3897 	    goto choose;
   3898 	case XSLT_FUNC_COMMENT:
   3899 	    /* <!-- Content: template --> */
   3900 	    goto sequence_constructor;
   3901 	case XSLT_FUNC_COPY:
   3902 	    /* <!-- Content: template --> */
   3903 	    goto sequence_constructor;
   3904 	case XSLT_FUNC_COPYOF:
   3905 	    /* EMPTY */
   3906 	    goto empty_content;
   3907 	case XSLT_FUNC_DOCUMENT: /* Extra one */
   3908 	    /* ?? template ?? */
   3909 	    goto sequence_constructor;
   3910 	case XSLT_FUNC_ELEMENT:
   3911 	    /* <!-- Content: template --> */
   3912 	    goto sequence_constructor;
   3913 	case XSLT_FUNC_FALLBACK:
   3914 	    /* <!-- Content: template --> */
   3915 	    goto sequence_constructor;
   3916 	case XSLT_FUNC_FOREACH:
   3917 	    /* <!-- Content: (xsl:sort*, template) --> */
   3918 	    goto for_each;
   3919 	case XSLT_FUNC_IF:
   3920 	    /* <!-- Content: template --> */
   3921 	    goto sequence_constructor;
   3922 	case XSLT_FUNC_OTHERWISE:
   3923 	    /* <!-- Content: template --> */
   3924 	    goto sequence_constructor;
   3925 	case XSLT_FUNC_MESSAGE:
   3926 	    /* <!-- Content: template --> */
   3927 	    goto sequence_constructor;
   3928 	case XSLT_FUNC_NUMBER:
   3929 	    /* EMPTY */
   3930 	    goto empty_content;
   3931 	case XSLT_FUNC_PARAM:
   3932 	    /*
   3933 	    * Check for redefinition.
   3934 	    */
   3935 	    if ((elem->psvi != NULL) && (cctxt->ivar != NULL)) {
   3936 		xsltVarInfoPtr ivar = cctxt->ivar;
   3937 
   3938 		do {
   3939 		    if ((ivar->name ==
   3940 			 ((xsltStyleItemParamPtr) elem->psvi)->name) &&
   3941 			(ivar->nsName ==
   3942 			 ((xsltStyleItemParamPtr) elem->psvi)->ns))
   3943 		    {
   3944 			elem->psvi = NULL;
   3945 			xsltTransformError(NULL, cctxt->style, elem,
   3946 			    "Redefinition of variable or parameter '%s'.\n",
   3947 			    ivar->name);
   3948 			cctxt->style->errors++;
   3949 			goto error;
   3950 		    }
   3951 		    ivar = ivar->prev;
   3952 		} while (ivar != NULL);
   3953 	    }
   3954 	    /*  <!-- Content: template --> */
   3955 	    goto sequence_constructor;
   3956 	case XSLT_FUNC_PI:
   3957 	    /*  <!-- Content: template --> */
   3958 	    goto sequence_constructor;
   3959 	case XSLT_FUNC_SORT:
   3960 	    /* EMPTY */
   3961 	    goto empty_content;
   3962 	case XSLT_FUNC_TEXT:
   3963 	    /* <!-- Content: #PCDATA --> */
   3964 	    goto text;
   3965 	case XSLT_FUNC_VALUEOF:
   3966 	    /* EMPTY */
   3967 	    goto empty_content;
   3968 	case XSLT_FUNC_VARIABLE:
   3969 	    /*
   3970 	    * Check for redefinition.
   3971 	    */
   3972 	    if ((elem->psvi != NULL) && (cctxt->ivar != NULL)) {
   3973 		xsltVarInfoPtr ivar = cctxt->ivar;
   3974 
   3975 		do {
   3976 		    if ((ivar->name ==
   3977 			 ((xsltStyleItemVariablePtr) elem->psvi)->name) &&
   3978 			(ivar->nsName ==
   3979 			 ((xsltStyleItemVariablePtr) elem->psvi)->ns))
   3980 		    {
   3981 			elem->psvi = NULL;
   3982 			xsltTransformError(NULL, cctxt->style, elem,
   3983 			    "Redefinition of variable or parameter '%s'.\n",
   3984 			    ivar->name);
   3985 			cctxt->style->errors++;
   3986 			goto error;
   3987 		    }
   3988 		    ivar = ivar->prev;
   3989 		} while (ivar != NULL);
   3990 	    }
   3991 	    /* <!-- Content: template --> */
   3992 	    goto sequence_constructor;
   3993 	case XSLT_FUNC_WHEN:
   3994 	    /* <!-- Content: template --> */
   3995 	    goto sequence_constructor;
   3996 	case XSLT_FUNC_WITHPARAM:
   3997 	    /* <!-- Content: template --> */
   3998 	    goto sequence_constructor;
   3999 	default:
   4000 #ifdef WITH_XSLT_DEBUG_PARSING
   4001 	    xsltGenericDebug(xsltGenericDebugContext,
   4002 		"xsltParseXSLTNode: Unhandled XSLT element '%s'.\n",
   4003 		elem->name);
   4004 #endif
   4005 	    xsltTransformError(NULL, cctxt->style, elem,
   4006 		"xsltParseXSLTNode: Internal error; "
   4007 		"unhandled XSLT element '%s'.\n", elem->name);
   4008 	    cctxt->style->errors++;
   4009 	    goto internal_err;
   4010     }
   4011 
   4012 apply_templates:
   4013     /* <!-- Content: (xsl:sort | xsl:with-param)* --> */
   4014     if (elem->children != NULL) {
   4015 	xmlNodePtr child = elem->children;
   4016 	do {
   4017 	    if (child->type == XML_ELEMENT_NODE) {
   4018 		if (IS_XSLT_ELEM_FAST(child)) {
   4019 		    if (xmlStrEqual(child->name, BAD_CAST "with-param")) {
   4020 			cctxt->inode->curChildType = XSLT_FUNC_WITHPARAM;
   4021 			xsltParseAnyXSLTElem(cctxt, child);
   4022 		    } else if (xmlStrEqual(child->name, BAD_CAST "sort")) {
   4023 			cctxt->inode->curChildType = XSLT_FUNC_SORT;
   4024 			xsltParseAnyXSLTElem(cctxt, child);
   4025 		    } else
   4026 			xsltParseContentError(cctxt->style, child);
   4027 		} else
   4028 		    xsltParseContentError(cctxt->style, child);
   4029 	    }
   4030 	    child = child->next;
   4031 	} while (child != NULL);
   4032     }
   4033     goto exit;
   4034 
   4035 call_template:
   4036     /* <!-- Content: xsl:with-param* --> */
   4037     if (elem->children != NULL) {
   4038 	xmlNodePtr child = elem->children;
   4039 	do {
   4040 	    if (child->type == XML_ELEMENT_NODE) {
   4041 		if (IS_XSLT_ELEM_FAST(child)) {
   4042 		    xsltStyleType type;
   4043 
   4044 		    type = xsltGetXSLTElementTypeByNode(cctxt, child);
   4045 		    if (type == XSLT_FUNC_WITHPARAM) {
   4046 			cctxt->inode->curChildType = XSLT_FUNC_WITHPARAM;
   4047 			xsltParseAnyXSLTElem(cctxt, child);
   4048 		    } else {
   4049 			xsltParseContentError(cctxt->style, child);
   4050 		    }
   4051 		} else
   4052 		    xsltParseContentError(cctxt->style, child);
   4053 	    }
   4054 	    child = child->next;
   4055 	} while (child != NULL);
   4056     }
   4057     goto exit;
   4058 
   4059 text:
   4060     if (elem->children != NULL) {
   4061 	xmlNodePtr child = elem->children;
   4062 	do {
   4063 	    if ((child->type != XML_TEXT_NODE) &&
   4064 		(child->type != XML_CDATA_SECTION_NODE))
   4065 	    {
   4066 		xsltTransformError(NULL, cctxt->style, elem,
   4067 		    "The XSLT 'text' element must have only character "
   4068 		    "data as content.\n");
   4069 	    }
   4070 	    child = child->next;
   4071 	} while (child != NULL);
   4072     }
   4073     goto exit;
   4074 
   4075 empty_content:
   4076     if (elem->children != NULL) {
   4077 	xmlNodePtr child = elem->children;
   4078 	/*
   4079 	* Relaxed behaviour: we will allow whitespace-only text-nodes.
   4080 	*/
   4081 	do {
   4082 	    if (((child->type != XML_TEXT_NODE) &&
   4083 		 (child->type != XML_CDATA_SECTION_NODE)) ||
   4084 		(! IS_BLANK_NODE(child)))
   4085 	    {
   4086 		xsltTransformError(NULL, cctxt->style, elem,
   4087 		    "This XSLT element must have no content.\n");
   4088 		cctxt->style->errors++;
   4089 		break;
   4090 	    }
   4091 	    child = child->next;
   4092 	} while (child != NULL);
   4093     }
   4094     goto exit;
   4095 
   4096 choose:
   4097     /* <!-- Content: (xsl:when+, xsl:otherwise?) --> */
   4098     /*
   4099     * TODO: text-nodes in between are *not* allowed in XSLT 1.0.
   4100     *   The old behaviour did not check this.
   4101     * NOTE: In XSLT 2.0 they are stripped beforehand
   4102     *  if whitespace-only (regardless of xml:space).
   4103     */
   4104     if (elem->children != NULL) {
   4105 	xmlNodePtr child = elem->children;
   4106 	int nbWhen = 0, nbOtherwise = 0, err = 0;
   4107 	do {
   4108 	    if (child->type == XML_ELEMENT_NODE) {
   4109 		if (IS_XSLT_ELEM_FAST(child)) {
   4110 		    xsltStyleType type;
   4111 
   4112 		    type = xsltGetXSLTElementTypeByNode(cctxt, child);
   4113 		    if (type == XSLT_FUNC_WHEN) {
   4114 			nbWhen++;
   4115 			if (nbOtherwise) {
   4116 			    xsltParseContentError(cctxt->style, child);
   4117 			    err = 1;
   4118 			    break;
   4119 			}
   4120 			cctxt->inode->curChildType = XSLT_FUNC_WHEN;
   4121 			xsltParseAnyXSLTElem(cctxt, child);
   4122 		    } else if (type == XSLT_FUNC_OTHERWISE) {
   4123 			if (! nbWhen) {
   4124 			    xsltParseContentError(cctxt->style, child);
   4125 			    err = 1;
   4126 			    break;
   4127 			}
   4128 			if (nbOtherwise) {
   4129 			    xsltTransformError(NULL, cctxt->style, elem,
   4130 				"The XSLT 'choose' element must not contain "
   4131 				"more than one XSLT 'otherwise' element.\n");
   4132 			    cctxt->style->errors++;
   4133 			    err = 1;
   4134 			    break;
   4135 			}
   4136 			nbOtherwise++;
   4137 			cctxt->inode->curChildType = XSLT_FUNC_OTHERWISE;
   4138 			xsltParseAnyXSLTElem(cctxt, child);
   4139 		    } else
   4140 			xsltParseContentError(cctxt->style, child);
   4141 		} else
   4142 		    xsltParseContentError(cctxt->style, child);
   4143 	    }
   4144 	    /*
   4145 		else
   4146 		    xsltParseContentError(cctxt, child);
   4147 	    */
   4148 	    child = child->next;
   4149 	} while (child != NULL);
   4150 	if ((! err) && (! nbWhen)) {
   4151 	    xsltTransformError(NULL, cctxt->style, elem,
   4152 		"The XSLT element 'choose' must contain at least one "
   4153 		"XSLT element 'when'.\n");
   4154 		cctxt->style->errors++;
   4155 	}
   4156     }
   4157     goto exit;
   4158 
   4159 for_each:
   4160     /* <!-- Content: (xsl:sort*, template) --> */
   4161     /*
   4162     * NOTE: Text-nodes before xsl:sort are *not* allowed in XSLT 1.0.
   4163     *   The old behaviour did not allow this, but it catched this
   4164     *   only at transformation-time.
   4165     *   In XSLT 2.0 they are stripped beforehand if whitespace-only
   4166     *   (regardless of xml:space).
   4167     */
   4168     if (elem->children != NULL) {
   4169 	xmlNodePtr child = elem->children;
   4170 	/*
   4171 	* Parse xsl:sort first.
   4172 	*/
   4173 	do {
   4174 	    if ((child->type == XML_ELEMENT_NODE) &&
   4175 		IS_XSLT_ELEM_FAST(child))
   4176 	    {
   4177 		if (xsltGetXSLTElementTypeByNode(cctxt, child) ==
   4178 		    XSLT_FUNC_SORT)
   4179 		{
   4180 		    cctxt->inode->curChildType = XSLT_FUNC_SORT;
   4181 		    xsltParseAnyXSLTElem(cctxt, child);
   4182 		} else
   4183 		    break;
   4184 	    } else
   4185 		break;
   4186 	    child = child->next;
   4187 	} while (child != NULL);
   4188 	/*
   4189 	* Parse the sequece constructor.
   4190 	*/
   4191 	if (child != NULL)
   4192 	    xsltParseSequenceConstructor(cctxt, child);
   4193     }
   4194     goto exit;
   4195 
   4196 sequence_constructor:
   4197     /*
   4198     * Parse the sequence constructor.
   4199     */
   4200     if (elem->children != NULL)
   4201 	xsltParseSequenceConstructor(cctxt, elem->children);
   4202 
   4203     /*
   4204     * Register information for vars/params. Only needed if there
   4205     * are any following siblings.
   4206     */
   4207     if ((elem->next != NULL) &&
   4208 	((cctxt->inode->type == XSLT_FUNC_VARIABLE) ||
   4209 	 (cctxt->inode->type == XSLT_FUNC_PARAM)))
   4210     {
   4211 	if ((elem->psvi != NULL) &&
   4212 	    (((xsltStyleBasicItemVariablePtr) elem->psvi)->name))
   4213 	{
   4214 	    xsltCompilerVarInfoPush(cctxt, elem,
   4215 		((xsltStyleBasicItemVariablePtr) elem->psvi)->name,
   4216 		((xsltStyleBasicItemVariablePtr) elem->psvi)->ns);
   4217 	}
   4218     }
   4219 
   4220 error:
   4221 exit:
   4222     xsltCompilerNodePop(cctxt, elem);
   4223     return(0);
   4224 
   4225 internal_err:
   4226     xsltCompilerNodePop(cctxt, elem);
   4227     return(-1);
   4228 }
   4229 
   4230 /**
   4231  * xsltForwardsCompatUnkownItemCreate:
   4232  *
   4233  * @cctxt: the compilation context
   4234  *
   4235  * Creates a compiled representation of the unknown
   4236  * XSLT instruction.
   4237  *
   4238  * Returns the compiled representation.
   4239  */
   4240 static xsltStyleItemUknownPtr
   4241 xsltForwardsCompatUnkownItemCreate(xsltCompilerCtxtPtr cctxt)
   4242 {
   4243     xsltStyleItemUknownPtr item;
   4244 
   4245     item = (xsltStyleItemUknownPtr) xmlMalloc(sizeof(xsltStyleItemUknown));
   4246     if (item == NULL) {
   4247 	xsltTransformError(NULL, cctxt->style, NULL,
   4248 	    "Internal error in xsltForwardsCompatUnkownItemCreate(): "
   4249 	    "Failed to allocate memory.\n");
   4250 	cctxt->style->errors++;
   4251 	return(NULL);
   4252     }
   4253     memset(item, 0, sizeof(xsltStyleItemUknown));
   4254     item->type = XSLT_FUNC_UNKOWN_FORWARDS_COMPAT;
   4255     /*
   4256     * Store it in the stylesheet.
   4257     */
   4258     item->next = cctxt->style->preComps;
   4259     cctxt->style->preComps = (xsltElemPreCompPtr) item;
   4260     return(item);
   4261 }
   4262 
   4263 /**
   4264  * xsltParseUnknownXSLTElem:
   4265  *
   4266  * @cctxt: the compilation context
   4267  * @node: the element of the unknown XSLT instruction
   4268  *
   4269  * Parses an unknown XSLT element.
   4270  * If forwards compatible mode is enabled this will allow
   4271  * such an unknown XSLT and; otherwise it is rejected.
   4272  *
   4273  * Returns 1 in the unknown XSLT instruction is rejected,
   4274  *         0 if everything's fine and
   4275  *         -1 on API or internal errors.
   4276  */
   4277 static int
   4278 xsltParseUnknownXSLTElem(xsltCompilerCtxtPtr cctxt,
   4279 			    xmlNodePtr node)
   4280 {
   4281     if ((cctxt == NULL) || (node == NULL))
   4282 	return(-1);
   4283 
   4284     /*
   4285     * Detection of handled content of extension instructions.
   4286     */
   4287     if (cctxt->inode->category == XSLT_ELEMENT_CATEGORY_EXTENSION) {
   4288 	cctxt->inode->extContentHandled = 1;
   4289     }
   4290     if (cctxt->inode->forwardsCompat == 0) {
   4291 	/*
   4292 	* We are not in forwards-compatible mode, so raise an error.
   4293 	*/
   4294 	xsltTransformError(NULL, cctxt->style, node,
   4295 	    "Unknown XSLT element '%s'.\n", node->name);
   4296 	cctxt->style->errors++;
   4297 	return(1);
   4298     }
   4299     /*
   4300     * Forwards-compatible mode.
   4301     * ------------------------
   4302     *
   4303     * Parse/compile xsl:fallback elements.
   4304     *
   4305     * QUESTION: Do we have to raise an error if there's no xsl:fallback?
   4306     * ANSWER: No, since in the stylesheet the fallback behaviour might
   4307     *  also be provided by using the XSLT function "element-available".
   4308     */
   4309     if (cctxt->unknownItem == NULL) {
   4310 	/*
   4311 	* Create a singleton for all unknown XSLT instructions.
   4312 	*/
   4313 	cctxt->unknownItem = xsltForwardsCompatUnkownItemCreate(cctxt);
   4314 	if (cctxt->unknownItem == NULL) {
   4315 	    node->psvi = NULL;
   4316 	    return(-1);
   4317 	}
   4318     }
   4319     node->psvi = cctxt->unknownItem;
   4320     if (node->children == NULL)
   4321 	return(0);
   4322     else {
   4323 	xmlNodePtr child = node->children;
   4324 
   4325 	xsltCompilerNodePush(cctxt, node);
   4326 	/*
   4327 	* Update the in-scope namespaces if needed.
   4328 	*/
   4329 	if (node->nsDef != NULL)
   4330 	    cctxt->inode->inScopeNs =
   4331 		xsltCompilerBuildInScopeNsList(cctxt, node);
   4332 	/*
   4333 	* Parse all xsl:fallback children.
   4334 	*/
   4335 	do {
   4336 	    if ((child->type == XML_ELEMENT_NODE) &&
   4337 		IS_XSLT_ELEM_FAST(child) &&
   4338 		IS_XSLT_NAME(child, "fallback"))
   4339 	    {
   4340 		cctxt->inode->curChildType = XSLT_FUNC_FALLBACK;
   4341 		xsltParseAnyXSLTElem(cctxt, child);
   4342 	    }
   4343 	    child = child->next;
   4344 	} while (child != NULL);
   4345 
   4346 	xsltCompilerNodePop(cctxt, node);
   4347     }
   4348     return(0);
   4349 }
   4350 
   4351 /**
   4352  * xsltParseSequenceConstructor:
   4353  *
   4354  * @cctxt: the compilation context
   4355  * @cur: the start-node of the content to be parsed
   4356  *
   4357  * Parses a "template" content (or "sequence constructor" in XSLT 2.0 terms).
   4358  * This will additionally remove xsl:text elements from the tree.
   4359  */
   4360 void
   4361 xsltParseSequenceConstructor(xsltCompilerCtxtPtr cctxt, xmlNodePtr cur)
   4362 {
   4363     xsltStyleType type;
   4364     xmlNodePtr deleteNode = NULL;
   4365 
   4366     if (cctxt == NULL) {
   4367 	xmlGenericError(xmlGenericErrorContext,
   4368 	    "xsltParseSequenceConstructor: Bad arguments\n");
   4369 	cctxt->style->errors++;
   4370 	return;
   4371     }
   4372     /*
   4373     * Detection of handled content of extension instructions.
   4374     */
   4375     if (cctxt->inode->category == XSLT_ELEMENT_CATEGORY_EXTENSION) {
   4376 	cctxt->inode->extContentHandled = 1;
   4377     }
   4378     if (cur == NULL)
   4379 	return;
   4380     /*
   4381     * This is the content reffered to as a "template".
   4382     * E.g. an xsl:element has such content model:
   4383     * <xsl:element
   4384     *   name = { qname }
   4385     *   namespace = { uri-reference }
   4386     *   use-attribute-sets = qnames>
   4387     * <!-- Content: template -->
   4388     *
   4389     * NOTE that in XSLT-2 the term "template" was abandoned due to
   4390     *  confusion with xsl:template and the term "sequence constructor"
   4391     *  was introduced instead.
   4392     *
   4393     * The following XSLT-instructions are allowed to appear:
   4394     *  xsl:apply-templates, xsl:call-template, xsl:apply-imports,
   4395     *  xsl:for-each, xsl:value-of, xsl:copy-of, xsl:number,
   4396     *  xsl:choose, xsl:if, xsl:text, xsl:copy, xsl:variable,
   4397     *  xsl:message, xsl:fallback,
   4398     *  xsl:processing-instruction, xsl:comment, xsl:element
   4399     *  xsl:attribute.
   4400     * Additional allowed content:
   4401     * 1) extension instructions
   4402     * 2) literal result elements
   4403     * 3) PCDATA
   4404     *
   4405     * NOTE that this content model does *not* allow xsl:param.
   4406     */
   4407     while (cur != NULL) {
   4408 	if (deleteNode != NULL)	{
   4409 #ifdef WITH_XSLT_DEBUG_BLANKS
   4410 	    xsltGenericDebug(xsltGenericDebugContext,
   4411 	     "xsltParseSequenceConstructor: removing xsl:text element\n");
   4412 #endif
   4413 	    xmlUnlinkNode(deleteNode);
   4414 	    xmlFreeNode(deleteNode);
   4415 	    deleteNode = NULL;
   4416 	}
   4417 	if (cur->type == XML_ELEMENT_NODE) {
   4418 
   4419 	    if (cur->psvi == xsltXSLTTextMarker) {
   4420 		/*
   4421 		* xsl:text elements
   4422 		* --------------------------------------------------------
   4423 		*/
   4424 		xmlNodePtr tmp;
   4425 
   4426 		cur->psvi = NULL;
   4427 		/*
   4428 		* Mark the xsl:text element for later deletion.
   4429 		*/
   4430 		deleteNode = cur;
   4431 		/*
   4432 		* Validate content.
   4433 		*/
   4434 		tmp = cur->children;
   4435 		if (tmp) {
   4436 		    /*
   4437 		    * We don't expect more than one text-node in the
   4438 		    * content, since we already merged adjacent
   4439 		    * text/CDATA-nodes and eliminated PI/comment-nodes.
   4440 		    */
   4441 		    if ((tmp->type == XML_TEXT_NODE) ||
   4442 			(tmp->next == NULL))
   4443 		    {
   4444 			/*
   4445 			* Leave the contained text-node in the tree.
   4446 			*/
   4447 			xmlUnlinkNode(tmp);
   4448 			xmlAddPrevSibling(cur, tmp);
   4449 		    } else {
   4450 			tmp = NULL;
   4451 			xsltTransformError(NULL, cctxt->style, cur,
   4452 			    "Element 'xsl:text': Invalid type "
   4453 			    "of node found in content.\n");
   4454 			cctxt->style->errors++;
   4455 		    }
   4456 		}
   4457 		if (cur->properties) {
   4458 		    xmlAttrPtr attr;
   4459 		    /*
   4460 		    * TODO: We need to report errors for
   4461 		    *  invalid attrs.
   4462 		    */
   4463 		    attr = cur->properties;
   4464 		    do {
   4465 			if ((attr->ns == NULL) &&
   4466 			    (attr->name != NULL) &&
   4467 			    (attr->name[0] == 'd') &&
   4468 			    xmlStrEqual(attr->name,
   4469 			    BAD_CAST "disable-output-escaping"))
   4470 			{
   4471 			    /*
   4472 			    * Attr "disable-output-escaping".
   4473 			    * XSLT-2: This attribute is deprecated.
   4474 			    */
   4475 			    if ((attr->children != NULL) &&
   4476 				xmlStrEqual(attr->children->content,
   4477 				BAD_CAST "yes"))
   4478 			    {
   4479 				/*
   4480 				* Disable output escaping for this
   4481 				* text node.
   4482 				*/
   4483 				if (tmp)
   4484 				    tmp->name = xmlStringTextNoenc;
   4485 			    } else if ((attr->children == NULL) ||
   4486 				(attr->children->content == NULL) ||
   4487 				(!xmlStrEqual(attr->children->content,
   4488 				BAD_CAST "no")))
   4489 			    {
   4490 				xsltTransformError(NULL, cctxt->style,
   4491 				    cur,
   4492 				    "Attribute 'disable-output-escaping': "
   4493 				    "Invalid value. Expected is "
   4494 				    "'yes' or 'no'.\n");
   4495 				cctxt->style->errors++;
   4496 			    }
   4497 			    break;
   4498 			}
   4499 			attr = attr->next;
   4500 		    } while (attr != NULL);
   4501 		}
   4502 	    } else if (IS_XSLT_ELEM_FAST(cur)) {
   4503 		/*
   4504 		* TODO: Using the XSLT-marker is still not stable yet.
   4505 		*/
   4506 		/* if (cur->psvi == xsltXSLTElemMarker) { */
   4507 		/*
   4508 		* XSLT instructions
   4509 		* --------------------------------------------------------
   4510 		*/
   4511 		cur->psvi = NULL;
   4512 		type = xsltGetXSLTElementTypeByNode(cctxt, cur);
   4513 		switch (type) {
   4514 		    case XSLT_FUNC_APPLYIMPORTS:
   4515 		    case XSLT_FUNC_APPLYTEMPLATES:
   4516 		    case XSLT_FUNC_ATTRIBUTE:
   4517 		    case XSLT_FUNC_CALLTEMPLATE:
   4518 		    case XSLT_FUNC_CHOOSE:
   4519 		    case XSLT_FUNC_COMMENT:
   4520 		    case XSLT_FUNC_COPY:
   4521 		    case XSLT_FUNC_COPYOF:
   4522 		    case XSLT_FUNC_DOCUMENT: /* Extra one */
   4523 		    case XSLT_FUNC_ELEMENT:
   4524 		    case XSLT_FUNC_FALLBACK:
   4525 		    case XSLT_FUNC_FOREACH:
   4526 		    case XSLT_FUNC_IF:
   4527 		    case XSLT_FUNC_MESSAGE:
   4528 		    case XSLT_FUNC_NUMBER:
   4529 		    case XSLT_FUNC_PI:
   4530 		    case XSLT_FUNC_TEXT:
   4531 		    case XSLT_FUNC_VALUEOF:
   4532 		    case XSLT_FUNC_VARIABLE:
   4533 			/*
   4534 			* Parse the XSLT element.
   4535 			*/
   4536 			cctxt->inode->curChildType = type;
   4537 			xsltParseAnyXSLTElem(cctxt, cur);
   4538 			break;
   4539 		    default:
   4540 			xsltParseUnknownXSLTElem(cctxt, cur);
   4541 			cur = cur->next;
   4542 			continue;
   4543 		}
   4544 	    } else {
   4545 		/*
   4546 		* Non-XSLT elements
   4547 		* -----------------
   4548 		*/
   4549 		xsltCompilerNodePush(cctxt, cur);
   4550 		/*
   4551 		* Update the in-scope namespaces if needed.
   4552 		*/
   4553 		if (cur->nsDef != NULL)
   4554 		    cctxt->inode->inScopeNs =
   4555 			xsltCompilerBuildInScopeNsList(cctxt, cur);
   4556 		/*
   4557 		* The current element is either a literal result element
   4558 		* or an extension instruction.
   4559 		*
   4560 		* Process attr "xsl:extension-element-prefixes".
   4561 		* FUTURE TODO: IIRC in XSLT 2.0 this attribute must be
   4562 		* processed by the implementor of the extension function;
   4563 		* i.e., it won't be handled by the XSLT processor.
   4564 		*/
   4565 		/* SPEC 1.0:
   4566 		*   "exclude-result-prefixes" is only allowed on literal
   4567 		*   result elements and "xsl:exclude-result-prefixes"
   4568 		*   on xsl:stylesheet/xsl:transform.
   4569 		* SPEC 2.0:
   4570 		*   "There are a number of standard attributes
   4571 		*   that may appear on any XSLT element: specifically
   4572 		*   version, exclude-result-prefixes,
   4573 		*   extension-element-prefixes, xpath-default-namespace,
   4574 		*   default-collation, and use-when."
   4575 		*
   4576 		* SPEC 2.0:
   4577 		*   For literal result elements:
   4578 		*   "xsl:version, xsl:exclude-result-prefixes,
   4579 		*    xsl:extension-element-prefixes,
   4580 		*    xsl:xpath-default-namespace,
   4581 		*    xsl:default-collation, or xsl:use-when."
   4582 		*/
   4583 		if (cur->properties)
   4584 		    cctxt->inode->extElemNs =
   4585 			xsltParseExtElemPrefixes(cctxt,
   4586 			    cur, cctxt->inode->extElemNs,
   4587 			    XSLT_ELEMENT_CATEGORY_LRE);
   4588 		/*
   4589 		* Eval if we have an extension instruction here.
   4590 		*/
   4591 		if ((cur->ns != NULL) &&
   4592 		    (cctxt->inode->extElemNs != NULL) &&
   4593 		    (xsltCheckExtPrefix(cctxt->style, cur->ns->href) == 1))
   4594 		{
   4595 		    /*
   4596 		    * Extension instructions
   4597 		    * ----------------------------------------------------
   4598 		    * Mark the node information.
   4599 		    */
   4600 		    cctxt->inode->category = XSLT_ELEMENT_CATEGORY_EXTENSION;
   4601 		    cctxt->inode->extContentHandled = 0;
   4602 		    if (cur->psvi != NULL) {
   4603 			cur->psvi = NULL;
   4604 			/*
   4605 			* TODO: Temporary sanity check.
   4606 			*/
   4607 			xsltTransformError(NULL, cctxt->style, cur,
   4608 			    "Internal error in xsltParseSequenceConstructor(): "
   4609 			    "Occupied PSVI field.\n");
   4610 			cctxt->style->errors++;
   4611 			cur = cur->next;
   4612 			continue;
   4613 		    }
   4614 		    cur->psvi = (void *)
   4615 			xsltPreComputeExtModuleElement(cctxt->style, cur);
   4616 
   4617 		    if (cur->psvi == NULL) {
   4618 			/*
   4619 			* OLD COMMENT: "Unknown element, maybe registered
   4620 			*  at the context level. Mark it for later
   4621 			*  recognition."
   4622 			* QUESTION: What does the xsltExtMarker mean?
   4623 			*  ANSWER: It is used in
   4624 			*   xsltApplySequenceConstructor() at
   4625 			*   transformation-time to look out for extension
   4626 			*   registered in the transformation context.
   4627 			*/
   4628 			cur->psvi = (void *) xsltExtMarker;
   4629 		    }
   4630 		    /*
   4631 		    * BIG NOTE: Now the ugly part. In previous versions
   4632 		    *  of Libxslt (until 1.1.16), all the content of an
   4633 		    *  extension instruction was processed and compiled without
   4634 		    *  the need of the extension-author to explicitely call
   4635 		    *  such a processing;.We now need to mimic this old
   4636 		    *  behaviour in order to avoid breaking old code
   4637 		    *  on the extension-author's side.
   4638 		    * The mechanism:
   4639 		    *  1) If the author does *not* set the
   4640 		    *    compile-time-flag @extContentHandled, then we'll
   4641 		    *    parse the content assuming that it's a "template"
   4642 		    *    (or "sequence constructor in XSLT 2.0 terms).
   4643 		    *    NOTE: If the extension is registered at
   4644 		    *    transformation-time only, then there's no way of
   4645 		    *    knowing that content shall be valid, and we'll
   4646 		    *    process the content the same way.
   4647 		    *  2) If the author *does* set the flag, then we'll assume
   4648 		    *   that the author has handled the parsing him/herself
   4649 		    *   (e.g. called xsltParseSequenceConstructor(), etc.
   4650 		    *   explicitely in his/her code).
   4651 		    */
   4652 		    if ((cur->children != NULL) &&
   4653 			(cctxt->inode->extContentHandled == 0))
   4654 		    {
   4655 			/*
   4656 			* Default parsing of the content using the
   4657 			* sequence-constructor model.
   4658 			*/
   4659 			xsltParseSequenceConstructor(cctxt, cur->children);
   4660 		    }
   4661 		} else {
   4662 		    /*
   4663 		    * Literal result element
   4664 		    * ----------------------------------------------------
   4665 		    * Allowed XSLT attributes:
   4666 		    *  xsl:extension-element-prefixes CDATA #IMPLIED
   4667 		    *  xsl:exclude-result-prefixes CDATA #IMPLIED
   4668 		    *  TODO: xsl:use-attribute-sets %qnames; #IMPLIED
   4669 		    *  xsl:version NMTOKEN #IMPLIED
   4670 		    */
   4671 		    cur->psvi = NULL;
   4672 		    cctxt->inode->category = XSLT_ELEMENT_CATEGORY_LRE;
   4673 		    if (cur->properties != NULL) {
   4674 			xmlAttrPtr attr = cur->properties;
   4675 			/*
   4676 			* Attribute "xsl:exclude-result-prefixes".
   4677 			*/
   4678 			cctxt->inode->exclResultNs =
   4679 			    xsltParseExclResultPrefixes(cctxt, cur,
   4680 				cctxt->inode->exclResultNs,
   4681 				XSLT_ELEMENT_CATEGORY_LRE);
   4682 			/*
   4683 			* Attribute "xsl:version".
   4684 			*/
   4685 			xsltParseAttrXSLTVersion(cctxt, cur,
   4686 			    XSLT_ELEMENT_CATEGORY_LRE);
   4687 			/*
   4688 			* Report invalid XSLT attributes.
   4689 			* For XSLT 1.0 only xsl:use-attribute-sets is allowed
   4690 			* next to xsl:version, xsl:exclude-result-prefixes and
   4691 			* xsl:extension-element-prefixes.
   4692 			*
   4693 			* Mark all XSLT attributes, in order to skip such
   4694 			* attributes when instantiating the LRE.
   4695 			*/
   4696 			do {
   4697 			    if ((attr->psvi != xsltXSLTAttrMarker) &&
   4698 				IS_XSLT_ATTR_FAST(attr))
   4699 			    {
   4700 				if (! xmlStrEqual(attr->name,
   4701 				    BAD_CAST "use-attribute-sets"))
   4702 				{
   4703 				    xsltTransformError(NULL, cctxt->style,
   4704 					cur,
   4705 					"Unknown XSLT attribute '%s'.\n",
   4706 					attr->name);
   4707 				    cctxt->style->errors++;
   4708 				} else {
   4709 				    /*
   4710 				    * XSLT attr marker.
   4711 				    */
   4712 				    attr->psvi = (void *) xsltXSLTAttrMarker;
   4713 				}
   4714 			    }
   4715 			    attr = attr->next;
   4716 			} while (attr != NULL);
   4717 		    }
   4718 		    /*
   4719 		    * Create/reuse info for the literal result element.
   4720 		    */
   4721 		    if (cctxt->inode->nsChanged)
   4722 			xsltLREInfoCreate(cctxt, cur, 1);
   4723 		    cur->psvi = cctxt->inode->litResElemInfo;
   4724 		    /*
   4725 		    * Apply ns-aliasing on the element and on its attributes.
   4726 		    */
   4727 		    if (cctxt->hasNsAliases)
   4728 			xsltLREBuildEffectiveNs(cctxt, cur);
   4729 		    /*
   4730 		    * Compile attribute value templates (AVT).
   4731 		    */
   4732 		    if (cur->properties) {
   4733 			xmlAttrPtr attr = cur->properties;
   4734 
   4735 			while (attr != NULL) {
   4736 			    xsltCompileAttr(cctxt->style, attr);
   4737 			    attr = attr->next;
   4738 			}
   4739 		    }
   4740 		    /*
   4741 		    * Parse the content, which is defined to be a "template"
   4742 		    * (or "sequence constructor" in XSLT 2.0 terms).
   4743 		    */
   4744 		    if (cur->children != NULL) {
   4745 			xsltParseSequenceConstructor(cctxt, cur->children);
   4746 		    }
   4747 		}
   4748 		/*
   4749 		* Leave the non-XSLT element.
   4750 		*/
   4751 		xsltCompilerNodePop(cctxt, cur);
   4752 	    }
   4753 	}
   4754 	cur = cur->next;
   4755     }
   4756     if (deleteNode != NULL) {
   4757 #ifdef WITH_XSLT_DEBUG_BLANKS
   4758 	xsltGenericDebug(xsltGenericDebugContext,
   4759 	    "xsltParseSequenceConstructor: removing xsl:text element\n");
   4760 #endif
   4761 	xmlUnlinkNode(deleteNode);
   4762 	xmlFreeNode(deleteNode);
   4763 	deleteNode = NULL;
   4764     }
   4765 }
   4766 
   4767 /**
   4768  * xsltParseTemplateContent:
   4769  * @style:  the XSLT stylesheet
   4770  * @templ:  the node containing the content to be parsed
   4771  *
   4772  * Parses and compiles the content-model of an xsl:template element.
   4773  * Note that this is *not* the "template" content model (or "sequence
   4774  *  constructor" in XSLT 2.0); it it allows addional xsl:param
   4775  *  elements as immediate children of @templ.
   4776  *
   4777  * Called by:
   4778  *   exsltFuncFunctionComp() (EXSLT, functions.c)
   4779  *   So this is intended to be called from extension functions.
   4780  */
   4781 void
   4782 xsltParseTemplateContent(xsltStylesheetPtr style, xmlNodePtr templ) {
   4783     if ((style == NULL) || (templ == NULL))
   4784 	return;
   4785 
   4786     /*
   4787     * Detection of handled content of extension instructions.
   4788     */
   4789     if (XSLT_CCTXT(style)->inode->category == XSLT_ELEMENT_CATEGORY_EXTENSION) {
   4790 	XSLT_CCTXT(style)->inode->extContentHandled = 1;
   4791     }
   4792 
   4793     if (templ->children != NULL) {
   4794 	xmlNodePtr child = templ->children;
   4795 	/*
   4796 	* Process xsl:param elements, which can only occur as the
   4797 	* immediate children of xsl:template (well, and of any
   4798 	* user-defined extension instruction if needed).
   4799 	*/
   4800 	do {
   4801 	    if ((child->type == XML_ELEMENT_NODE) &&
   4802 		IS_XSLT_ELEM_FAST(child) &&
   4803 		IS_XSLT_NAME(child, "param"))
   4804 	    {
   4805 		XSLT_CCTXT(style)->inode->curChildType = XSLT_FUNC_PARAM;
   4806 		xsltParseAnyXSLTElem(XSLT_CCTXT(style), child);
   4807 	    } else
   4808 		break;
   4809 	    child = child->next;
   4810 	} while (child != NULL);
   4811 	/*
   4812 	* Parse the content and register the pattern.
   4813 	*/
   4814 	xsltParseSequenceConstructor(XSLT_CCTXT(style), child);
   4815     }
   4816 }
   4817 
   4818 #else /* XSLT_REFACTORED */
   4819 
   4820 /**
   4821  * xsltParseTemplateContent:
   4822  * @style:  the XSLT stylesheet
   4823  * @templ:  the container node (can be a document for literal results)
   4824  *
   4825  * parse a template content-model
   4826  * Clean-up the template content from unwanted ignorable blank nodes
   4827  * and process xslt:text
   4828  */
   4829 void
   4830 xsltParseTemplateContent(xsltStylesheetPtr style, xmlNodePtr templ) {
   4831     xmlNodePtr cur, delete;
   4832     /*
   4833      * This content comes from the stylesheet
   4834      * For stylesheets, the set of whitespace-preserving
   4835      * element names consists of just xsl:text.
   4836      */
   4837     cur = templ->children;
   4838     delete = NULL;
   4839     while (cur != NULL) {
   4840 	if (delete != NULL) {
   4841 #ifdef WITH_XSLT_DEBUG_BLANKS
   4842 	    xsltGenericDebug(xsltGenericDebugContext,
   4843 	     "xsltParseTemplateContent: removing text\n");
   4844 #endif
   4845 	    xmlUnlinkNode(delete);
   4846 	    xmlFreeNode(delete);
   4847 	    delete = NULL;
   4848 	}
   4849 	if (IS_XSLT_ELEM(cur)) {
   4850 	    if (IS_XSLT_NAME(cur, "text")) {
   4851 		/*
   4852 		* TODO: Processing of xsl:text should be moved to
   4853 		*   xsltPrecomputeStylesheet(), since otherwise this
   4854 		*   will be performed for every multiply included
   4855 		*   stylesheet; i.e. this here is not skipped with
   4856 		*   the use of the style->nopreproc flag.
   4857 		*/
   4858 		if (cur->children != NULL) {
   4859 		    xmlChar *prop;
   4860 		    xmlNodePtr text = cur->children, next;
   4861 		    int noesc = 0;
   4862 
   4863 		    prop = xmlGetNsProp(cur,
   4864 			(const xmlChar *)"disable-output-escaping",
   4865 			NULL);
   4866 		    if (prop != NULL) {
   4867 #ifdef WITH_XSLT_DEBUG_PARSING
   4868 			xsltGenericDebug(xsltGenericDebugContext,
   4869 			     "Disable escaping: %s\n", text->content);
   4870 #endif
   4871 			if (xmlStrEqual(prop, (const xmlChar *)"yes")) {
   4872 			    noesc = 1;
   4873 			} else if (!xmlStrEqual(prop,
   4874 						(const xmlChar *)"no")){
   4875 			    xsltTransformError(NULL, style, cur,
   4876 	     "xsl:text: disable-output-escaping allows only yes or no\n");
   4877 			    style->warnings++;
   4878 
   4879 			}
   4880 			xmlFree(prop);
   4881 		    }
   4882 
   4883 		    while (text != NULL) {
   4884 			if (text->type == XML_COMMENT_NODE) {
   4885 			    text = text->next;
   4886 			    continue;
   4887 			}
   4888 			if ((text->type != XML_TEXT_NODE) &&
   4889 			     (text->type != XML_CDATA_SECTION_NODE)) {
   4890 			    xsltTransformError(NULL, style, cur,
   4891 		 "xsltParseTemplateContent: xslt:text content problem\n");
   4892 			    style->errors++;
   4893 			    break;
   4894 			}
   4895 			if ((noesc) && (text->type != XML_CDATA_SECTION_NODE))
   4896 			    text->name = xmlStringTextNoenc;
   4897 			text = text->next;
   4898 		    }
   4899 
   4900 		    /*
   4901 		     * replace xsl:text by the list of childs
   4902 		     */
   4903 		    if (text == NULL) {
   4904 			text = cur->children;
   4905 			while (text != NULL) {
   4906 			    if ((style->internalized) &&
   4907 			        (text->content != NULL) &&
   4908 			        (!xmlDictOwns(style->dict, text->content))) {
   4909 
   4910 				/*
   4911 				 * internalize the text string
   4912 				 */
   4913 				if (text->doc->dict != NULL) {
   4914 				    const xmlChar *tmp;
   4915 
   4916 				    tmp = xmlDictLookup(text->doc->dict,
   4917 				                        text->content, -1);
   4918 				    if (tmp != text->content) {
   4919 				        xmlNodeSetContent(text, NULL);
   4920 					text->content = (xmlChar *) tmp;
   4921 				    }
   4922 				}
   4923 			    }
   4924 
   4925 			    next = text->next;
   4926 			    xmlUnlinkNode(text);
   4927 			    xmlAddPrevSibling(cur, text);
   4928 			    text = next;
   4929 			}
   4930 		    }
   4931 		}
   4932 		delete = cur;
   4933 		goto skip_children;
   4934 	    }
   4935 	}
   4936 	else if ((cur->ns != NULL) && (style->nsDefs != NULL) &&
   4937 	    (xsltCheckExtPrefix(style, cur->ns->prefix)))
   4938 	{
   4939 	    /*
   4940 	     * okay this is an extension element compile it too
   4941 	     */
   4942 	    xsltStylePreCompute(style, cur);
   4943 	}
   4944 	else if (cur->type == XML_ELEMENT_NODE)
   4945 	{
   4946 	    /*
   4947 	     * This is an element which will be output as part of the
   4948 	     * template exectution, precompile AVT if found.
   4949 	     */
   4950 	    if ((cur->ns == NULL) && (style->defaultAlias != NULL)) {
   4951 		cur->ns = xmlSearchNsByHref(cur->doc, cur,
   4952 			style->defaultAlias);
   4953 	    }
   4954 	    if (cur->properties != NULL) {
   4955 	        xmlAttrPtr attr = cur->properties;
   4956 
   4957 		while (attr != NULL) {
   4958 		    xsltCompileAttr(style, attr);
   4959 		    attr = attr->next;
   4960 		}
   4961 	    }
   4962 	}
   4963 	/*
   4964 	 * Skip to next node
   4965 	 */
   4966 	if (cur->children != NULL) {
   4967 	    if (cur->children->type != XML_ENTITY_DECL) {
   4968 		cur = cur->children;
   4969 		continue;
   4970 	    }
   4971 	}
   4972 skip_children:
   4973 	if (cur->next != NULL) {
   4974 	    cur = cur->next;
   4975 	    continue;
   4976 	}
   4977 
   4978 	do {
   4979 	    cur = cur->parent;
   4980 	    if (cur == NULL)
   4981 		break;
   4982 	    if (cur == templ) {
   4983 		cur = NULL;
   4984 		break;
   4985 	    }
   4986 	    if (cur->next != NULL) {
   4987 		cur = cur->next;
   4988 		break;
   4989 	    }
   4990 	} while (cur != NULL);
   4991     }
   4992     if (delete != NULL) {
   4993 #ifdef WITH_XSLT_DEBUG_PARSING
   4994 	xsltGenericDebug(xsltGenericDebugContext,
   4995 	 "xsltParseTemplateContent: removing text\n");
   4996 #endif
   4997 	xmlUnlinkNode(delete);
   4998 	xmlFreeNode(delete);
   4999 	delete = NULL;
   5000     }
   5001 
   5002     /*
   5003      * Skip the first params
   5004      */
   5005     cur = templ->children;
   5006     while (cur != NULL) {
   5007 	if ((IS_XSLT_ELEM(cur)) && (!(IS_XSLT_NAME(cur, "param"))))
   5008 	    break;
   5009 	cur = cur->next;
   5010     }
   5011 
   5012     /*
   5013      * Browse the remainder of the template
   5014      */
   5015     while (cur != NULL) {
   5016 	if ((IS_XSLT_ELEM(cur)) && (IS_XSLT_NAME(cur, "param"))) {
   5017 	    xmlNodePtr param = cur;
   5018 
   5019 	    xsltTransformError(NULL, style, cur,
   5020 		"xsltParseTemplateContent: ignoring misplaced param element\n");
   5021 	    if (style != NULL) style->warnings++;
   5022             cur = cur->next;
   5023 	    xmlUnlinkNode(param);
   5024 	    xmlFreeNode(param);
   5025 	} else
   5026 	    break;
   5027     }
   5028 }
   5029 
   5030 #endif /* else XSLT_REFACTORED */
   5031 
   5032 /**
   5033  * xsltParseStylesheetKey:
   5034  * @style:  the XSLT stylesheet
   5035  * @key:  the "key" element
   5036  *
   5037  * <!-- Category: top-level-element -->
   5038  * <xsl:key name = qname, match = pattern, use = expression />
   5039  *
   5040  * parse an XSLT stylesheet key definition and register it
   5041  */
   5042 
   5043 static void
   5044 xsltParseStylesheetKey(xsltStylesheetPtr style, xmlNodePtr key) {
   5045     xmlChar *prop = NULL;
   5046     xmlChar *use = NULL;
   5047     xmlChar *match = NULL;
   5048     xmlChar *name = NULL;
   5049     xmlChar *nameURI = NULL;
   5050 
   5051     if ((style == NULL) || (key == NULL))
   5052 	return;
   5053 
   5054     /*
   5055      * Get arguments
   5056      */
   5057     prop = xmlGetNsProp(key, (const xmlChar *)"name", NULL);
   5058     if (prop != NULL) {
   5059         const xmlChar *URI;
   5060 
   5061 	/*
   5062 	* TODO: Don't use xsltGetQNameURI().
   5063 	*/
   5064 	URI = xsltGetQNameURI(key, &prop);
   5065 	if (prop == NULL) {
   5066 	    if (style != NULL) style->errors++;
   5067 	    goto error;
   5068 	} else {
   5069 	    name = prop;
   5070 	    if (URI != NULL)
   5071 		nameURI = xmlStrdup(URI);
   5072 	}
   5073 #ifdef WITH_XSLT_DEBUG_PARSING
   5074 	xsltGenericDebug(xsltGenericDebugContext,
   5075 	     "xsltParseStylesheetKey: name %s\n", name);
   5076 #endif
   5077     } else {
   5078 	xsltTransformError(NULL, style, key,
   5079 	    "xsl:key : error missing name\n");
   5080 	if (style != NULL) style->errors++;
   5081 	goto error;
   5082     }
   5083 
   5084     match = xmlGetNsProp(key, (const xmlChar *)"match", NULL);
   5085     if (match == NULL) {
   5086 	xsltTransformError(NULL, style, key,
   5087 	    "xsl:key : error missing match\n");
   5088 	if (style != NULL) style->errors++;
   5089 	goto error;
   5090     }
   5091 
   5092     use = xmlGetNsProp(key, (const xmlChar *)"use", NULL);
   5093     if (use == NULL) {
   5094 	xsltTransformError(NULL, style, key,
   5095 	    "xsl:key : error missing use\n");
   5096 	if (style != NULL) style->errors++;
   5097 	goto error;
   5098     }
   5099 
   5100     /*
   5101      * register the keys
   5102      */
   5103     xsltAddKey(style, name, nameURI, match, use, key);
   5104 
   5105 
   5106 error:
   5107     if (use != NULL)
   5108 	xmlFree(use);
   5109     if (match != NULL)
   5110 	xmlFree(match);
   5111     if (name != NULL)
   5112 	xmlFree(name);
   5113     if (nameURI != NULL)
   5114 	xmlFree(nameURI);
   5115 
   5116     if (key->children != NULL) {
   5117 	xsltParseContentError(style, key->children);
   5118     }
   5119 }
   5120 
   5121 #ifdef XSLT_REFACTORED
   5122 /**
   5123  * xsltParseXSLTTemplate:
   5124  * @style:  the XSLT stylesheet
   5125  * @template:  the "template" element
   5126  *
   5127  * parse an XSLT stylesheet template building the associated structures
   5128  * TODO: Is @style ever expected to be NULL?
   5129  *
   5130  * Called from:
   5131  *   xsltParseXSLTStylesheet()
   5132  *   xsltParseStylesheetTop()
   5133  */
   5134 
   5135 static void
   5136 xsltParseXSLTTemplate(xsltCompilerCtxtPtr cctxt, xmlNodePtr templNode) {
   5137     xsltTemplatePtr templ;
   5138     xmlChar *prop;
   5139     double  priority;
   5140 
   5141     if ((cctxt == NULL) || (templNode == NULL))
   5142 	return;
   5143 
   5144     /*
   5145      * Create and link the structure
   5146      */
   5147     templ = xsltNewTemplate();
   5148     if (templ == NULL)
   5149 	return;
   5150 
   5151     xsltCompilerNodePush(cctxt, templNode);
   5152     if (templNode->nsDef != NULL)
   5153 	cctxt->inode->inScopeNs =
   5154 	    xsltCompilerBuildInScopeNsList(cctxt, templNode);
   5155 
   5156     templ->next = cctxt->style->templates;
   5157     cctxt->style->templates = templ;
   5158     templ->style = cctxt->style;
   5159 
   5160     /*
   5161     * Attribute "mode".
   5162     */
   5163     prop = xmlGetNsProp(templNode, (const xmlChar *)"mode", NULL);
   5164     if (prop != NULL) {
   5165         const xmlChar *modeURI;
   5166 
   5167 	/*
   5168 	* TODO: We need a standardized function for extraction
   5169 	*  of namespace names and local names from QNames.
   5170 	*  Don't use xsltGetQNameURI() as it cannot channe
   5171 	*  reports through the context.
   5172 	*/
   5173 	modeURI = xsltGetQNameURI(templNode, &prop);
   5174 	if (prop == NULL) {
   5175 	    cctxt->style->errors++;
   5176 	    goto error;
   5177 	}
   5178 	templ->mode = xmlDictLookup(cctxt->style->dict, prop, -1);
   5179 	xmlFree(prop);
   5180 	prop = NULL;
   5181 	if (xmlValidateNCName(templ->mode, 0)) {
   5182 	    xsltTransformError(NULL, cctxt->style, templNode,
   5183 		"xsl:template: Attribute 'mode': The local part '%s' "
   5184 		"of the value is not a valid NCName.\n", templ->name);
   5185 	    cctxt->style->errors++;
   5186 	    goto error;
   5187 	}
   5188 	if (modeURI != NULL)
   5189 	    templ->modeURI = xmlDictLookup(cctxt->style->dict, modeURI, -1);
   5190 #ifdef WITH_XSLT_DEBUG_PARSING
   5191 	xsltGenericDebug(xsltGenericDebugContext,
   5192 	     "xsltParseXSLTTemplate: mode %s\n", templ->mode);
   5193 #endif
   5194     }
   5195     /*
   5196     * Attribute "match".
   5197     */
   5198     prop = xmlGetNsProp(templNode, (const xmlChar *)"match", NULL);
   5199     if (prop != NULL) {
   5200 	templ->match  = prop;
   5201 	prop = NULL;
   5202     }
   5203     /*
   5204     * Attribute "priority".
   5205     */
   5206     prop = xmlGetNsProp(templNode, (const xmlChar *)"priority", NULL);
   5207     if (prop != NULL) {
   5208 	priority = xmlXPathStringEvalNumber(prop);
   5209 	templ->priority = (float) priority;
   5210 	xmlFree(prop);
   5211 	prop = NULL;
   5212     }
   5213     /*
   5214     * Attribute "name".
   5215     */
   5216     prop = xmlGetNsProp(templNode, (const xmlChar *)"name", NULL);
   5217     if (prop != NULL) {
   5218         const xmlChar *nameURI;
   5219 	xsltTemplatePtr curTempl;
   5220 
   5221 	/*
   5222 	* TODO: Don't use xsltGetQNameURI().
   5223 	*/
   5224 	nameURI = xsltGetQNameURI(templNode, &prop);
   5225 	if (prop == NULL) {
   5226 	    cctxt->style->errors++;
   5227 	    goto error;
   5228 	}
   5229 	templ->name = xmlDictLookup(cctxt->style->dict, prop, -1);
   5230 	xmlFree(prop);
   5231 	prop = NULL;
   5232 	if (xmlValidateNCName(templ->name, 0)) {
   5233 	    xsltTransformError(NULL, cctxt->style, templNode,
   5234 		"xsl:template: Attribute 'name': The local part '%s' of "
   5235 		"the value is not a valid NCName.\n", templ->name);
   5236 	    cctxt->style->errors++;
   5237 	    goto error;
   5238 	}
   5239 	if (nameURI != NULL)
   5240 	    templ->nameURI = xmlDictLookup(cctxt->style->dict, nameURI, -1);
   5241 	curTempl = templ->next;
   5242 	while (curTempl != NULL) {
   5243 	    if ((nameURI != NULL && xmlStrEqual(curTempl->name, templ->name) &&
   5244 		xmlStrEqual(curTempl->nameURI, nameURI) ) ||
   5245 		(nameURI == NULL && curTempl->nameURI == NULL &&
   5246 		xmlStrEqual(curTempl->name, templ->name)))
   5247 	    {
   5248 		xsltTransformError(NULL, cctxt->style, templNode,
   5249 		    "xsl:template: error duplicate name '%s'\n", templ->name);
   5250 		cctxt->style->errors++;
   5251 		goto error;
   5252 	    }
   5253 	    curTempl = curTempl->next;
   5254 	}
   5255     }
   5256     if (templNode->children != NULL) {
   5257 	xsltParseTemplateContent(cctxt->style, templNode);
   5258 	/*
   5259 	* MAYBE TODO: Custom behaviour: In order to stay compatible with
   5260 	* Xalan and MSXML(.NET), we could allow whitespace
   5261 	* to appear before an xml:param element; this whitespace
   5262 	* will additionally become part of the "template".
   5263 	* NOTE that this is totally deviates from the spec, but
   5264 	* is the de facto behaviour of Xalan and MSXML(.NET).
   5265 	* Personally I wouldn't allow this, since if we have:
   5266 	* <xsl:template ...xml:space="preserve">
   5267 	*   <xsl:param name="foo"/>
   5268 	*   <xsl:param name="bar"/>
   5269 	*   <xsl:param name="zoo"/>
   5270 	* ... the whitespace between every xsl:param would be
   5271 	* added to the result tree.
   5272 	*/
   5273     }
   5274 
   5275     templ->elem = templNode;
   5276     templ->content = templNode->children;
   5277     xsltAddTemplate(cctxt->style, templ, templ->mode, templ->modeURI);
   5278 
   5279 error:
   5280     xsltCompilerNodePop(cctxt, templNode);
   5281     return;
   5282 }
   5283 
   5284 #else /* XSLT_REFACTORED */
   5285 
   5286 /**
   5287  * xsltParseStylesheetTemplate:
   5288  * @style:  the XSLT stylesheet
   5289  * @template:  the "template" element
   5290  *
   5291  * parse an XSLT stylesheet template building the associated structures
   5292  */
   5293 
   5294 static void
   5295 xsltParseStylesheetTemplate(xsltStylesheetPtr style, xmlNodePtr template) {
   5296     xsltTemplatePtr ret;
   5297     xmlChar *prop;
   5298     xmlChar *mode = NULL;
   5299     xmlChar *modeURI = NULL;
   5300     double  priority;
   5301 
   5302     if (template == NULL)
   5303 	return;
   5304 
   5305     /*
   5306      * Create and link the structure
   5307      */
   5308     ret = xsltNewTemplate();
   5309     if (ret == NULL)
   5310 	return;
   5311     ret->next = style->templates;
   5312     style->templates = ret;
   5313     ret->style = style;
   5314 
   5315     /*
   5316      * Get inherited namespaces
   5317      */
   5318     /*
   5319     * TODO: Apply the optimized in-scope-namespace mechanism
   5320     *   as for the other XSLT instructions.
   5321     */
   5322     xsltGetInheritedNsList(style, ret, template);
   5323 
   5324     /*
   5325      * Get arguments
   5326      */
   5327     prop = xmlGetNsProp(template, (const xmlChar *)"mode", NULL);
   5328     if (prop != NULL) {
   5329         const xmlChar *URI;
   5330 
   5331 	/*
   5332 	* TODO: Don't use xsltGetQNameURI().
   5333 	*/
   5334 	URI = xsltGetQNameURI(template, &prop);
   5335 	if (prop == NULL) {
   5336 	    if (style != NULL) style->errors++;
   5337 	    goto error;
   5338 	} else {
   5339 	    mode = prop;
   5340 	    if (URI != NULL)
   5341 		modeURI = xmlStrdup(URI);
   5342 	}
   5343 	ret->mode = xmlDictLookup(style->dict, mode, -1);
   5344 	ret->modeURI = xmlDictLookup(style->dict, modeURI, -1);
   5345 #ifdef WITH_XSLT_DEBUG_PARSING
   5346 	xsltGenericDebug(xsltGenericDebugContext,
   5347 	     "xsltParseStylesheetTemplate: mode %s\n", mode);
   5348 #endif
   5349         if (mode != NULL) xmlFree(mode);
   5350 	if (modeURI != NULL) xmlFree(modeURI);
   5351     }
   5352     prop = xmlGetNsProp(template, (const xmlChar *)"match", NULL);
   5353     if (prop != NULL) {
   5354 	if (ret->match != NULL) xmlFree(ret->match);
   5355 	ret->match  = prop;
   5356     }
   5357 
   5358     prop = xmlGetNsProp(template, (const xmlChar *)"priority", NULL);
   5359     if (prop != NULL) {
   5360 	priority = xmlXPathStringEvalNumber(prop);
   5361 	ret->priority = (float) priority;
   5362 	xmlFree(prop);
   5363     }
   5364 
   5365     prop = xmlGetNsProp(template, (const xmlChar *)"name", NULL);
   5366     if (prop != NULL) {
   5367         const xmlChar *URI;
   5368 	xsltTemplatePtr cur;
   5369 
   5370 	/*
   5371 	* TODO: Don't use xsltGetQNameURI().
   5372 	*/
   5373 	URI = xsltGetQNameURI(template, &prop);
   5374 	if (prop == NULL) {
   5375 	    if (style != NULL) style->errors++;
   5376 	    goto error;
   5377 	} else {
   5378 	    if (xmlValidateNCName(prop,0)) {
   5379 	        xsltTransformError(NULL, style, template,
   5380 	            "xsl:template : error invalid name '%s'\n", prop);
   5381 		if (style != NULL) style->errors++;
   5382 		goto error;
   5383 	    }
   5384 	    ret->name = xmlDictLookup(style->dict, BAD_CAST prop, -1);
   5385 	    xmlFree(prop);
   5386 	    prop = NULL;
   5387 	    if (URI != NULL)
   5388 		ret->nameURI = xmlDictLookup(style->dict, BAD_CAST URI, -1);
   5389 	    else
   5390 		ret->nameURI = NULL;
   5391 	    cur = ret->next;
   5392 	    while (cur != NULL) {
   5393 	        if ((URI != NULL && xmlStrEqual(cur->name, ret->name) &&
   5394 				xmlStrEqual(cur->nameURI, URI) ) ||
   5395 		    (URI == NULL && cur->nameURI == NULL &&
   5396 				xmlStrEqual(cur->name, ret->name))) {
   5397 		    xsltTransformError(NULL, style, template,
   5398 		        "xsl:template: error duplicate name '%s'\n", ret->name);
   5399 		    style->errors++;
   5400 		    goto error;
   5401 		}
   5402 		cur = cur->next;
   5403 	    }
   5404 	}
   5405     }
   5406 
   5407     /*
   5408      * parse the content and register the pattern
   5409      */
   5410     xsltParseTemplateContent(style, template);
   5411     ret->elem = template;
   5412     ret->content = template->children;
   5413     xsltAddTemplate(style, ret, ret->mode, ret->modeURI);
   5414 
   5415 error:
   5416     return;
   5417 }
   5418 
   5419 #endif /* else XSLT_REFACTORED */
   5420 
   5421 #ifdef XSLT_REFACTORED
   5422 
   5423 /**
   5424  * xsltIncludeComp:
   5425  * @cctxt: the compilation contenxt
   5426  * @node:  the xsl:include node
   5427  *
   5428  * Process the xslt include node on the source node
   5429  */
   5430 static xsltStyleItemIncludePtr
   5431 xsltCompileXSLTIncludeElem(xsltCompilerCtxtPtr cctxt, xmlNodePtr node) {
   5432     xsltStyleItemIncludePtr item;
   5433 
   5434     if ((cctxt == NULL) || (node == NULL))
   5435 	return(NULL);
   5436 
   5437     node->psvi = NULL;
   5438     item = (xsltStyleItemIncludePtr) xmlMalloc(sizeof(xsltStyleItemInclude));
   5439     if (item == NULL) {
   5440 	xsltTransformError(NULL, cctxt->style, node,
   5441 		"xsltIncludeComp : malloc failed\n");
   5442 	cctxt->style->errors++;
   5443 	return(NULL);
   5444     }
   5445     memset(item, 0, sizeof(xsltStyleItemInclude));
   5446 
   5447     node->psvi = item;
   5448     item->inst = node;
   5449     item->type = XSLT_FUNC_INCLUDE;
   5450 
   5451     item->next = cctxt->style->preComps;
   5452     cctxt->style->preComps = (xsltElemPreCompPtr) item;
   5453 
   5454     return(item);
   5455 }
   5456 
   5457 /**
   5458  * xsltParseFindTopLevelElem:
   5459  */
   5460 static int
   5461 xsltParseFindTopLevelElem(xsltCompilerCtxtPtr cctxt,
   5462 			      xmlNodePtr cur,
   5463 			      const xmlChar *name,
   5464 			      const xmlChar *namespaceURI,
   5465 			      int breakOnOtherElem,
   5466 			      xmlNodePtr *resultNode)
   5467 {
   5468     if (name == NULL)
   5469 	return(-1);
   5470 
   5471     *resultNode = NULL;
   5472     while (cur != NULL) {
   5473 	if (cur->type == XML_ELEMENT_NODE) {
   5474 	    if ((cur->ns != NULL) && (cur->name != NULL)) {
   5475 		if ((*(cur->name) == *name) &&
   5476 		    xmlStrEqual(cur->name, name) &&
   5477 		    xmlStrEqual(cur->ns->href, namespaceURI))
   5478 		{
   5479 		    *resultNode = cur;
   5480 		    return(1);
   5481 		}
   5482 	    }
   5483 	    if (breakOnOtherElem)
   5484 		break;
   5485 	}
   5486 	cur = cur->next;
   5487     }
   5488     *resultNode = cur;
   5489     return(0);
   5490 }
   5491 
   5492 static int
   5493 xsltParseTopLevelXSLTElem(xsltCompilerCtxtPtr cctxt,
   5494 			  xmlNodePtr node,
   5495 			  xsltStyleType type)
   5496 {
   5497     int ret = 0;
   5498 
   5499     /*
   5500     * TODO: The reason why this function exists:
   5501     *  due to historical reasons some of the
   5502     *  top-level declarations are processed by functions
   5503     *  in other files. Since we need still to set
   5504     *  up the node-info and generate information like
   5505     *  in-scope namespaces, this is a wrapper around
   5506     *  those old parsing functions.
   5507     */
   5508     xsltCompilerNodePush(cctxt, node);
   5509     if (node->nsDef != NULL)
   5510 	cctxt->inode->inScopeNs =
   5511 	    xsltCompilerBuildInScopeNsList(cctxt, node);
   5512     cctxt->inode->type = type;
   5513 
   5514     switch (type) {
   5515 	case XSLT_FUNC_INCLUDE:
   5516 	    {
   5517 		int oldIsInclude;
   5518 
   5519 		if (xsltCompileXSLTIncludeElem(cctxt, node) == NULL)
   5520 		    goto exit;
   5521 		/*
   5522 		* Mark this stylesheet tree as being currently included.
   5523 		*/
   5524 		oldIsInclude = cctxt->isInclude;
   5525 		cctxt->isInclude = 1;
   5526 
   5527 		if (xsltParseStylesheetInclude(cctxt->style, node) != 0) {
   5528 		    cctxt->style->errors++;
   5529 		}
   5530 		cctxt->isInclude = oldIsInclude;
   5531 	    }
   5532 	    break;
   5533 	case XSLT_FUNC_PARAM:
   5534 	    xsltStylePreCompute(cctxt->style, node);
   5535 	    xsltParseGlobalParam(cctxt->style, node);
   5536 	    break;
   5537 	case XSLT_FUNC_VARIABLE:
   5538 	    xsltStylePreCompute(cctxt->style, node);
   5539 	    xsltParseGlobalVariable(cctxt->style, node);
   5540 	    break;
   5541 	case XSLT_FUNC_ATTRSET:
   5542 	    xsltParseStylesheetAttributeSet(cctxt->style, node);
   5543 	    break;
   5544 	default:
   5545 	    xsltTransformError(NULL, cctxt->style, node,
   5546 		"Internal error: (xsltParseTopLevelXSLTElem) "
   5547 		"Cannot handle this top-level declaration.\n");
   5548 	    cctxt->style->errors++;
   5549 	    ret = -1;
   5550     }
   5551 
   5552 exit:
   5553     xsltCompilerNodePop(cctxt, node);
   5554 
   5555     return(ret);
   5556 }
   5557 
   5558 #if 0
   5559 static int
   5560 xsltParseRemoveWhitespace(xmlNodePtr node)
   5561 {
   5562     if ((node == NULL) || (node->children == NULL))
   5563 	return(0);
   5564     else {
   5565 	xmlNodePtr delNode = NULL, child = node->children;
   5566 
   5567 	do {
   5568 	    if (delNode) {
   5569 		xmlUnlinkNode(delNode);
   5570 		xmlFreeNode(delNode);
   5571 		delNode = NULL;
   5572 	    }
   5573 	    if (((child->type == XML_TEXT_NODE) ||
   5574 		 (child->type == XML_CDATA_SECTION_NODE)) &&
   5575 		(IS_BLANK_NODE(child)))
   5576 		delNode = child;
   5577 	    child = child->next;
   5578 	} while (child != NULL);
   5579 	if (delNode) {
   5580 	    xmlUnlinkNode(delNode);
   5581 	    xmlFreeNode(delNode);
   5582 	    delNode = NULL;
   5583 	}
   5584     }
   5585     return(0);
   5586 }
   5587 #endif
   5588 
   5589 static int
   5590 xsltParseXSLTStylesheetElemCore(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   5591 {
   5592 #ifdef WITH_XSLT_DEBUG_PARSING
   5593     int templates = 0;
   5594 #endif
   5595     xmlNodePtr cur, start = NULL;
   5596     xsltStylesheetPtr style;
   5597 
   5598     if ((cctxt == NULL) || (node == NULL) ||
   5599 	(node->type != XML_ELEMENT_NODE))
   5600 	return(-1);
   5601 
   5602     style = cctxt->style;
   5603     /*
   5604     * At this stage all import declarations of all stylesheet modules
   5605     * with the same stylesheet level have been processed.
   5606     * Now we can safely parse the rest of the declarations.
   5607     */
   5608     if (IS_XSLT_ELEM_FAST(node) && IS_XSLT_NAME(node, "include"))
   5609     {
   5610 	xsltDocumentPtr include;
   5611 	/*
   5612 	* URGENT TODO: Make this work with simplified stylesheets!
   5613 	*   I.e., when we won't find an xsl:stylesheet element.
   5614 	*/
   5615 	/*
   5616 	* This is as include declaration.
   5617 	*/
   5618 	include = ((xsltStyleItemIncludePtr) node->psvi)->include;
   5619 	if (include == NULL) {
   5620 	    /* TODO: raise error? */
   5621 	    return(-1);
   5622 	}
   5623 	/*
   5624 	* TODO: Actually an xsl:include should locate an embedded
   5625 	*  stylesheet as well; so the document-element won't always
   5626 	*  be the element where the actual stylesheet is rooted at.
   5627 	*  But such embedded stylesheets are not supported by Libxslt yet.
   5628 	*/
   5629 	node = xmlDocGetRootElement(include->doc);
   5630 	if (node == NULL) {
   5631 	    return(-1);
   5632 	}
   5633     }
   5634 
   5635     if (node->children == NULL)
   5636 	return(0);
   5637     /*
   5638     * Push the xsl:stylesheet/xsl:transform element.
   5639     */
   5640     xsltCompilerNodePush(cctxt, node);
   5641     cctxt->inode->isRoot = 1;
   5642     cctxt->inode->nsChanged = 0;
   5643     /*
   5644     * Start with the naked dummy info for literal result elements.
   5645     */
   5646     cctxt->inode->litResElemInfo = cctxt->inodeList->litResElemInfo;
   5647 
   5648     /*
   5649     * In every case, we need to have
   5650     * the in-scope namespaces of the element, where the
   5651     * stylesheet is rooted at, regardless if it's an XSLT
   5652     * instruction or a literal result instruction (or if
   5653     * this is an embedded stylesheet).
   5654     */
   5655     cctxt->inode->inScopeNs =
   5656 	xsltCompilerBuildInScopeNsList(cctxt, node);
   5657 
   5658     /*
   5659     * Process attributes of xsl:stylesheet/xsl:transform.
   5660     * --------------------------------------------------
   5661     * Allowed are:
   5662     *  id = id
   5663     *  extension-element-prefixes = tokens
   5664     *  exclude-result-prefixes = tokens
   5665     *  version = number (mandatory)
   5666     */
   5667     if (xsltParseAttrXSLTVersion(cctxt, node,
   5668 	XSLT_ELEMENT_CATEGORY_XSLT) == 0)
   5669     {
   5670 	/*
   5671 	* Attribute "version".
   5672 	* XSLT 1.0: "An xsl:stylesheet element *must* have a version
   5673 	*  attribute, indicating the version of XSLT that the
   5674 	*  stylesheet requires".
   5675 	* The root element of a simplified stylesheet must also have
   5676 	* this attribute.
   5677 	*/
   5678 #ifdef XSLT_REFACTORED_MANDATORY_VERSION
   5679 	if (isXsltElem)
   5680 	    xsltTransformError(NULL, cctxt->style, node,
   5681 		"The attribute 'version' is missing.\n");
   5682 	cctxt->style->errors++;
   5683 #else
   5684 	/* OLD behaviour. */
   5685 	xsltTransformError(NULL, cctxt->style, node,
   5686 	    "xsl:version is missing: document may not be a stylesheet\n");
   5687 	cctxt->style->warnings++;
   5688 #endif
   5689     }
   5690     /*
   5691     * The namespaces declared by the attributes
   5692     *  "extension-element-prefixes" and
   5693     *  "exclude-result-prefixes" are local to *this*
   5694     *  stylesheet tree; i.e., they are *not* visible to
   5695     *  other stylesheet-modules, whether imported or included.
   5696     *
   5697     * Attribute "extension-element-prefixes".
   5698     */
   5699     cctxt->inode->extElemNs =
   5700 	xsltParseExtElemPrefixes(cctxt, node, NULL,
   5701 	    XSLT_ELEMENT_CATEGORY_XSLT);
   5702     /*
   5703     * Attribute "exclude-result-prefixes".
   5704     */
   5705     cctxt->inode->exclResultNs =
   5706 	xsltParseExclResultPrefixes(cctxt, node, NULL,
   5707 	    XSLT_ELEMENT_CATEGORY_XSLT);
   5708     /*
   5709     * Create/reuse info for the literal result element.
   5710     */
   5711     if (cctxt->inode->nsChanged)
   5712 	xsltLREInfoCreate(cctxt, node, 0);
   5713     /*
   5714     * Processed top-level elements:
   5715     * ----------------------------
   5716     *  xsl:variable, xsl:param (QName, in-scope ns,
   5717     *    expression (vars allowed))
   5718     *  xsl:attribute-set (QName, in-scope ns)
   5719     *  xsl:strip-space, xsl:preserve-space (XPath NameTests,
   5720     *    in-scope ns)
   5721     *    I *think* global scope, merge with includes
   5722     *  xsl:output (QName, in-scope ns)
   5723     *  xsl:key (QName, in-scope ns, pattern,
   5724     *    expression (vars *not* allowed))
   5725     *  xsl:decimal-format (QName, needs in-scope ns)
   5726     *  xsl:namespace-alias (in-scope ns)
   5727     *    global scope, merge with includes
   5728     *  xsl:template (last, QName, pattern)
   5729     *
   5730     * (whitespace-only text-nodes have *not* been removed
   5731     *  yet; this will be done in xsltParseSequenceConstructor)
   5732     *
   5733     * Report misplaced child-nodes first.
   5734     */
   5735     cur = node->children;
   5736     while (cur != NULL) {
   5737 	if (cur->type == XML_TEXT_NODE) {
   5738 	    xsltTransformError(NULL, style, cur,
   5739 		"Misplaced text node (content: '%s').\n",
   5740 		(cur->content != NULL) ? cur->content : BAD_CAST "");
   5741 	    style->errors++;
   5742 	} else if (cur->type != XML_ELEMENT_NODE) {
   5743 	    xsltTransformError(NULL, style, cur, "Misplaced node.\n");
   5744 	    style->errors++;
   5745 	}
   5746 	cur = cur->next;
   5747     }
   5748     /*
   5749     * Skip xsl:import elements; they have been processed
   5750     * already.
   5751     */
   5752     cur = node->children;
   5753     while ((cur != NULL) && xsltParseFindTopLevelElem(cctxt, cur,
   5754 	    BAD_CAST "import", XSLT_NAMESPACE, 1, &cur) == 1)
   5755 	cur = cur->next;
   5756     if (cur == NULL)
   5757 	goto exit;
   5758 
   5759     start = cur;
   5760     /*
   5761     * Process all top-level xsl:param elements.
   5762     */
   5763     while ((cur != NULL) &&
   5764 	xsltParseFindTopLevelElem(cctxt, cur,
   5765 	BAD_CAST "param", XSLT_NAMESPACE, 0, &cur) == 1)
   5766     {
   5767 	xsltParseTopLevelXSLTElem(cctxt, cur, XSLT_FUNC_PARAM);
   5768 	cur = cur->next;
   5769     }
   5770     /*
   5771     * Process all top-level xsl:variable elements.
   5772     */
   5773     cur = start;
   5774     while ((cur != NULL) &&
   5775 	xsltParseFindTopLevelElem(cctxt, cur,
   5776 	BAD_CAST "variable", XSLT_NAMESPACE, 0, &cur) == 1)
   5777     {
   5778 	xsltParseTopLevelXSLTElem(cctxt, cur, XSLT_FUNC_VARIABLE);
   5779 	cur = cur->next;
   5780     }
   5781     /*
   5782     * Process all the rest of top-level elements.
   5783     */
   5784     cur = start;
   5785     while (cur != NULL) {
   5786 	/*
   5787 	* Process element nodes.
   5788 	*/
   5789 	if (cur->type == XML_ELEMENT_NODE) {
   5790 	    if (cur->ns == NULL) {
   5791 		xsltTransformError(NULL, style, cur,
   5792 		    "Unexpected top-level element in no namespace.\n");
   5793 		style->errors++;
   5794 		cur = cur->next;
   5795 		continue;
   5796 	    }
   5797 	    /*
   5798 	    * Process all XSLT elements.
   5799 	    */
   5800 	    if (IS_XSLT_ELEM_FAST(cur)) {
   5801 		/*
   5802 		* xsl:import is only allowed at the beginning.
   5803 		*/
   5804 		if (IS_XSLT_NAME(cur, "import")) {
   5805 		    xsltTransformError(NULL, style, cur,
   5806 			"Misplaced xsl:import element.\n");
   5807 		    style->errors++;
   5808 		    cur = cur->next;
   5809 		    continue;
   5810 		}
   5811 		/*
   5812 		* TODO: Change the return type of the parsing functions
   5813 		*  to int.
   5814 		*/
   5815 		if (IS_XSLT_NAME(cur, "template")) {
   5816 #ifdef WITH_XSLT_DEBUG_PARSING
   5817 		    templates++;
   5818 #endif
   5819 		    /*
   5820 		    * TODO: Is the position of xsl:template in the
   5821 		    *  tree significant? If not it would be easier to
   5822 		    *  parse them at a later stage.
   5823 		    */
   5824 		    xsltParseXSLTTemplate(cctxt, cur);
   5825 		} else if (IS_XSLT_NAME(cur, "variable")) {
   5826 		    /* NOP; done already */
   5827 		} else if (IS_XSLT_NAME(cur, "param")) {
   5828 		    /* NOP; done already */
   5829 		} else if (IS_XSLT_NAME(cur, "include")) {
   5830 		    if (cur->psvi != NULL)
   5831 			xsltParseXSLTStylesheetElemCore(cctxt, cur);
   5832 		    else {
   5833 			xsltTransformError(NULL, style, cur,
   5834 			    "Internal error: "
   5835 			    "(xsltParseXSLTStylesheetElemCore) "
   5836 			    "The xsl:include element was not compiled.\n");
   5837 			style->errors++;
   5838 		    }
   5839 		} else if (IS_XSLT_NAME(cur, "strip-space")) {
   5840 		    /* No node info needed. */
   5841 		    xsltParseStylesheetStripSpace(style, cur);
   5842 		} else if (IS_XSLT_NAME(cur, "preserve-space")) {
   5843 		    /* No node info needed. */
   5844 		    xsltParseStylesheetPreserveSpace(style, cur);
   5845 		} else if (IS_XSLT_NAME(cur, "output")) {
   5846 		    /* No node-info needed. */
   5847 		    xsltParseStylesheetOutput(style, cur);
   5848 		} else if (IS_XSLT_NAME(cur, "key")) {
   5849 		    /* TODO: node-info needed for expressions ? */
   5850 		    xsltParseStylesheetKey(style, cur);
   5851 		} else if (IS_XSLT_NAME(cur, "decimal-format")) {
   5852 		    /* No node-info needed. */
   5853 		    xsltParseStylesheetDecimalFormat(style, cur);
   5854 		} else if (IS_XSLT_NAME(cur, "attribute-set")) {
   5855 		    xsltParseTopLevelXSLTElem(cctxt, cur,
   5856 			XSLT_FUNC_ATTRSET);
   5857 		} else if (IS_XSLT_NAME(cur, "namespace-alias")) {
   5858 		    /* NOP; done already */
   5859 		} else {
   5860 		    if (cctxt->inode->forwardsCompat) {
   5861 			/*
   5862 			* Forwards-compatible mode:
   5863 			*
   5864 			* XSLT-1: "if it is a top-level element and
   5865 			*  XSLT 1.0 does not allow such elements as top-level
   5866 			*  elements, then the element must be ignored along
   5867 			*  with its content;"
   5868 			*/
   5869 			/*
   5870 			* TODO: I don't think we should generate a warning.
   5871 			*/
   5872 			xsltTransformError(NULL, style, cur,
   5873 			    "Forwards-compatible mode: Ignoring unknown XSLT "
   5874 			    "element '%s'.\n", cur->name);
   5875 			style->warnings++;
   5876 		    } else {
   5877 			xsltTransformError(NULL, style, cur,
   5878 			    "Unknown XSLT element '%s'.\n", cur->name);
   5879 			style->errors++;
   5880 		    }
   5881 		}
   5882 	    } else {
   5883 		xsltTopLevelFunction function;
   5884 
   5885 		/*
   5886 		* Process non-XSLT elements, which are in a
   5887 		*  non-NULL namespace.
   5888 		*/
   5889 		/*
   5890 		* QUESTION: What does xsltExtModuleTopLevelLookup()
   5891 		*  do exactly?
   5892 		*/
   5893 		function = xsltExtModuleTopLevelLookup(cur->name,
   5894 		    cur->ns->href);
   5895 		if (function != NULL)
   5896 		    function(style, cur);
   5897 #ifdef WITH_XSLT_DEBUG_PARSING
   5898 		xsltGenericDebug(xsltGenericDebugContext,
   5899 		    "xsltParseXSLTStylesheetElemCore : User-defined "
   5900 		    "data element '%s'.\n", cur->name);
   5901 #endif
   5902 	    }
   5903 	}
   5904 	cur = cur->next;
   5905     }
   5906 
   5907 exit:
   5908 
   5909 #ifdef WITH_XSLT_DEBUG_PARSING
   5910     xsltGenericDebug(xsltGenericDebugContext,
   5911 	"### END of parsing top-level elements of doc '%s'.\n",
   5912 	node->doc->URL);
   5913     xsltGenericDebug(xsltGenericDebugContext,
   5914 	"### Templates: %d\n", templates);
   5915 #ifdef XSLT_REFACTORED
   5916     xsltGenericDebug(xsltGenericDebugContext,
   5917 	"### Max inodes: %d\n", cctxt->maxNodeInfos);
   5918     xsltGenericDebug(xsltGenericDebugContext,
   5919 	"### Max LREs  : %d\n", cctxt->maxLREs);
   5920 #endif /* XSLT_REFACTORED */
   5921 #endif /* WITH_XSLT_DEBUG_PARSING */
   5922 
   5923     xsltCompilerNodePop(cctxt, node);
   5924     return(0);
   5925 }
   5926 
   5927 /**
   5928  * xsltParseXSLTStylesheet:
   5929  * @cctxt: the compiler context
   5930  * @node: the xsl:stylesheet/xsl:transform element-node
   5931  *
   5932  * Parses the xsl:stylesheet and xsl:transform element.
   5933  *
   5934  * <xsl:stylesheet
   5935  *  id = id
   5936  *  extension-element-prefixes = tokens
   5937  *  exclude-result-prefixes = tokens
   5938  *  version = number>
   5939  *  <!-- Content: (xsl:import*, top-level-elements) -->
   5940  * </xsl:stylesheet>
   5941  *
   5942  * BIG TODO: The xsl:include stuff.
   5943  *
   5944  * Called by xsltParseStylesheetTree()
   5945  *
   5946  * Returns 0 on success, a positive result on errors and
   5947  *         -1 on API or internal errors.
   5948  */
   5949 static int
   5950 xsltParseXSLTStylesheetElem(xsltCompilerCtxtPtr cctxt, xmlNodePtr node)
   5951 {
   5952     xmlNodePtr cur, start;
   5953 
   5954     if ((cctxt == NULL) || (node == NULL))
   5955 	return(-1);
   5956 
   5957     if (node->children == NULL)
   5958 	goto exit;
   5959 
   5960     /*
   5961     * Process top-level elements:
   5962     *  xsl:import (must be first)
   5963     *  xsl:include (this is just a pre-processing)
   5964     */
   5965     cur = node->children;
   5966     /*
   5967     * Process xsl:import elements.
   5968     * XSLT 1.0: "The xsl:import element children must precede all
   5969     *  other element children of an xsl:stylesheet element,
   5970     *  including any xsl:include element children."
   5971     */
   5972     while ((cur != NULL) &&
   5973 	xsltParseFindTopLevelElem(cctxt, cur,
   5974 	    BAD_CAST "import", XSLT_NAMESPACE, 1, &cur) == 1)
   5975     {
   5976 	if (xsltParseStylesheetImport(cctxt->style, cur) != 0) {
   5977 	    cctxt->style->errors++;
   5978 	}
   5979 	cur = cur->next;
   5980     }
   5981     if (cur == NULL)
   5982 	goto exit;
   5983     start = cur;
   5984     /*
   5985     * Pre-process all xsl:include elements.
   5986     */
   5987     cur = start;
   5988     while ((cur != NULL) &&
   5989 	xsltParseFindTopLevelElem(cctxt, cur,
   5990 	    BAD_CAST "include", XSLT_NAMESPACE, 0, &cur) == 1)
   5991     {
   5992 	xsltParseTopLevelXSLTElem(cctxt, cur, XSLT_FUNC_INCLUDE);
   5993 	cur = cur->next;
   5994     }
   5995     /*
   5996     * Pre-process all xsl:namespace-alias elements.
   5997     * URGENT TODO: This won't work correctly: the order of included
   5998     *  aliases and aliases defined here is significant.
   5999     */
   6000     cur = start;
   6001     while ((cur != NULL) &&
   6002 	xsltParseFindTopLevelElem(cctxt, cur,
   6003 	    BAD_CAST "namespace-alias", XSLT_NAMESPACE, 0, &cur) == 1)
   6004     {
   6005 	xsltNamespaceAlias(cctxt->style, cur);
   6006 	cur = cur->next;
   6007     }
   6008 
   6009     if (cctxt->isInclude) {
   6010 	/*
   6011 	* If this stylesheet is intended for inclusion, then
   6012 	* we will process only imports and includes.
   6013 	*/
   6014 	goto exit;
   6015     }
   6016     /*
   6017     * Now parse the rest of the top-level elements.
   6018     */
   6019     xsltParseXSLTStylesheetElemCore(cctxt, node);
   6020 exit:
   6021 
   6022     return(0);
   6023 }
   6024 
   6025 #else /* XSLT_REFACTORED */
   6026 
   6027 /**
   6028  * xsltParseStylesheetTop:
   6029  * @style:  the XSLT stylesheet
   6030  * @top:  the top level "stylesheet" or "transform" element
   6031  *
   6032  * scan the top level elements of an XSL stylesheet
   6033  */
   6034 static void
   6035 xsltParseStylesheetTop(xsltStylesheetPtr style, xmlNodePtr top) {
   6036     xmlNodePtr cur;
   6037     xmlChar *prop;
   6038 #ifdef WITH_XSLT_DEBUG_PARSING
   6039     int templates = 0;
   6040 #endif
   6041 
   6042     if (top == NULL)
   6043 	return;
   6044 
   6045     prop = xmlGetNsProp(top, (const xmlChar *)"version", NULL);
   6046     if (prop == NULL) {
   6047 	xsltTransformError(NULL, style, top,
   6048 	    "xsl:version is missing: document may not be a stylesheet\n");
   6049 	if (style != NULL) style->warnings++;
   6050     } else {
   6051 	if ((!xmlStrEqual(prop, (const xmlChar *)"1.0")) &&
   6052             (!xmlStrEqual(prop, (const xmlChar *)"1.1"))) {
   6053 	    xsltTransformError(NULL, style, top,
   6054 		"xsl:version: only 1.0 features are supported\n");
   6055 	     /* TODO set up compatibility when not XSLT 1.0 */
   6056 	    if (style != NULL) style->warnings++;
   6057 	}
   6058 	xmlFree(prop);
   6059     }
   6060 
   6061     /*
   6062      * process xsl:import elements
   6063      */
   6064     cur = top->children;
   6065     while (cur != NULL) {
   6066 	    if (IS_BLANK_NODE(cur)) {
   6067 		    cur = cur->next;
   6068 		    continue;
   6069 	    }
   6070 	    if (IS_XSLT_ELEM(cur) && IS_XSLT_NAME(cur, "import")) {
   6071 		    if (xsltParseStylesheetImport(style, cur) != 0)
   6072 			    if (style != NULL) style->errors++;
   6073 	    } else
   6074 		    break;
   6075 	    cur = cur->next;
   6076     }
   6077 
   6078     /*
   6079      * process other top-level elements
   6080      */
   6081     while (cur != NULL) {
   6082 	if (IS_BLANK_NODE(cur)) {
   6083 	    cur = cur->next;
   6084 	    continue;
   6085 	}
   6086 	if (cur->type == XML_TEXT_NODE) {
   6087 	    if (cur->content != NULL) {
   6088 		xsltTransformError(NULL, style, cur,
   6089 		    "misplaced text node: '%s'\n", cur->content);
   6090 	    }
   6091 	    if (style != NULL) style->errors++;
   6092             cur = cur->next;
   6093 	    continue;
   6094 	}
   6095 	if ((cur->type == XML_ELEMENT_NODE) && (cur->ns == NULL)) {
   6096 	    xsltGenericError(xsltGenericErrorContext,
   6097 		     "Found a top-level element %s with null namespace URI\n",
   6098 		     cur->name);
   6099 	    if (style != NULL) style->errors++;
   6100 	    cur = cur->next;
   6101 	    continue;
   6102 	}
   6103 	if ((cur->type == XML_ELEMENT_NODE) && (!(IS_XSLT_ELEM(cur)))) {
   6104 	    xsltTopLevelFunction function;
   6105 
   6106 	    function = xsltExtModuleTopLevelLookup(cur->name,
   6107 						   cur->ns->href);
   6108 	    if (function != NULL)
   6109 		function(style, cur);
   6110 
   6111 #ifdef WITH_XSLT_DEBUG_PARSING
   6112 	    xsltGenericDebug(xsltGenericDebugContext,
   6113 		    "xsltParseStylesheetTop : found foreign element %s\n",
   6114 		    cur->name);
   6115 #endif
   6116             cur = cur->next;
   6117 	    continue;
   6118 	}
   6119 	if (IS_XSLT_NAME(cur, "import")) {
   6120 	    xsltTransformError(NULL, style, cur,
   6121 			"xsltParseStylesheetTop: ignoring misplaced import element\n");
   6122 	    if (style != NULL) style->errors++;
   6123     } else if (IS_XSLT_NAME(cur, "include")) {
   6124 	    if (xsltParseStylesheetInclude(style, cur) != 0)
   6125 		if (style != NULL) style->errors++;
   6126     } else if (IS_XSLT_NAME(cur, "strip-space")) {
   6127 	    xsltParseStylesheetStripSpace(style, cur);
   6128     } else if (IS_XSLT_NAME(cur, "preserve-space")) {
   6129 	    xsltParseStylesheetPreserveSpace(style, cur);
   6130     } else if (IS_XSLT_NAME(cur, "output")) {
   6131 	    xsltParseStylesheetOutput(style, cur);
   6132     } else if (IS_XSLT_NAME(cur, "key")) {
   6133 	    xsltParseStylesheetKey(style, cur);
   6134     } else if (IS_XSLT_NAME(cur, "decimal-format")) {
   6135 	    xsltParseStylesheetDecimalFormat(style, cur);
   6136     } else if (IS_XSLT_NAME(cur, "attribute-set")) {
   6137 	    xsltParseStylesheetAttributeSet(style, cur);
   6138     } else if (IS_XSLT_NAME(cur, "variable")) {
   6139 	    xsltParseGlobalVariable(style, cur);
   6140     } else if (IS_XSLT_NAME(cur, "param")) {
   6141 	    xsltParseGlobalParam(style, cur);
   6142     } else if (IS_XSLT_NAME(cur, "template")) {
   6143 #ifdef WITH_XSLT_DEBUG_PARSING
   6144 	    templates++;
   6145 #endif
   6146 	    xsltParseStylesheetTemplate(style, cur);
   6147     } else if (IS_XSLT_NAME(cur, "namespace-alias")) {
   6148 	    xsltNamespaceAlias(style, cur);
   6149 	} else {
   6150 	    /*
   6151 	    * BUG TODO: The version of the *doc* is irrelevant for
   6152 	    *  the forwards-compatible mode.
   6153 	    */
   6154             if ((style != NULL) && (style->doc->version != NULL) &&
   6155 	        (!strncmp((const char *) style->doc->version, "1.0", 3))) {
   6156 	        xsltTransformError(NULL, style, cur,
   6157 			"xsltParseStylesheetTop: unknown %s element\n",
   6158 			cur->name);
   6159 	        if (style != NULL) style->errors++;
   6160 	    }
   6161 	    else {
   6162                 /* do Forwards-Compatible Processing */
   6163 	        xsltTransformError(NULL, style, cur,
   6164 			"xsltParseStylesheetTop: ignoring unknown %s element\n",
   6165 			cur->name);
   6166 	        if (style != NULL) style->warnings++;
   6167             }
   6168 	}
   6169 	cur = cur->next;
   6170     }
   6171 #ifdef WITH_XSLT_DEBUG_PARSING
   6172     xsltGenericDebug(xsltGenericDebugContext,
   6173 		    "parsed %d templates\n", templates);
   6174 #endif
   6175 }
   6176 
   6177 #endif /* else of XSLT_REFACTORED */
   6178 
   6179 #ifdef XSLT_REFACTORED
   6180 /**
   6181  * xsltParseSimplifiedStylesheetTree:
   6182  *
   6183  * @style: the stylesheet (TODO: Change this to the compiler context)
   6184  * @doc: the document containing the stylesheet.
   6185  * @node: the node where the stylesheet is rooted at
   6186  *
   6187  * Returns 0 in case of success, a positive result if an error occurred
   6188  *         and -1 on API and internal errors.
   6189  */
   6190 static int
   6191 xsltParseSimplifiedStylesheetTree(xsltCompilerCtxtPtr cctxt,
   6192 				  xmlDocPtr doc,
   6193 				  xmlNodePtr node)
   6194 {
   6195     xsltTemplatePtr templ;
   6196 
   6197     if ((cctxt == NULL) || (node == NULL))
   6198 	return(-1);
   6199 
   6200     if (xsltParseAttrXSLTVersion(cctxt, node, 0) == XSLT_ELEMENT_CATEGORY_LRE)
   6201     {
   6202 	/*
   6203 	* TODO: Adjust report, since this might be an
   6204 	* embedded stylesheet.
   6205 	*/
   6206 	xsltTransformError(NULL, cctxt->style, node,
   6207 	    "The attribute 'xsl:version' is missing; cannot identify "
   6208 	    "this document as an XSLT stylesheet document.\n");
   6209 	cctxt->style->errors++;
   6210 	return(1);
   6211     }
   6212 
   6213 #ifdef WITH_XSLT_DEBUG_PARSING
   6214     xsltGenericDebug(xsltGenericDebugContext,
   6215 	"xsltParseSimplifiedStylesheetTree: document is stylesheet\n");
   6216 #endif
   6217 
   6218     /*
   6219     * Create and link the template
   6220     */
   6221     templ = xsltNewTemplate();
   6222     if (templ == NULL) {
   6223 	return(-1);
   6224     }
   6225     templ->next = cctxt->style->templates;
   6226     cctxt->style->templates = templ;
   6227     templ->match = xmlStrdup(BAD_CAST "/");
   6228 
   6229     /*
   6230     * Note that we push the document-node in this special case.
   6231     */
   6232     xsltCompilerNodePush(cctxt, (xmlNodePtr) doc);
   6233     /*
   6234     * In every case, we need to have
   6235     * the in-scope namespaces of the element, where the
   6236     * stylesheet is rooted at, regardless if it's an XSLT
   6237     * instruction or a literal result instruction (or if
   6238     * this is an embedded stylesheet).
   6239     */
   6240     cctxt->inode->inScopeNs =
   6241 	xsltCompilerBuildInScopeNsList(cctxt, node);
   6242     /*
   6243     * Parse the content and register the match-pattern.
   6244     */
   6245     xsltParseSequenceConstructor(cctxt, node);
   6246     xsltCompilerNodePop(cctxt, (xmlNodePtr) doc);
   6247 
   6248     templ->elem = (xmlNodePtr) doc;
   6249     templ->content = node;
   6250     xsltAddTemplate(cctxt->style, templ, NULL, NULL);
   6251     cctxt->style->literal_result = 1;
   6252     return(0);
   6253 }
   6254 
   6255 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
   6256 /**
   6257  * xsltRestoreDocumentNamespaces:
   6258  * @ns: map of namespaces
   6259  * @doc: the document
   6260  *
   6261  * Restore the namespaces for the document
   6262  *
   6263  * Returns 0 in case of success, -1 in case of failure
   6264  */
   6265 int
   6266 xsltRestoreDocumentNamespaces(xsltNsMapPtr ns, xmlDocPtr doc)
   6267 {
   6268     if (doc == NULL)
   6269 	return(-1);
   6270     /*
   6271     * Revert the changes we have applied to the namespace-URIs of
   6272     * ns-decls.
   6273     */
   6274     while (ns != NULL) {
   6275 	if ((ns->doc == doc) && (ns->ns != NULL)) {
   6276 	    ns->ns->href = ns->origNsName;
   6277 	    ns->origNsName = NULL;
   6278 	    ns->ns = NULL;
   6279 	}
   6280 	ns = ns->next;
   6281     }
   6282     return(0);
   6283 }
   6284 #endif /* XSLT_REFACTORED_XSLT_NSCOMP */
   6285 
   6286 /**
   6287  * xsltParseStylesheetProcess:
   6288  * @style:  the XSLT stylesheet (the current stylesheet-level)
   6289  * @doc:  and xmlDoc parsed XML
   6290  *
   6291  * Parses an XSLT stylesheet, adding the associated structures.
   6292  * Called by:
   6293  *  xsltParseStylesheetImportedDoc() (xslt.c)
   6294  *  xsltParseStylesheetInclude() (imports.c)
   6295  *
   6296  * Returns the value of the @style parameter if everything
   6297  * went right, NULL if something went amiss.
   6298  */
   6299 xsltStylesheetPtr
   6300 xsltParseStylesheetProcess(xsltStylesheetPtr style, xmlDocPtr doc)
   6301 {
   6302     xsltCompilerCtxtPtr cctxt;
   6303     xmlNodePtr cur;
   6304     int oldIsSimplifiedStylesheet;
   6305 
   6306     xsltInitGlobals();
   6307 
   6308     if ((style == NULL) || (doc == NULL))
   6309 	return(NULL);
   6310 
   6311     cctxt = XSLT_CCTXT(style);
   6312 
   6313     cur = xmlDocGetRootElement(doc);
   6314     if (cur == NULL) {
   6315 	xsltTransformError(NULL, style, (xmlNodePtr) doc,
   6316 		"xsltParseStylesheetProcess : empty stylesheet\n");
   6317 	return(NULL);
   6318     }
   6319     oldIsSimplifiedStylesheet = cctxt->simplified;
   6320 
   6321     if ((IS_XSLT_ELEM(cur)) &&
   6322 	((IS_XSLT_NAME(cur, "stylesheet")) ||
   6323 	 (IS_XSLT_NAME(cur, "transform")))) {
   6324 #ifdef WITH_XSLT_DEBUG_PARSING
   6325 	xsltGenericDebug(xsltGenericDebugContext,
   6326 		"xsltParseStylesheetProcess : found stylesheet\n");
   6327 #endif
   6328 	cctxt->simplified = 0;
   6329 	style->literal_result = 0;
   6330     } else {
   6331 	cctxt->simplified = 1;
   6332 	style->literal_result = 1;
   6333     }
   6334     /*
   6335     * Pre-process the stylesheet if not already done before.
   6336     *  This will remove PIs and comments, merge adjacent
   6337     *  text nodes, internalize strings, etc.
   6338     */
   6339     if (! style->nopreproc)
   6340 	xsltParsePreprocessStylesheetTree(cctxt, cur);
   6341     /*
   6342     * Parse and compile the stylesheet.
   6343     */
   6344     if (style->literal_result == 0) {
   6345 	if (xsltParseXSLTStylesheetElem(cctxt, cur) != 0)
   6346 	    return(NULL);
   6347     } else {
   6348 	if (xsltParseSimplifiedStylesheetTree(cctxt, doc, cur) != 0)
   6349 	    return(NULL);
   6350     }
   6351 
   6352     cctxt->simplified = oldIsSimplifiedStylesheet;
   6353 
   6354     return(style);
   6355 }
   6356 
   6357 #else /* XSLT_REFACTORED */
   6358 
   6359 /**
   6360  * xsltParseStylesheetProcess:
   6361  * @ret:  the XSLT stylesheet (the current stylesheet-level)
   6362  * @doc:  and xmlDoc parsed XML
   6363  *
   6364  * Parses an XSLT stylesheet, adding the associated structures.
   6365  * Called by:
   6366  *  xsltParseStylesheetImportedDoc() (xslt.c)
   6367  *  xsltParseStylesheetInclude() (imports.c)
   6368  *
   6369  * Returns the value of the @style parameter if everything
   6370  * went right, NULL if something went amiss.
   6371  */
   6372 xsltStylesheetPtr
   6373 xsltParseStylesheetProcess(xsltStylesheetPtr ret, xmlDocPtr doc) {
   6374     xmlNodePtr cur;
   6375 
   6376     xsltInitGlobals();
   6377 
   6378     if (doc == NULL)
   6379 	return(NULL);
   6380     if (ret == NULL)
   6381 	return(ret);
   6382 
   6383     /*
   6384      * First steps, remove blank nodes,
   6385      * locate the xsl:stylesheet element and the
   6386      * namespace declaration.
   6387      */
   6388     cur = xmlDocGetRootElement(doc);
   6389     if (cur == NULL) {
   6390 	xsltTransformError(NULL, ret, (xmlNodePtr) doc,
   6391 		"xsltParseStylesheetProcess : empty stylesheet\n");
   6392 	return(NULL);
   6393     }
   6394 
   6395     if ((IS_XSLT_ELEM(cur)) &&
   6396 	((IS_XSLT_NAME(cur, "stylesheet")) ||
   6397 	 (IS_XSLT_NAME(cur, "transform")))) {
   6398 #ifdef WITH_XSLT_DEBUG_PARSING
   6399 	xsltGenericDebug(xsltGenericDebugContext,
   6400 		"xsltParseStylesheetProcess : found stylesheet\n");
   6401 #endif
   6402 	ret->literal_result = 0;
   6403 	xsltParseStylesheetExcludePrefix(ret, cur, 1);
   6404 	xsltParseStylesheetExtPrefix(ret, cur, 1);
   6405     } else {
   6406 	xsltParseStylesheetExcludePrefix(ret, cur, 0);
   6407 	xsltParseStylesheetExtPrefix(ret, cur, 0);
   6408 	ret->literal_result = 1;
   6409     }
   6410     if (!ret->nopreproc) {
   6411 	xsltPrecomputeStylesheet(ret, cur);
   6412     }
   6413     if (ret->literal_result == 0) {
   6414 	xsltParseStylesheetTop(ret, cur);
   6415     } else {
   6416 	xmlChar *prop;
   6417 	xsltTemplatePtr template;
   6418 
   6419 	/*
   6420 	 * the document itself might be the template, check xsl:version
   6421 	 */
   6422 	prop = xmlGetNsProp(cur, (const xmlChar *)"version", XSLT_NAMESPACE);
   6423 	if (prop == NULL) {
   6424 	    xsltTransformError(NULL, ret, cur,
   6425 		"xsltParseStylesheetProcess : document is not a stylesheet\n");
   6426 	    return(NULL);
   6427 	}
   6428 
   6429 #ifdef WITH_XSLT_DEBUG_PARSING
   6430         xsltGenericDebug(xsltGenericDebugContext,
   6431 		"xsltParseStylesheetProcess : document is stylesheet\n");
   6432 #endif
   6433 
   6434 	if (!xmlStrEqual(prop, (const xmlChar *)"1.0")) {
   6435 	    xsltTransformError(NULL, ret, cur,
   6436 		"xsl:version: only 1.0 features are supported\n");
   6437 	     /* TODO set up compatibility when not XSLT 1.0 */
   6438 	    ret->warnings++;
   6439 	}
   6440 	xmlFree(prop);
   6441 
   6442 	/*
   6443 	 * Create and link the template
   6444 	 */
   6445 	template = xsltNewTemplate();
   6446 	if (template == NULL) {
   6447 	    return(NULL);
   6448 	}
   6449 	template->next = ret->templates;
   6450 	ret->templates = template;
   6451 	template->match = xmlStrdup((const xmlChar *)"/");
   6452 
   6453 	/*
   6454 	 * parse the content and register the pattern
   6455 	 */
   6456 	xsltParseTemplateContent(ret, (xmlNodePtr) doc);
   6457 	template->elem = (xmlNodePtr) doc;
   6458 	template->content = doc->children;
   6459 	xsltAddTemplate(ret, template, NULL, NULL);
   6460 	ret->literal_result = 1;
   6461     }
   6462 
   6463     return(ret);
   6464 }
   6465 
   6466 #endif /* else of XSLT_REFACTORED */
   6467 
   6468 /**
   6469  * xsltParseStylesheetImportedDoc:
   6470  * @doc:  an xmlDoc parsed XML
   6471  * @parentStyle: pointer to the parent stylesheet (if it exists)
   6472  *
   6473  * parse an XSLT stylesheet building the associated structures
   6474  * except the processing not needed for imported documents.
   6475  *
   6476  * Returns a new XSLT stylesheet structure.
   6477  */
   6478 
   6479 xsltStylesheetPtr
   6480 xsltParseStylesheetImportedDoc(xmlDocPtr doc,
   6481 			       xsltStylesheetPtr parentStyle) {
   6482     xsltStylesheetPtr retStyle;
   6483 
   6484     if (doc == NULL)
   6485 	return(NULL);
   6486 
   6487     retStyle = xsltNewStylesheet();
   6488     if (retStyle == NULL)
   6489 	return(NULL);
   6490     /*
   6491     * Set the importing stylesheet module; also used to detect recursion.
   6492     */
   6493     retStyle->parent = parentStyle;
   6494     /*
   6495     * Adjust the string dict.
   6496     */
   6497     if (doc->dict != NULL) {
   6498         xmlDictFree(retStyle->dict);
   6499 	retStyle->dict = doc->dict;
   6500 #ifdef WITH_XSLT_DEBUG
   6501         xsltGenericDebug(xsltGenericDebugContext,
   6502 	    "reusing dictionary from %s for stylesheet\n",
   6503 	    doc->URL);
   6504 #endif
   6505 	xmlDictReference(retStyle->dict);
   6506     }
   6507 
   6508     /*
   6509     * TODO: Eliminate xsltGatherNamespaces(); we must not restrict
   6510     *  the stylesheet to containt distinct namespace prefixes.
   6511     */
   6512     xsltGatherNamespaces(retStyle);
   6513 
   6514 #ifdef XSLT_REFACTORED
   6515     {
   6516 	xsltCompilerCtxtPtr cctxt;
   6517 	xsltStylesheetPtr oldCurSheet;
   6518 
   6519 	if (parentStyle == NULL) {
   6520 	    xsltPrincipalStylesheetDataPtr principalData;
   6521 	    /*
   6522 	    * Principal stylesheet
   6523 	    * --------------------
   6524 	    */
   6525 	    retStyle->principal = retStyle;
   6526 	    /*
   6527 	    * Create extra data for the principal stylesheet.
   6528 	    */
   6529 	    principalData = xsltNewPrincipalStylesheetData();
   6530 	    if (principalData == NULL) {
   6531 		xsltFreeStylesheet(retStyle);
   6532 		return(NULL);
   6533 	    }
   6534 	    retStyle->principalData = principalData;
   6535 	    /*
   6536 	    * Create the compilation context
   6537 	    * ------------------------------
   6538 	    * (only once; for the principal stylesheet).
   6539 	    * This is currently the only function where the
   6540 	    * compilation context is created.
   6541 	    */
   6542 	    cctxt = xsltCompilationCtxtCreate(retStyle);
   6543 	    if (cctxt == NULL) {
   6544 		xsltFreeStylesheet(retStyle);
   6545 		return(NULL);
   6546 	    }
   6547 	    retStyle->compCtxt = (void *) cctxt;
   6548 	    cctxt->style = retStyle;
   6549 	    cctxt->dict = retStyle->dict;
   6550 	    cctxt->psData = principalData;
   6551 	    /*
   6552 	    * Push initial dummy node info.
   6553 	    */
   6554 	    cctxt->depth = -1;
   6555 	    xsltCompilerNodePush(cctxt, (xmlNodePtr) doc);
   6556 	} else {
   6557 	    /*
   6558 	    * Imported stylesheet.
   6559 	    */
   6560 	    retStyle->principal = parentStyle->principal;
   6561 	    cctxt = parentStyle->compCtxt;
   6562 	    retStyle->compCtxt = cctxt;
   6563 	}
   6564 	/*
   6565 	* Save the old and set the current stylesheet structure in the
   6566 	* compilation context.
   6567 	*/
   6568 	oldCurSheet = cctxt->style;
   6569 	cctxt->style = retStyle;
   6570 
   6571 	retStyle->doc = doc;
   6572 	xsltParseStylesheetProcess(retStyle, doc);
   6573 
   6574 	cctxt->style = oldCurSheet;
   6575 	if (parentStyle == NULL) {
   6576 	    /*
   6577 	    * Pop the initial dummy node info.
   6578 	    */
   6579 	    xsltCompilerNodePop(cctxt, (xmlNodePtr) doc);
   6580 	} else {
   6581 	    /*
   6582 	    * Clear the compilation context of imported
   6583 	    * stylesheets.
   6584 	    * TODO: really?
   6585 	    */
   6586 	    /* retStyle->compCtxt = NULL; */
   6587 	}
   6588 	/*
   6589 	* Free the stylesheet if there were errors.
   6590 	*/
   6591 	if (retStyle != NULL) {
   6592 	    if (retStyle->errors != 0) {
   6593 #ifdef XSLT_REFACTORED_XSLT_NSCOMP
   6594 		/*
   6595 		* Restore all changes made to namespace URIs of ns-decls.
   6596 		*/
   6597 		if (cctxt->psData->nsMap)
   6598 		    xsltRestoreDocumentNamespaces(cctxt->psData->nsMap, doc);
   6599 #endif
   6600 		/*
   6601 		* Detach the doc from the stylesheet; otherwise the doc
   6602 		* will be freed in xsltFreeStylesheet().
   6603 		*/
   6604 		retStyle->doc = NULL;
   6605 		/*
   6606 		* Cleanup the doc if its the main stylesheet.
   6607 		*/
   6608 		if (parentStyle == NULL) {
   6609 		    xsltCleanupStylesheetTree(doc, xmlDocGetRootElement(doc));
   6610 		    if (retStyle->compCtxt != NULL) {
   6611 			xsltCompilationCtxtFree(retStyle->compCtxt);
   6612 			retStyle->compCtxt = NULL;
   6613 		    }
   6614 		}
   6615 
   6616 		xsltFreeStylesheet(retStyle);
   6617 		retStyle = NULL;
   6618 	    }
   6619 	}
   6620     }
   6621 
   6622 #else /* XSLT_REFACTORED */
   6623     /*
   6624     * Old behaviour.
   6625     */
   6626     retStyle->doc = doc;
   6627     if (xsltParseStylesheetProcess(retStyle, doc) == NULL) {
   6628 		retStyle->doc = NULL;
   6629 		xsltFreeStylesheet(retStyle);
   6630 		retStyle = NULL;
   6631     }
   6632     if (retStyle != NULL) {
   6633 	if (retStyle->errors != 0) {
   6634 	    retStyle->doc = NULL;
   6635 	    if (parentStyle == NULL)
   6636 		xsltCleanupStylesheetTree(doc,
   6637 		    xmlDocGetRootElement(doc));
   6638 	    xsltFreeStylesheet(retStyle);
   6639 	    retStyle = NULL;
   6640 	}
   6641     }
   6642 #endif /* else of XSLT_REFACTORED */
   6643 
   6644     return(retStyle);
   6645 }
   6646 
   6647 /**
   6648  * xsltParseStylesheetDoc:
   6649  * @doc:  and xmlDoc parsed XML
   6650  *
   6651  * parse an XSLT stylesheet, building the associated structures.  doc
   6652  * is kept as a reference within the returned stylesheet, so changes
   6653  * to doc after the parsing will be reflected when the stylesheet
   6654  * is applied, and the doc is automatically freed when the
   6655  * stylesheet is closed.
   6656  *
   6657  * Returns a new XSLT stylesheet structure.
   6658  */
   6659 
   6660 xsltStylesheetPtr
   6661 xsltParseStylesheetDoc(xmlDocPtr doc) {
   6662     xsltStylesheetPtr ret;
   6663 
   6664     xsltInitGlobals();
   6665 
   6666     ret = xsltParseStylesheetImportedDoc(doc, NULL);
   6667     if (ret == NULL)
   6668 	return(NULL);
   6669 
   6670     xsltResolveStylesheetAttributeSet(ret);
   6671 #ifdef XSLT_REFACTORED
   6672     /*
   6673     * Free the compilation context.
   6674     * TODO: Check if it's better to move this cleanup to
   6675     *   xsltParseStylesheetImportedDoc().
   6676     */
   6677     if (ret->compCtxt != NULL) {
   6678 	xsltCompilationCtxtFree(XSLT_CCTXT(ret));
   6679 	ret->compCtxt = NULL;
   6680     }
   6681 #endif
   6682     return(ret);
   6683 }
   6684 
   6685 /**
   6686  * xsltParseStylesheetFile:
   6687  * @filename:  the filename/URL to the stylesheet
   6688  *
   6689  * Load and parse an XSLT stylesheet
   6690  *
   6691  * Returns a new XSLT stylesheet structure.
   6692  */
   6693 
   6694 xsltStylesheetPtr
   6695 xsltParseStylesheetFile(const xmlChar* filename) {
   6696     xsltSecurityPrefsPtr sec;
   6697     xsltStylesheetPtr ret;
   6698     xmlDocPtr doc;
   6699 
   6700     xsltInitGlobals();
   6701 
   6702     if (filename == NULL)
   6703 	return(NULL);
   6704 
   6705 #ifdef WITH_XSLT_DEBUG_PARSING
   6706     xsltGenericDebug(xsltGenericDebugContext,
   6707 	    "xsltParseStylesheetFile : parse %s\n", filename);
   6708 #endif
   6709 
   6710     /*
   6711      * Security framework check
   6712      */
   6713     sec = xsltGetDefaultSecurityPrefs();
   6714     if (sec != NULL) {
   6715 	int res;
   6716 
   6717 	res = xsltCheckRead(sec, NULL, filename);
   6718 	if (res == 0) {
   6719 	    xsltTransformError(NULL, NULL, NULL,
   6720 		 "xsltParseStylesheetFile: read rights for %s denied\n",
   6721 			     filename);
   6722 	    return(NULL);
   6723 	}
   6724     }
   6725 
   6726     doc = xsltDocDefaultLoader(filename, NULL, XSLT_PARSE_OPTIONS,
   6727                                NULL, XSLT_LOAD_START);
   6728     if (doc == NULL) {
   6729 	xsltTransformError(NULL, NULL, NULL,
   6730 		"xsltParseStylesheetFile : cannot parse %s\n", filename);
   6731 	return(NULL);
   6732     }
   6733     ret = xsltParseStylesheetDoc(doc);
   6734     if (ret == NULL) {
   6735 	xmlFreeDoc(doc);
   6736 	return(NULL);
   6737     }
   6738 
   6739     return(ret);
   6740 }
   6741 
   6742 /************************************************************************
   6743  *									*
   6744  *			Handling of Stylesheet PI			*
   6745  *									*
   6746  ************************************************************************/
   6747 
   6748 #define CUR (*cur)
   6749 #define SKIP(val) cur += (val)
   6750 #define NXT(val) cur[(val)]
   6751 #define SKIP_BLANKS						\
   6752     while (IS_BLANK(CUR)) NEXT
   6753 #define NEXT ((*cur) ?  cur++ : cur)
   6754 
   6755 /**
   6756  * xsltParseStylesheetPI:
   6757  * @value: the value of the PI
   6758  *
   6759  * This function checks that the type is text/xml and extracts
   6760  * the URI-Reference for the stylesheet
   6761  *
   6762  * Returns the URI-Reference for the stylesheet or NULL (it need to
   6763  *         be freed by the caller)
   6764  */
   6765 static xmlChar *
   6766 xsltParseStylesheetPI(const xmlChar *value) {
   6767     const xmlChar *cur;
   6768     const xmlChar *start;
   6769     xmlChar *val;
   6770     xmlChar tmp;
   6771     xmlChar *href = NULL;
   6772     int isXml = 0;
   6773 
   6774     if (value == NULL)
   6775 	return(NULL);
   6776 
   6777     cur = value;
   6778     while (CUR != 0) {
   6779 	SKIP_BLANKS;
   6780 	if ((CUR == 't') && (NXT(1) == 'y') && (NXT(2) == 'p') &&
   6781 	    (NXT(3) == 'e')) {
   6782 	    SKIP(4);
   6783 	    SKIP_BLANKS;
   6784 	    if (CUR != '=')
   6785 		continue;
   6786 	    NEXT;
   6787 	    if ((CUR != '\'') && (CUR != '"'))
   6788 		continue;
   6789 	    tmp = CUR;
   6790 	    NEXT;
   6791 	    start = cur;
   6792 	    while ((CUR != 0) && (CUR != tmp))
   6793 		NEXT;
   6794 	    if (CUR != tmp)
   6795 		continue;
   6796 	    val = xmlStrndup(start, cur - start);
   6797 	    NEXT;
   6798 	    if (val == NULL)
   6799 		return(NULL);
   6800 	    if ((xmlStrcasecmp(val, BAD_CAST "text/xml")) &&
   6801 		(xmlStrcasecmp(val, BAD_CAST "text/xsl"))) {
   6802                 xmlFree(val);
   6803 		break;
   6804 	    }
   6805 	    isXml = 1;
   6806 	    xmlFree(val);
   6807 	} else if ((CUR == 'h') && (NXT(1) == 'r') && (NXT(2) == 'e') &&
   6808 	    (NXT(3) == 'f')) {
   6809 	    SKIP(4);
   6810 	    SKIP_BLANKS;
   6811 	    if (CUR != '=')
   6812 		continue;
   6813 	    NEXT;
   6814 	    if ((CUR != '\'') && (CUR != '"'))
   6815 		continue;
   6816 	    tmp = CUR;
   6817 	    NEXT;
   6818 	    start = cur;
   6819 	    while ((CUR != 0) && (CUR != tmp))
   6820 		NEXT;
   6821 	    if (CUR != tmp)
   6822 		continue;
   6823 	    if (href == NULL)
   6824 		href = xmlStrndup(start, cur - start);
   6825 	    NEXT;
   6826 	} else {
   6827 	    while ((CUR != 0) && (!IS_BLANK(CUR)))
   6828 		NEXT;
   6829 	}
   6830 
   6831     }
   6832 
   6833     if (!isXml) {
   6834 	if (href != NULL)
   6835 	    xmlFree(href);
   6836 	href = NULL;
   6837     }
   6838     return(href);
   6839 }
   6840 
   6841 /**
   6842  * xsltLoadStylesheetPI:
   6843  * @doc:  a document to process
   6844  *
   6845  * This function tries to locate the stylesheet PI in the given document
   6846  * If found, and if contained within the document, it will extract
   6847  * that subtree to build the stylesheet to process @doc (doc itself will
   6848  * be modified). If found but referencing an external document it will
   6849  * attempt to load it and generate a stylesheet from it. In both cases,
   6850  * the resulting stylesheet and the document need to be freed once the
   6851  * transformation is done.
   6852  *
   6853  * Returns a new XSLT stylesheet structure or NULL if not found.
   6854  */
   6855 xsltStylesheetPtr
   6856 xsltLoadStylesheetPI(xmlDocPtr doc) {
   6857     xmlNodePtr child;
   6858     xsltStylesheetPtr ret = NULL;
   6859     xmlChar *href = NULL;
   6860     xmlURIPtr URI;
   6861 
   6862     xsltInitGlobals();
   6863 
   6864     if (doc == NULL)
   6865 	return(NULL);
   6866 
   6867     /*
   6868      * Find the text/xml stylesheet PI id any before the root
   6869      */
   6870     child = doc->children;
   6871     while ((child != NULL) && (child->type != XML_ELEMENT_NODE)) {
   6872 	if ((child->type == XML_PI_NODE) &&
   6873 	    (xmlStrEqual(child->name, BAD_CAST "xml-stylesheet"))) {
   6874 	    href = xsltParseStylesheetPI(child->content);
   6875 	    if (href != NULL)
   6876 		break;
   6877 	}
   6878 	child = child->next;
   6879     }
   6880 
   6881     /*
   6882      * If found check the href to select processing
   6883      */
   6884     if (href != NULL) {
   6885 #ifdef WITH_XSLT_DEBUG_PARSING
   6886 	xsltGenericDebug(xsltGenericDebugContext,
   6887 		"xsltLoadStylesheetPI : found PI href=%s\n", href);
   6888 #endif
   6889 	URI = xmlParseURI((const char *) href);
   6890 	if (URI == NULL) {
   6891 	    xsltTransformError(NULL, NULL, child,
   6892 		    "xml-stylesheet : href %s is not valid\n", href);
   6893 	    xmlFree(href);
   6894 	    return(NULL);
   6895 	}
   6896 	if ((URI->fragment != NULL) && (URI->scheme == NULL) &&
   6897             (URI->opaque == NULL) && (URI->authority == NULL) &&
   6898             (URI->server == NULL) && (URI->user == NULL) &&
   6899             (URI->path == NULL) && (URI->query == NULL)) {
   6900 	    xmlAttrPtr ID;
   6901 
   6902 #ifdef WITH_XSLT_DEBUG_PARSING
   6903 	    xsltGenericDebug(xsltGenericDebugContext,
   6904 		    "xsltLoadStylesheetPI : Reference to ID %s\n", href);
   6905 #endif
   6906 	    if (URI->fragment[0] == '#')
   6907 		ID = xmlGetID(doc, (const xmlChar *) &(URI->fragment[1]));
   6908 	    else
   6909 		ID = xmlGetID(doc, (const xmlChar *) URI->fragment);
   6910 	    if (ID == NULL) {
   6911 		xsltTransformError(NULL, NULL, child,
   6912 		    "xml-stylesheet : no ID %s found\n", URI->fragment);
   6913 	    } else {
   6914 		xmlDocPtr fake;
   6915 		xmlNodePtr subtree, newtree;
   6916 		xmlNsPtr ns;
   6917 
   6918 #ifdef WITH_XSLT_DEBUG
   6919 		xsltGenericDebug(xsltGenericDebugContext,
   6920 		    "creating new document from %s for embedded stylesheet\n",
   6921 		    doc->URL);
   6922 #endif
   6923 		/*
   6924 		 * move the subtree in a new document passed to
   6925 		 * the stylesheet analyzer
   6926 		 */
   6927 		subtree = ID->parent;
   6928 		fake = xmlNewDoc(NULL);
   6929 		if (fake != NULL) {
   6930 		    /*
   6931 		    * Should the dictionary still be shared even though
   6932 		    * the nodes are being copied rather than moved?
   6933 		    */
   6934 		    fake->dict = doc->dict;
   6935 		    xmlDictReference(doc->dict);
   6936 #ifdef WITH_XSLT_DEBUG
   6937 		    xsltGenericDebug(xsltGenericDebugContext,
   6938 			"reusing dictionary from %s for embedded stylesheet\n",
   6939 			doc->URL);
   6940 #endif
   6941 
   6942 		    newtree = xmlDocCopyNode(subtree, fake, 1);
   6943 
   6944 		    fake->URL = xmlNodeGetBase(doc, subtree->parent);
   6945 #ifdef WITH_XSLT_DEBUG
   6946 		    xsltGenericDebug(xsltGenericDebugContext,
   6947 			"set base URI for embedded stylesheet as %s\n",
   6948 			fake->URL);
   6949 #endif
   6950 
   6951 		    /*
   6952 		    * Add all namespaces in scope of embedded stylesheet to
   6953 		    * root element of newly created stylesheet document
   6954 		    */
   6955 		    while ((subtree = subtree->parent) != (xmlNodePtr)doc) {
   6956 			for (ns = subtree->ns; ns; ns = ns->next) {
   6957 			    xmlNewNs(newtree,  ns->href, ns->prefix);
   6958 			}
   6959 		    }
   6960 
   6961 		    xmlAddChild((xmlNodePtr)fake, newtree);
   6962 		    ret = xsltParseStylesheetDoc(fake);
   6963 		    if (ret == NULL)
   6964 			xmlFreeDoc(fake);
   6965 		}
   6966 	    }
   6967 	} else {
   6968 	    xmlChar *URL, *base;
   6969 
   6970 	    /*
   6971 	     * Reference to an external stylesheet
   6972 	     */
   6973 
   6974 	    base = xmlNodeGetBase(doc, (xmlNodePtr) doc);
   6975 	    URL = xmlBuildURI(href, base);
   6976 	    if (URL != NULL) {
   6977 #ifdef WITH_XSLT_DEBUG_PARSING
   6978 		xsltGenericDebug(xsltGenericDebugContext,
   6979 			"xsltLoadStylesheetPI : fetching %s\n", URL);
   6980 #endif
   6981 		ret = xsltParseStylesheetFile(URL);
   6982 		xmlFree(URL);
   6983 	    } else {
   6984 #ifdef WITH_XSLT_DEBUG_PARSING
   6985 		xsltGenericDebug(xsltGenericDebugContext,
   6986 			"xsltLoadStylesheetPI : fetching %s\n", href);
   6987 #endif
   6988 		ret = xsltParseStylesheetFile(href);
   6989 	    }
   6990 	    if (base != NULL)
   6991 		xmlFree(base);
   6992 	}
   6993 	xmlFreeURI(URI);
   6994 	xmlFree(href);
   6995     }
   6996     return(ret);
   6997 }
   6998