mirror of
https://gitlab.gnome.org/GNOME/libxslt
synced 2025-11-06 23:49:25 +03:00
* libxslt/xslt.c: fixed bug #60624 * libxslt/xsltutils.c: improver the error context reporting * tests/reports/Makefile.am tests/reports/tst-2.*: added a specific regression test * xsltproc/xsltproc: free the stylesheet if it contained an error. Daniel
631 lines
17 KiB
C
631 lines
17 KiB
C
/*
|
|
* xsltproc.c: user program for the XSL Transformation 1.0 engine
|
|
*
|
|
* See Copyright for the status of this software.
|
|
*
|
|
* daniel@veillard.com
|
|
*/
|
|
|
|
#include "libxslt/libxslt.h"
|
|
#include "libexslt/exslt.h"
|
|
#ifdef HAVE_STRING_H
|
|
#include <string.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_TIME_H
|
|
#include <sys/time.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_STAT_H
|
|
#include <sys/stat.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_STDLIB_H
|
|
#include <stdlib.h>
|
|
#endif
|
|
#include <libxml/xmlmemory.h>
|
|
#include <libxml/debugXML.h>
|
|
#include <libxml/HTMLtree.h>
|
|
#include <libxml/xmlIO.h>
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
#include <libxml/DOCBparser.h>
|
|
#endif
|
|
#ifdef LIBXML_XINCLUDE_ENABLED
|
|
#include <libxml/xinclude.h>
|
|
#endif
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
#include <libxml/catalog.h>
|
|
#endif
|
|
#include <libxml/parserInternals.h>
|
|
|
|
#include <libxslt/xslt.h>
|
|
#include <libxslt/xsltInternals.h>
|
|
#include <libxslt/transform.h>
|
|
#include <libxslt/xsltutils.h>
|
|
#include <libxslt/extensions.h>
|
|
|
|
#include <libexslt/exsltconfig.h>
|
|
|
|
#ifdef WIN32
|
|
#ifdef _MSC_VER
|
|
#include <winsock2.h>
|
|
#pragma comment(lib, "ws2_32.lib")
|
|
#define gettimeofday(p1,p2)
|
|
#endif /* _MS_VER */
|
|
#else /* WIN32 */
|
|
#include <sys/time.h>
|
|
#endif /* WIN32 */
|
|
|
|
#ifndef HAVE_STAT
|
|
# ifdef HAVE__STAT
|
|
/* MS C library seems to define stat and _stat. The definition
|
|
* is identical. Still, mapping them to each other causes a warning. */
|
|
# ifndef _MSC_VER
|
|
# define stat(x,y) _stat(x,y)
|
|
# endif
|
|
# define HAVE_STAT
|
|
# endif
|
|
#endif
|
|
|
|
static int debug = 0;
|
|
static int repeat = 0;
|
|
static int timing = 0;
|
|
static int novalid = 0;
|
|
static int noout = 0;
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
static int docbook = 0;
|
|
#endif
|
|
#ifdef LIBXML_HTML_ENABLED
|
|
static int html = 0;
|
|
#endif
|
|
#ifdef LIBXML_XINCLUDE_ENABLED
|
|
static int xinclude = 0;
|
|
#endif
|
|
static int profile = 0;
|
|
static int nonet;
|
|
static xmlExternalEntityLoader defaultLoader = NULL;
|
|
|
|
static struct timeval begin, end;
|
|
static const char *params[16 + 1];
|
|
static int nbparams = 0;
|
|
static const char *output = NULL;
|
|
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
static int xsltNoNetExists(const char *URL) {
|
|
#ifdef HAVE_STAT
|
|
int ret;
|
|
struct stat info;
|
|
const char *path;
|
|
|
|
if (URL == NULL)
|
|
return(0);
|
|
|
|
if (!xmlStrncmp(BAD_CAST URL, BAD_CAST "file://localhost", 16))
|
|
path = &URL[16];
|
|
else if (!xmlStrncmp(BAD_CAST URL, BAD_CAST "file:///", 8)) {
|
|
#ifdef _WIN32
|
|
path = &URL[8];
|
|
#else
|
|
path = &URL[7];
|
|
#endif
|
|
} else
|
|
path = URL;
|
|
ret = stat(path, &info);
|
|
if (ret == 0)
|
|
return(1);
|
|
#endif
|
|
return(0);
|
|
}
|
|
#endif
|
|
|
|
static xmlParserInputPtr
|
|
xsltNoNetExternalEntityLoader(const char *URL, const char *ID,
|
|
xmlParserCtxtPtr ctxt) {
|
|
xmlParserInputPtr input = NULL;
|
|
xmlChar *resource = NULL;
|
|
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
xmlCatalogAllow pref;
|
|
|
|
/*
|
|
* If the resource doesn't exists as a file,
|
|
* try to load it from the resource pointed in the catalogs
|
|
*/
|
|
pref = xmlCatalogGetDefaults();
|
|
|
|
if ((pref != XML_CATA_ALLOW_NONE) && (!xsltNoNetExists(URL))) {
|
|
/*
|
|
* Do a local lookup
|
|
*/
|
|
if ((ctxt->catalogs != NULL) &&
|
|
((pref == XML_CATA_ALLOW_ALL) ||
|
|
(pref == XML_CATA_ALLOW_DOCUMENT))) {
|
|
resource = xmlCatalogLocalResolve(ctxt->catalogs,
|
|
(const xmlChar *)ID,
|
|
(const xmlChar *)URL);
|
|
}
|
|
/*
|
|
* Try a global lookup
|
|
*/
|
|
if ((resource == NULL) &&
|
|
((pref == XML_CATA_ALLOW_ALL) ||
|
|
(pref == XML_CATA_ALLOW_GLOBAL))) {
|
|
resource = xmlCatalogResolve((const xmlChar *)ID,
|
|
(const xmlChar *)URL);
|
|
}
|
|
if ((resource == NULL) && (URL != NULL))
|
|
resource = xmlStrdup((const xmlChar *) URL);
|
|
|
|
/*
|
|
* TODO: do an URI lookup on the reference
|
|
*/
|
|
if ((resource != NULL) && (!xsltNoNetExists((const char *)resource))) {
|
|
xmlChar *tmp = NULL;
|
|
|
|
if ((ctxt->catalogs != NULL) &&
|
|
((pref == XML_CATA_ALLOW_ALL) ||
|
|
(pref == XML_CATA_ALLOW_DOCUMENT))) {
|
|
tmp = xmlCatalogLocalResolveURI(ctxt->catalogs, resource);
|
|
}
|
|
if ((tmp == NULL) &&
|
|
((pref == XML_CATA_ALLOW_ALL) ||
|
|
(pref == XML_CATA_ALLOW_GLOBAL))) {
|
|
tmp = xmlCatalogResolveURI(resource);
|
|
}
|
|
|
|
if (tmp != NULL) {
|
|
xmlFree(resource);
|
|
resource = tmp;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
if (resource == NULL)
|
|
resource = (xmlChar *) URL;
|
|
|
|
if (resource != NULL) {
|
|
if ((!xmlStrncasecmp((const xmlChar *) resource,
|
|
(const xmlChar *) "ftp://", 6)) ||
|
|
(!xmlStrncasecmp((const xmlChar *) resource,
|
|
(const xmlChar *) "http://", 7))) {
|
|
fprintf(stderr, "Attempt to load network entity %s \n", resource);
|
|
|
|
if (nonet) {
|
|
if (resource != (xmlChar *) URL)
|
|
xmlFree(resource);
|
|
return(NULL);
|
|
}
|
|
}
|
|
}
|
|
if (defaultLoader != NULL) {
|
|
input = defaultLoader((const char *) resource, ID, ctxt);
|
|
}
|
|
if (resource != (xmlChar *) URL)
|
|
xmlFree(resource);
|
|
return(input);
|
|
}
|
|
|
|
static void
|
|
xsltProcess(xmlDocPtr doc, xsltStylesheetPtr cur, const char *filename) {
|
|
xmlDocPtr res;
|
|
|
|
#ifdef LIBXML_XINCLUDE_ENABLED
|
|
if (xinclude) {
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
xmlXIncludeProcess(doc);
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr, "XInclude processing %s took %ld ms\n",
|
|
filename, msec);
|
|
}
|
|
}
|
|
#endif
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
if (output == NULL) {
|
|
if (repeat) {
|
|
int j;
|
|
|
|
for (j = 1; j < repeat; j++) {
|
|
res = xsltApplyStylesheet(cur, doc, params);
|
|
xmlFreeDoc(res);
|
|
xmlFreeDoc(doc);
|
|
#ifdef LIBXML_HTML_ENABLED
|
|
if (html)
|
|
doc = htmlParseFile(filename, NULL);
|
|
else
|
|
#endif
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
if (docbook)
|
|
doc = docbParseFile(filename, NULL);
|
|
else
|
|
#endif
|
|
doc = xmlParseFile(filename);
|
|
}
|
|
}
|
|
if (profile) {
|
|
res = xsltProfileStylesheet(cur, doc, params, stderr);
|
|
} else {
|
|
res = xsltApplyStylesheet(cur, doc, params);
|
|
}
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
if (repeat)
|
|
fprintf(stderr,
|
|
"Applying stylesheet %d times took %ld ms\n",
|
|
repeat, msec);
|
|
else
|
|
fprintf(stderr,
|
|
"Applying stylesheet took %ld ms\n", msec);
|
|
}
|
|
xmlFreeDoc(doc);
|
|
if (res == NULL) {
|
|
fprintf(stderr, "no result for %s\n", filename);
|
|
return;
|
|
}
|
|
if (noout) {
|
|
xmlFreeDoc(res);
|
|
return;
|
|
}
|
|
#ifdef LIBXML_DEBUG_ENABLED
|
|
if (debug)
|
|
xmlDebugDumpDocument(stdout, res);
|
|
else {
|
|
#endif
|
|
if (cur->methodURI == NULL) {
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
xsltSaveResultToFile(stdout, res, cur);
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr, "Saving result took %ld ms\n",
|
|
msec);
|
|
}
|
|
} else {
|
|
if (xmlStrEqual
|
|
(cur->method, (const xmlChar *) "xhtml")) {
|
|
fprintf(stderr, "non standard output xhtml\n");
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
xsltSaveResultToFile(stdout, res, cur);
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec +=
|
|
(end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr,
|
|
"Saving result took %ld ms\n",
|
|
msec);
|
|
}
|
|
} else {
|
|
fprintf(stderr,
|
|
"Unsupported non standard output %s\n",
|
|
cur->method);
|
|
}
|
|
}
|
|
#ifdef LIBXML_DEBUG_ENABLED
|
|
}
|
|
#endif
|
|
|
|
xmlFreeDoc(res);
|
|
} else {
|
|
xsltRunStylesheet(cur, doc, params, output, NULL, NULL);
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr,
|
|
"Running stylesheet and saving result took %ld ms\n",
|
|
msec);
|
|
}
|
|
xmlFreeDoc(doc);
|
|
}
|
|
}
|
|
|
|
static void usage(const char *name) {
|
|
printf("Usage: %s [options] stylesheet file [file ...]\n", name);
|
|
printf(" Options:\n");
|
|
printf(" --version or -V: show the version of libxml and libxslt used\n");
|
|
printf(" --verbose or -v: show logs of what's happening\n");
|
|
printf(" --output file or -o file: save to a given file\n");
|
|
printf(" --timing: display the time used\n");
|
|
printf(" --repeat: run the transformation 20 times\n");
|
|
printf(" --debug: dump the tree of the result instead\n");
|
|
printf(" --novalid: skip the Dtd loading phase\n");
|
|
printf(" --noout: do not dump the result\n");
|
|
printf(" --maxdepth val : increase the maximum depth\n");
|
|
#ifdef LIBXML_HTML_ENABLED
|
|
printf(" --html: the input document is(are) an HTML file(s)\n");
|
|
#endif
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
printf(" --docbook: the input document is SGML docbook\n");
|
|
#endif
|
|
printf(" --param name value : pass a (parameter,value) pair\n");
|
|
printf(" string values must be quoted like \"'string'\"\n");
|
|
printf(" --nonet refuse to fetch DTDs or entities over network\n");
|
|
printf(" --warnnet warn against fetching over the network\n");
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
printf(" --catalogs : use the catalogs from $SGML_CATALOG_FILES\n");
|
|
#endif
|
|
#ifdef LIBXML_XINCLUDE_ENABLED
|
|
printf(" --xinclude : do XInclude processing on document intput\n");
|
|
#endif
|
|
printf(" --profile or --norman : dump profiling informations \n");
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
int i;
|
|
xsltStylesheetPtr cur = NULL;
|
|
xmlDocPtr doc, style;
|
|
|
|
if (argc <= 1) {
|
|
usage(argv[0]);
|
|
return (1);
|
|
}
|
|
|
|
xmlInitMemory();
|
|
|
|
LIBXML_TEST_VERSION
|
|
|
|
defaultLoader = xmlGetExternalEntityLoader();
|
|
xmlLineNumbersDefault(1);
|
|
|
|
if (novalid == 0)
|
|
xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS;
|
|
else
|
|
xmlLoadExtDtdDefaultValue = 0;
|
|
for (i = 1; i < argc; i++) {
|
|
if (!strcmp(argv[i], "-"))
|
|
break;
|
|
|
|
if (argv[i][0] != '-')
|
|
continue;
|
|
#ifdef LIBXML_DEBUG_ENABLED
|
|
if ((!strcmp(argv[i], "-debug")) || (!strcmp(argv[i], "--debug"))) {
|
|
debug++;
|
|
} else
|
|
#endif
|
|
if ((!strcmp(argv[i], "-v")) ||
|
|
(!strcmp(argv[i], "-verbose")) ||
|
|
(!strcmp(argv[i], "--verbose"))) {
|
|
xsltSetGenericDebugFunc(stderr, NULL);
|
|
} else if ((!strcmp(argv[i], "-o")) ||
|
|
(!strcmp(argv[i], "-output")) ||
|
|
(!strcmp(argv[i], "--output"))) {
|
|
i++;
|
|
output = argv[i++];
|
|
} else if ((!strcmp(argv[i], "-V")) ||
|
|
(!strcmp(argv[i], "-version")) ||
|
|
(!strcmp(argv[i], "--version"))) {
|
|
printf("Using libxml %s, libxslt %s and libexslt %s\n",
|
|
xmlParserVersion, xsltEngineVersion, exsltLibraryVersion);
|
|
printf
|
|
("xsltproc was compiled against libxml %d, libxslt %d and libexslt %d\n",
|
|
LIBXML_VERSION, LIBXSLT_VERSION, LIBEXSLT_VERSION);
|
|
printf("libxslt %d was compiled against libxml %d\n",
|
|
xsltLibxsltVersion, xsltLibxmlVersion);
|
|
printf("libexslt %d was compiled against libxml %d\n",
|
|
exsltLibexsltVersion, exsltLibxmlVersion);
|
|
} else if ((!strcmp(argv[i], "-repeat"))
|
|
|| (!strcmp(argv[i], "--repeat"))) {
|
|
if (repeat == 0)
|
|
repeat = 20;
|
|
else
|
|
repeat = 100;
|
|
} else if ((!strcmp(argv[i], "-novalid")) ||
|
|
(!strcmp(argv[i], "--novalid"))) {
|
|
novalid++;
|
|
} else if ((!strcmp(argv[i], "-noout")) ||
|
|
(!strcmp(argv[i], "--noout"))) {
|
|
noout++;
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
} else if ((!strcmp(argv[i], "-docbook")) ||
|
|
(!strcmp(argv[i], "--docbook"))) {
|
|
docbook++;
|
|
#endif
|
|
#ifdef LIBXML_HTML_ENABLED
|
|
} else if ((!strcmp(argv[i], "-html")) ||
|
|
(!strcmp(argv[i], "--html"))) {
|
|
html++;
|
|
#endif
|
|
} else if ((!strcmp(argv[i], "-timing")) ||
|
|
(!strcmp(argv[i], "--timing"))) {
|
|
timing++;
|
|
} else if ((!strcmp(argv[i], "-profile")) ||
|
|
(!strcmp(argv[i], "--profile"))) {
|
|
profile++;
|
|
} else if ((!strcmp(argv[i], "-norman")) ||
|
|
(!strcmp(argv[i], "--norman"))) {
|
|
profile++;
|
|
} else if ((!strcmp(argv[i], "-warnnet")) ||
|
|
(!strcmp(argv[i], "--warnnet"))) {
|
|
xmlSetExternalEntityLoader(xsltNoNetExternalEntityLoader);
|
|
} else if ((!strcmp(argv[i], "-nonet")) ||
|
|
(!strcmp(argv[i], "--nonet"))) {
|
|
xmlSetExternalEntityLoader(xsltNoNetExternalEntityLoader);
|
|
nonet = 1;
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
} else if ((!strcmp(argv[i], "-catalogs")) ||
|
|
(!strcmp(argv[i], "--catalogs"))) {
|
|
const char *catalogs;
|
|
|
|
catalogs = getenv("SGML_CATALOG_FILES");
|
|
if (catalogs == NULL) {
|
|
fprintf(stderr, "Variable $SGML_CATALOG_FILES not set\n");
|
|
} else {
|
|
xmlLoadCatalogs(catalogs);
|
|
}
|
|
#endif
|
|
#ifdef LIBXML_XINCLUDE_ENABLED
|
|
} else if ((!strcmp(argv[i], "-xinclude")) ||
|
|
(!strcmp(argv[i], "--xinclude"))) {
|
|
xinclude++;
|
|
xsltSetXIncludeDefault(1);
|
|
#endif
|
|
} else if ((!strcmp(argv[i], "-param")) ||
|
|
(!strcmp(argv[i], "--param"))) {
|
|
i++;
|
|
params[nbparams++] = argv[i++];
|
|
params[nbparams++] = argv[i];
|
|
if (nbparams >= 16) {
|
|
fprintf(stderr, "too many params\n");
|
|
return (1);
|
|
}
|
|
} else if ((!strcmp(argv[i], "-maxdepth")) ||
|
|
(!strcmp(argv[i], "--maxdepth"))) {
|
|
int value;
|
|
|
|
i++;
|
|
if (sscanf(argv[i], "%d", &value) == 1) {
|
|
if (value > 0)
|
|
xsltMaxDepth = value;
|
|
}
|
|
} else {
|
|
fprintf(stderr, "Unknown option %s\n", argv[i]);
|
|
usage(argv[0]);
|
|
return (1);
|
|
}
|
|
}
|
|
params[nbparams] = NULL;
|
|
|
|
/*
|
|
* Replace entities with their content.
|
|
*/
|
|
xmlSubstituteEntitiesDefault(1);
|
|
|
|
/*
|
|
* Register the EXSLT extensions and the test module
|
|
*/
|
|
exsltRegisterAll();
|
|
xsltRegisterTestModule();
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
if ((!strcmp(argv[i], "-maxdepth")) ||
|
|
(!strcmp(argv[i], "--maxdepth"))) {
|
|
i++;
|
|
continue;
|
|
} else if ((!strcmp(argv[i], "-o")) ||
|
|
(!strcmp(argv[i], "-output")) ||
|
|
(!strcmp(argv[i], "--output"))) {
|
|
i++;
|
|
continue;
|
|
}
|
|
if ((!strcmp(argv[i], "-param")) || (!strcmp(argv[i], "--param"))) {
|
|
i += 2;
|
|
continue;
|
|
}
|
|
if ((argv[i][0] != '-') || (strcmp(argv[i], "-") == 0)) {
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
style = xmlParseFile((const char *) argv[i]);
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr, "Parsing stylesheet %s took %ld ms\n",
|
|
argv[i], msec);
|
|
}
|
|
if (style == NULL) {
|
|
fprintf(stderr, "cannot parse %s\n", argv[i]);
|
|
cur = NULL;
|
|
} else {
|
|
cur = xsltLoadStylesheetPI(style);
|
|
if (cur != NULL) {
|
|
/* it is an embedded stylesheet */
|
|
xsltProcess(style, cur, argv[i]);
|
|
xsltFreeStylesheet(cur);
|
|
goto done;
|
|
}
|
|
cur = xsltParseStylesheetDoc(style);
|
|
if (cur != NULL) {
|
|
if (cur->indent == 1)
|
|
xmlIndentTreeOutput = 1;
|
|
else
|
|
xmlIndentTreeOutput = 0;
|
|
i++;
|
|
} else {
|
|
xmlFreeDoc(style);
|
|
goto done;
|
|
}
|
|
}
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
/*
|
|
* disable CDATA from being built in the document tree
|
|
*/
|
|
xmlDefaultSAXHandlerInit();
|
|
xmlDefaultSAXHandler.cdataBlock = NULL;
|
|
|
|
if ((cur != NULL) && (cur->errors == 0)) {
|
|
for (; i < argc; i++) {
|
|
doc = NULL;
|
|
if (timing)
|
|
gettimeofday(&begin, NULL);
|
|
#ifdef LIBXML_HTML_ENABLED
|
|
if (html)
|
|
doc = htmlParseFile(argv[i], NULL);
|
|
else
|
|
#endif
|
|
#ifdef LIBXML_DOCB_ENABLED
|
|
if (docbook)
|
|
doc = docbParseFile(argv[i], NULL);
|
|
else
|
|
#endif
|
|
doc = xmlParseFile(argv[i]);
|
|
if (doc == NULL) {
|
|
fprintf(stderr, "unable to parse %s\n", argv[i]);
|
|
continue;
|
|
}
|
|
if (timing) {
|
|
long msec;
|
|
|
|
gettimeofday(&end, NULL);
|
|
msec = end.tv_sec - begin.tv_sec;
|
|
msec *= 1000;
|
|
msec += (end.tv_usec - begin.tv_usec) / 1000;
|
|
fprintf(stderr, "Parsing document %s took %ld ms\n",
|
|
argv[i], msec);
|
|
}
|
|
xsltProcess(doc, cur, argv[i]);
|
|
}
|
|
}
|
|
if (cur != NULL)
|
|
xsltFreeStylesheet(cur);
|
|
done:
|
|
xsltCleanupGlobals();
|
|
xmlCleanupParser();
|
|
xmlMemoryDump();
|
|
return (0);
|
|
}
|
|
|