File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / libxml2 / xmlwriter.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Tue Feb 21 23:37:58 2012 UTC (12 years, 4 months ago) by misho
Branches: libxml2, MAIN
CVS tags: v2_8_0p0, v2_8_0, v2_7_8, HEAD
libxml2

    1: 
    2: /*
    3:  * xmlwriter.c: XML text writer implementation
    4:  *
    5:  * For license and disclaimer see the license and disclaimer of
    6:  * libxml2.
    7:  *
    8:  * alfred@mickautsch.de
    9:  */
   10: 
   11: #define IN_LIBXML
   12: #include "libxml.h"
   13: #include <string.h>
   14: 
   15: #include <libxml/xmlmemory.h>
   16: #include <libxml/parser.h>
   17: #include <libxml/uri.h>
   18: #include <libxml/HTMLtree.h>
   19: 
   20: #ifdef LIBXML_WRITER_ENABLED
   21: 
   22: #include <libxml/xmlwriter.h>
   23: 
   24: #define B64LINELEN 72
   25: #define B64CRLF "\r\n"
   26: 
   27: /*
   28:  * The following VA_COPY was coded following an example in
   29:  * the Samba project.  It may not be sufficient for some
   30:  * esoteric implementations of va_list (i.e. it may need
   31:  * something involving a memcpy) but (hopefully) will be
   32:  * sufficient for libxml2.
   33:  */
   34: #ifndef VA_COPY
   35:   #ifdef HAVE_VA_COPY
   36:     #define VA_COPY(dest, src) va_copy(dest, src)
   37:   #else
   38:     #ifdef HAVE___VA_COPY
   39:       #define VA_COPY(dest,src) __va_copy(dest, src)
   40:     #else
   41:       #define VA_COPY(dest,src) (dest) = (src)
   42:     #endif
   43:   #endif
   44: #endif
   45: 
   46: /*
   47:  * Types are kept private
   48:  */
   49: typedef enum {
   50:     XML_TEXTWRITER_NONE = 0,
   51:     XML_TEXTWRITER_NAME,
   52:     XML_TEXTWRITER_ATTRIBUTE,
   53:     XML_TEXTWRITER_TEXT,
   54:     XML_TEXTWRITER_PI,
   55:     XML_TEXTWRITER_PI_TEXT,
   56:     XML_TEXTWRITER_CDATA,
   57:     XML_TEXTWRITER_DTD,
   58:     XML_TEXTWRITER_DTD_TEXT,
   59:     XML_TEXTWRITER_DTD_ELEM,
   60:     XML_TEXTWRITER_DTD_ELEM_TEXT,
   61:     XML_TEXTWRITER_DTD_ATTL,
   62:     XML_TEXTWRITER_DTD_ATTL_TEXT,
   63:     XML_TEXTWRITER_DTD_ENTY,    /* entity */
   64:     XML_TEXTWRITER_DTD_ENTY_TEXT,
   65:     XML_TEXTWRITER_DTD_PENT,    /* parameter entity */
   66:     XML_TEXTWRITER_COMMENT
   67: } xmlTextWriterState;
   68: 
   69: typedef struct _xmlTextWriterStackEntry xmlTextWriterStackEntry;
   70: 
   71: struct _xmlTextWriterStackEntry {
   72:     xmlChar *name;
   73:     xmlTextWriterState state;
   74: };
   75: 
   76: typedef struct _xmlTextWriterNsStackEntry xmlTextWriterNsStackEntry;
   77: struct _xmlTextWriterNsStackEntry {
   78:     xmlChar *prefix;
   79:     xmlChar *uri;
   80:     xmlLinkPtr elem;
   81: };
   82: 
   83: struct _xmlTextWriter {
   84:     xmlOutputBufferPtr out;     /* output buffer */
   85:     xmlListPtr nodes;           /* element name stack */
   86:     xmlListPtr nsstack;         /* name spaces stack */
   87:     int level;
   88:     int indent;                 /* enable indent */
   89:     int doindent;               /* internal indent flag */
   90:     xmlChar *ichar;             /* indent character */
   91:     char qchar;                 /* character used for quoting attribute values */
   92:     xmlParserCtxtPtr ctxt;
   93:     int no_doc_free;
   94:     xmlDocPtr doc;
   95: };
   96: 
   97: static void xmlFreeTextWriterStackEntry(xmlLinkPtr lk);
   98: static int xmlCmpTextWriterStackEntry(const void *data0,
   99:                                       const void *data1);
  100: static int xmlTextWriterOutputNSDecl(xmlTextWriterPtr writer);
  101: static void xmlFreeTextWriterNsStackEntry(xmlLinkPtr lk);
  102: static int xmlCmpTextWriterNsStackEntry(const void *data0,
  103:                                         const void *data1);
  104: static int xmlTextWriterWriteDocCallback(void *context,
  105:                                          const xmlChar * str, int len);
  106: static int xmlTextWriterCloseDocCallback(void *context);
  107: 
  108: static xmlChar *xmlTextWriterVSprintf(const char *format, va_list argptr);
  109: static int xmlOutputBufferWriteBase64(xmlOutputBufferPtr out, int len,
  110:                                       const unsigned char *data);
  111: static void xmlTextWriterStartDocumentCallback(void *ctx);
  112: static int xmlTextWriterWriteIndent(xmlTextWriterPtr writer);
  113: static int
  114:   xmlTextWriterHandleStateDependencies(xmlTextWriterPtr writer,
  115:                                        xmlTextWriterStackEntry * p);
  116: 
  117: /**
  118:  * xmlWriterErrMsg:
  119:  * @ctxt:  a writer context
  120:  * @error:  the error number
  121:  * @msg:  the error message
  122:  *
  123:  * Handle a writer error
  124:  */
  125: static void
  126: xmlWriterErrMsg(xmlTextWriterPtr ctxt, xmlParserErrors error,
  127:                const char *msg)
  128: {
  129:     if (ctxt != NULL) {
  130: 	__xmlRaiseError(NULL, NULL, NULL, ctxt->ctxt,
  131: 	            NULL, XML_FROM_WRITER, error, XML_ERR_FATAL,
  132: 		    NULL, 0, NULL, NULL, NULL, 0, 0, "%s", msg);
  133:     } else {
  134: 	__xmlRaiseError(NULL, NULL, NULL, NULL, NULL, XML_FROM_WRITER, error,
  135:                     XML_ERR_FATAL, NULL, 0, NULL, NULL, NULL, 0, 0, "%s", msg);
  136:     }
  137: }
  138: 
  139: /**
  140:  * xmlWriterErrMsgInt:
  141:  * @ctxt:  a writer context
  142:  * @error:  the error number
  143:  * @msg:  the error message
  144:  * @val:  an int
  145:  *
  146:  * Handle a writer error
  147:  */
  148: static void
  149: xmlWriterErrMsgInt(xmlTextWriterPtr ctxt, xmlParserErrors error,
  150:                const char *msg, int val)
  151: {
  152:     if (ctxt != NULL) {
  153: 	__xmlRaiseError(NULL, NULL, NULL, ctxt->ctxt,
  154: 	            NULL, XML_FROM_WRITER, error, XML_ERR_FATAL,
  155: 		    NULL, 0, NULL, NULL, NULL, val, 0, msg, val);
  156:     } else {
  157: 	__xmlRaiseError(NULL, NULL, NULL, NULL, NULL, XML_FROM_WRITER, error,
  158:                     XML_ERR_FATAL, NULL, 0, NULL, NULL, NULL, val, 0, msg, val);
  159:     }
  160: }
  161: 
  162: /**
  163:  * xmlNewTextWriter:
  164:  * @out:  an xmlOutputBufferPtr
  165:  *
  166:  * Create a new xmlNewTextWriter structure using an xmlOutputBufferPtr
  167:  * NOTE: the @out parameter will be deallocated when the writer is closed
  168:  *       (if the call succeed.)
  169:  *
  170:  * Returns the new xmlTextWriterPtr or NULL in case of error
  171:  */
  172: xmlTextWriterPtr
  173: xmlNewTextWriter(xmlOutputBufferPtr out)
  174: {
  175:     xmlTextWriterPtr ret;
  176: 
  177:     ret = (xmlTextWriterPtr) xmlMalloc(sizeof(xmlTextWriter));
  178:     if (ret == NULL) {
  179:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  180:                         "xmlNewTextWriter : out of memory!\n");
  181:         return NULL;
  182:     }
  183:     memset(ret, 0, (size_t) sizeof(xmlTextWriter));
  184: 
  185:     ret->nodes = xmlListCreate((xmlListDeallocator)
  186:                                xmlFreeTextWriterStackEntry,
  187:                                (xmlListDataCompare)
  188:                                xmlCmpTextWriterStackEntry);
  189:     if (ret->nodes == NULL) {
  190:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  191:                         "xmlNewTextWriter : out of memory!\n");
  192:         xmlFree(ret);
  193:         return NULL;
  194:     }
  195: 
  196:     ret->nsstack = xmlListCreate((xmlListDeallocator)
  197:                                  xmlFreeTextWriterNsStackEntry,
  198:                                  (xmlListDataCompare)
  199:                                  xmlCmpTextWriterNsStackEntry);
  200:     if (ret->nsstack == NULL) {
  201:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  202:                         "xmlNewTextWriter : out of memory!\n");
  203:         xmlListDelete(ret->nodes);
  204:         xmlFree(ret);
  205:         return NULL;
  206:     }
  207: 
  208:     ret->out = out;
  209:     ret->ichar = xmlStrdup(BAD_CAST " ");
  210:     ret->qchar = '"';
  211: 
  212:     if (!ret->ichar) {
  213:         xmlListDelete(ret->nodes);
  214:         xmlListDelete(ret->nsstack);
  215:         xmlFree(ret);
  216:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  217:                         "xmlNewTextWriter : out of memory!\n");
  218:         return NULL;
  219:     }
  220: 
  221:     ret->doc = xmlNewDoc(NULL);
  222: 
  223:     ret->no_doc_free = 0;
  224: 
  225:     return ret;
  226: }
  227: 
  228: /**
  229:  * xmlNewTextWriterFilename:
  230:  * @uri:  the URI of the resource for the output
  231:  * @compression:  compress the output?
  232:  *
  233:  * Create a new xmlNewTextWriter structure with @uri as output
  234:  *
  235:  * Returns the new xmlTextWriterPtr or NULL in case of error
  236:  */
  237: xmlTextWriterPtr
  238: xmlNewTextWriterFilename(const char *uri, int compression)
  239: {
  240:     xmlTextWriterPtr ret;
  241:     xmlOutputBufferPtr out;
  242: 
  243:     out = xmlOutputBufferCreateFilename(uri, NULL, compression);
  244:     if (out == NULL) {
  245:         xmlWriterErrMsg(NULL, XML_IO_EIO,
  246:                         "xmlNewTextWriterFilename : cannot open uri\n");
  247:         return NULL;
  248:     }
  249: 
  250:     ret = xmlNewTextWriter(out);
  251:     if (ret == NULL) {
  252:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  253:                         "xmlNewTextWriterFilename : out of memory!\n");
  254:         xmlOutputBufferClose(out);
  255:         return NULL;
  256:     }
  257: 
  258:     ret->indent = 0;
  259:     ret->doindent = 0;
  260:     return ret;
  261: }
  262: 
  263: /**
  264:  * xmlNewTextWriterMemory:
  265:  * @buf:  xmlBufferPtr
  266:  * @compression:  compress the output?
  267:  *
  268:  * Create a new xmlNewTextWriter structure with @buf as output
  269:  * TODO: handle compression
  270:  *
  271:  * Returns the new xmlTextWriterPtr or NULL in case of error
  272:  */
  273: xmlTextWriterPtr
  274: xmlNewTextWriterMemory(xmlBufferPtr buf, int compression ATTRIBUTE_UNUSED)
  275: {
  276:     xmlTextWriterPtr ret;
  277:     xmlOutputBufferPtr out;
  278: 
  279: /*::todo handle compression */
  280:     out = xmlOutputBufferCreateBuffer(buf, NULL);
  281: 
  282:     if (out == NULL) {
  283:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  284:                         "xmlNewTextWriterMemory : out of memory!\n");
  285:         return NULL;
  286:     }
  287: 
  288:     ret = xmlNewTextWriter(out);
  289:     if (ret == NULL) {
  290:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
  291:                         "xmlNewTextWriterMemory : out of memory!\n");
  292:         xmlOutputBufferClose(out);
  293:         return NULL;
  294:     }
  295: 
  296:     return ret;
  297: }
  298: 
  299: /**
  300:  * xmlNewTextWriterPushParser:
  301:  * @ctxt: xmlParserCtxtPtr to hold the new XML document tree
  302:  * @compression:  compress the output?
  303:  *
  304:  * Create a new xmlNewTextWriter structure with @ctxt as output
  305:  * NOTE: the @ctxt context will be freed with the resulting writer
  306:  *       (if the call succeeds).
  307:  * TODO: handle compression
  308:  *
  309:  * Returns the new xmlTextWriterPtr or NULL in case of error
  310:  */
  311: xmlTextWriterPtr
  312: xmlNewTextWriterPushParser(xmlParserCtxtPtr ctxt,
  313:                            int compression ATTRIBUTE_UNUSED)
  314: {
  315:     xmlTextWriterPtr ret;
  316:     xmlOutputBufferPtr out;
  317: 
  318:     if (ctxt == NULL) {
  319:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  320:                         "xmlNewTextWriterPushParser : invalid context!\n");
  321:         return NULL;
  322:     }
  323: 
  324:     out = xmlOutputBufferCreateIO((xmlOutputWriteCallback)
  325:                                   xmlTextWriterWriteDocCallback,
  326:                                   (xmlOutputCloseCallback)
  327:                                   xmlTextWriterCloseDocCallback,
  328:                                   (void *) ctxt, NULL);
  329:     if (out == NULL) {
  330:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  331:                         "xmlNewTextWriterPushParser : error at xmlOutputBufferCreateIO!\n");
  332:         return NULL;
  333:     }
  334: 
  335:     ret = xmlNewTextWriter(out);
  336:     if (ret == NULL) {
  337:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  338:                         "xmlNewTextWriterPushParser : error at xmlNewTextWriter!\n");
  339:         xmlOutputBufferClose(out);
  340:         return NULL;
  341:     }
  342: 
  343:     ret->ctxt = ctxt;
  344: 
  345:     return ret;
  346: }
  347: 
  348: /**
  349:  * xmlNewTextWriterDoc:
  350:  * @doc: address of a xmlDocPtr to hold the new XML document tree
  351:  * @compression:  compress the output?
  352:  *
  353:  * Create a new xmlNewTextWriter structure with @*doc as output
  354:  *
  355:  * Returns the new xmlTextWriterPtr or NULL in case of error
  356:  */
  357: xmlTextWriterPtr
  358: xmlNewTextWriterDoc(xmlDocPtr * doc, int compression)
  359: {
  360:     xmlTextWriterPtr ret;
  361:     xmlSAXHandler saxHandler;
  362:     xmlParserCtxtPtr ctxt;
  363: 
  364:     memset(&saxHandler, '\0', sizeof(saxHandler));
  365:     xmlSAX2InitDefaultSAXHandler(&saxHandler, 1);
  366:     saxHandler.startDocument = xmlTextWriterStartDocumentCallback;
  367:     saxHandler.startElement = xmlSAX2StartElement;
  368:     saxHandler.endElement = xmlSAX2EndElement;
  369: 
  370:     ctxt = xmlCreatePushParserCtxt(&saxHandler, NULL, NULL, 0, NULL);
  371:     if (ctxt == NULL) {
  372:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  373:                 "xmlNewTextWriterDoc : error at xmlCreatePushParserCtxt!\n");
  374:         return NULL;
  375:     }
  376:     /*
  377:      * For some reason this seems to completely break if node names
  378:      * are interned.
  379:      */
  380:     ctxt->dictNames = 0;
  381: 
  382:     ctxt->myDoc = xmlNewDoc(BAD_CAST XML_DEFAULT_VERSION);
  383:     if (ctxt->myDoc == NULL) {
  384:         xmlFreeParserCtxt(ctxt);
  385:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  386:                         "xmlNewTextWriterDoc : error at xmlNewDoc!\n");
  387:         return NULL;
  388:     }
  389: 
  390:     ret = xmlNewTextWriterPushParser(ctxt, compression);
  391:     if (ret == NULL) {
  392:         xmlFreeDoc(ctxt->myDoc);
  393:         xmlFreeParserCtxt(ctxt);
  394:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  395:                 "xmlNewTextWriterDoc : error at xmlNewTextWriterPushParser!\n");
  396:         return NULL;
  397:     }
  398: 
  399:     xmlSetDocCompressMode(ctxt->myDoc, compression);
  400: 
  401:     if (doc != NULL) {
  402:         *doc = ctxt->myDoc;
  403: 	ret->no_doc_free = 1;
  404:     }
  405: 
  406:     return ret;
  407: }
  408: 
  409: /**
  410:  * xmlNewTextWriterTree:
  411:  * @doc: xmlDocPtr
  412:  * @node: xmlNodePtr or NULL for doc->children
  413:  * @compression:  compress the output?
  414:  *
  415:  * Create a new xmlNewTextWriter structure with @doc as output
  416:  * starting at @node
  417:  *
  418:  * Returns the new xmlTextWriterPtr or NULL in case of error
  419:  */
  420: xmlTextWriterPtr
  421: xmlNewTextWriterTree(xmlDocPtr doc, xmlNodePtr node, int compression)
  422: {
  423:     xmlTextWriterPtr ret;
  424:     xmlSAXHandler saxHandler;
  425:     xmlParserCtxtPtr ctxt;
  426: 
  427:     if (doc == NULL) {
  428:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  429:                         "xmlNewTextWriterTree : invalid document tree!\n");
  430:         return NULL;
  431:     }
  432: 
  433:     memset(&saxHandler, '\0', sizeof(saxHandler));
  434:     xmlSAX2InitDefaultSAXHandler(&saxHandler, 1);
  435:     saxHandler.startDocument = xmlTextWriterStartDocumentCallback;
  436:     saxHandler.startElement = xmlSAX2StartElement;
  437:     saxHandler.endElement = xmlSAX2EndElement;
  438: 
  439:     ctxt = xmlCreatePushParserCtxt(&saxHandler, NULL, NULL, 0, NULL);
  440:     if (ctxt == NULL) {
  441:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  442:                         "xmlNewTextWriterDoc : error at xmlCreatePushParserCtxt!\n");
  443:         return NULL;
  444:     }
  445:     /*
  446:      * For some reason this seems to completely break if node names
  447:      * are interned.
  448:      */
  449:     ctxt->dictNames = 0;
  450: 
  451:     ret = xmlNewTextWriterPushParser(ctxt, compression);
  452:     if (ret == NULL) {
  453:         xmlFreeParserCtxt(ctxt);
  454:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
  455:                         "xmlNewTextWriterDoc : error at xmlNewTextWriterPushParser!\n");
  456:         return NULL;
  457:     }
  458: 
  459:     ctxt->myDoc = doc;
  460:     ctxt->node = node;
  461:     ret->no_doc_free = 1;
  462: 
  463:     xmlSetDocCompressMode(doc, compression);
  464: 
  465:     return ret;
  466: }
  467: 
  468: /**
  469:  * xmlFreeTextWriter:
  470:  * @writer:  the xmlTextWriterPtr
  471:  *
  472:  * Deallocate all the resources associated to the writer
  473:  */
  474: void
  475: xmlFreeTextWriter(xmlTextWriterPtr writer)
  476: {
  477:     if (writer == NULL)
  478:         return;
  479: 
  480:     if (writer->out != NULL)
  481:         xmlOutputBufferClose(writer->out);
  482: 
  483:     if (writer->nodes != NULL)
  484:         xmlListDelete(writer->nodes);
  485: 
  486:     if (writer->nsstack != NULL)
  487:         xmlListDelete(writer->nsstack);
  488: 
  489:     if (writer->ctxt != NULL) {
  490:         if ((writer->ctxt->myDoc != NULL) && (writer->no_doc_free == 0)) {
  491: 	    xmlFreeDoc(writer->ctxt->myDoc);
  492: 	    writer->ctxt->myDoc = NULL;
  493: 	}
  494:         xmlFreeParserCtxt(writer->ctxt);
  495:     }
  496: 
  497:     if (writer->doc != NULL)
  498:         xmlFreeDoc(writer->doc);
  499: 
  500:     if (writer->ichar != NULL)
  501:         xmlFree(writer->ichar);
  502:     xmlFree(writer);
  503: }
  504: 
  505: /**
  506:  * xmlTextWriterStartDocument:
  507:  * @writer:  the xmlTextWriterPtr
  508:  * @version:  the xml version ("1.0") or NULL for default ("1.0")
  509:  * @encoding:  the encoding or NULL for default
  510:  * @standalone: "yes" or "no" or NULL for default
  511:  *
  512:  * Start a new xml document
  513:  *
  514:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  515:  */
  516: int
  517: xmlTextWriterStartDocument(xmlTextWriterPtr writer, const char *version,
  518:                            const char *encoding, const char *standalone)
  519: {
  520:     int count;
  521:     int sum;
  522:     xmlLinkPtr lk;
  523:     xmlCharEncodingHandlerPtr encoder;
  524: 
  525:     if ((writer == NULL) || (writer->out == NULL)) {
  526:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  527:                         "xmlTextWriterStartDocument : invalid writer!\n");
  528:         return -1;
  529:     }
  530: 
  531:     lk = xmlListFront(writer->nodes);
  532:     if ((lk != NULL) && (xmlLinkGetData(lk) != NULL)) {
  533:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  534:                         "xmlTextWriterStartDocument : not allowed in this context!\n");
  535:         return -1;
  536:     }
  537: 
  538:     encoder = NULL;
  539:     if (encoding != NULL) {
  540:         encoder = xmlFindCharEncodingHandler(encoding);
  541:         if (encoder == NULL) {
  542:             xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
  543:                             "xmlTextWriterStartDocument : out of memory!\n");
  544:             return -1;
  545:         }
  546:     }
  547: 
  548:     writer->out->encoder = encoder;
  549:     if (encoder != NULL) {
  550: 	if (writer->out->conv == NULL) {
  551: 	    writer->out->conv = xmlBufferCreateSize(4000);
  552: 	}
  553:         xmlCharEncOutFunc(encoder, writer->out->conv, NULL);
  554:         if ((writer->doc != NULL) && (writer->doc->encoding == NULL))
  555:             writer->doc->encoding = xmlStrdup((xmlChar *)writer->out->encoder->name);
  556:     } else
  557:         writer->out->conv = NULL;
  558: 
  559:     sum = 0;
  560:     count = xmlOutputBufferWriteString(writer->out, "<?xml version=");
  561:     if (count < 0)
  562:         return -1;
  563:     sum += count;
  564:     count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  565:     if (count < 0)
  566:         return -1;
  567:     sum += count;
  568:     if (version != 0)
  569:         count = xmlOutputBufferWriteString(writer->out, version);
  570:     else
  571:         count = xmlOutputBufferWriteString(writer->out, "1.0");
  572:     if (count < 0)
  573:         return -1;
  574:     sum += count;
  575:     count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  576:     if (count < 0)
  577:         return -1;
  578:     sum += count;
  579:     if (writer->out->encoder != 0) {
  580:         count = xmlOutputBufferWriteString(writer->out, " encoding=");
  581:         if (count < 0)
  582:             return -1;
  583:         sum += count;
  584:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  585:         if (count < 0)
  586:             return -1;
  587:         sum += count;
  588:         count =
  589:             xmlOutputBufferWriteString(writer->out,
  590:                                        writer->out->encoder->name);
  591:         if (count < 0)
  592:             return -1;
  593:         sum += count;
  594:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  595:         if (count < 0)
  596:             return -1;
  597:         sum += count;
  598:     }
  599: 
  600:     if (standalone != 0) {
  601:         count = xmlOutputBufferWriteString(writer->out, " standalone=");
  602:         if (count < 0)
  603:             return -1;
  604:         sum += count;
  605:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  606:         if (count < 0)
  607:             return -1;
  608:         sum += count;
  609:         count = xmlOutputBufferWriteString(writer->out, standalone);
  610:         if (count < 0)
  611:             return -1;
  612:         sum += count;
  613:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
  614:         if (count < 0)
  615:             return -1;
  616:         sum += count;
  617:     }
  618: 
  619:     count = xmlOutputBufferWriteString(writer->out, "?>\n");
  620:     if (count < 0)
  621:         return -1;
  622:     sum += count;
  623: 
  624:     return sum;
  625: }
  626: 
  627: /**
  628:  * xmlTextWriterEndDocument:
  629:  * @writer:  the xmlTextWriterPtr
  630:  *
  631:  * End an xml document. All open elements are closed, and
  632:  * the content is flushed to the output.
  633:  *
  634:  * Returns the bytes written or -1 in case of error
  635:  */
  636: int
  637: xmlTextWriterEndDocument(xmlTextWriterPtr writer)
  638: {
  639:     int count;
  640:     int sum;
  641:     xmlLinkPtr lk;
  642:     xmlTextWriterStackEntry *p;
  643: 
  644:     if (writer == NULL) {
  645:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  646:                         "xmlTextWriterEndDocument : invalid writer!\n");
  647:         return -1;
  648:     }
  649: 
  650:     sum = 0;
  651:     while ((lk = xmlListFront(writer->nodes)) != NULL) {
  652:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
  653:         if (p == 0)
  654:             break;
  655:         switch (p->state) {
  656:             case XML_TEXTWRITER_NAME:
  657:             case XML_TEXTWRITER_ATTRIBUTE:
  658:             case XML_TEXTWRITER_TEXT:
  659:                 count = xmlTextWriterEndElement(writer);
  660:                 if (count < 0)
  661:                     return -1;
  662:                 sum += count;
  663:                 break;
  664:             case XML_TEXTWRITER_PI:
  665:             case XML_TEXTWRITER_PI_TEXT:
  666:                 count = xmlTextWriterEndPI(writer);
  667:                 if (count < 0)
  668:                     return -1;
  669:                 sum += count;
  670:                 break;
  671:             case XML_TEXTWRITER_CDATA:
  672:                 count = xmlTextWriterEndCDATA(writer);
  673:                 if (count < 0)
  674:                     return -1;
  675:                 sum += count;
  676:                 break;
  677:             case XML_TEXTWRITER_DTD:
  678:             case XML_TEXTWRITER_DTD_TEXT:
  679:             case XML_TEXTWRITER_DTD_ELEM:
  680:             case XML_TEXTWRITER_DTD_ELEM_TEXT:
  681:             case XML_TEXTWRITER_DTD_ATTL:
  682:             case XML_TEXTWRITER_DTD_ATTL_TEXT:
  683:             case XML_TEXTWRITER_DTD_ENTY:
  684:             case XML_TEXTWRITER_DTD_ENTY_TEXT:
  685:             case XML_TEXTWRITER_DTD_PENT:
  686:                 count = xmlTextWriterEndDTD(writer);
  687:                 if (count < 0)
  688:                     return -1;
  689:                 sum += count;
  690:                 break;
  691:             case XML_TEXTWRITER_COMMENT:
  692:                 count = xmlTextWriterEndComment(writer);
  693:                 if (count < 0)
  694:                     return -1;
  695:                 sum += count;
  696:                 break;
  697:             default:
  698:                 break;
  699:         }
  700:     }
  701: 
  702:     if (!writer->indent) {
  703:         count = xmlOutputBufferWriteString(writer->out, "\n");
  704:         if (count < 0)
  705:             return -1;
  706:         sum += count;
  707:     }
  708: 
  709:     sum += xmlTextWriterFlush(writer);
  710: 
  711:     return sum;
  712: }
  713: 
  714: /**
  715:  * xmlTextWriterStartComment:
  716:  * @writer:  the xmlTextWriterPtr
  717:  *
  718:  * Start an xml comment.
  719:  *
  720:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  721:  */
  722: int
  723: xmlTextWriterStartComment(xmlTextWriterPtr writer)
  724: {
  725:     int count;
  726:     int sum;
  727:     xmlLinkPtr lk;
  728:     xmlTextWriterStackEntry *p;
  729: 
  730:     if (writer == NULL) {
  731:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  732:                         "xmlTextWriterStartComment : invalid writer!\n");
  733:         return -1;
  734:     }
  735: 
  736:     sum = 0;
  737:     lk = xmlListFront(writer->nodes);
  738:     if (lk != 0) {
  739:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
  740:         if (p != 0) {
  741:             switch (p->state) {
  742:                 case XML_TEXTWRITER_TEXT:
  743:                 case XML_TEXTWRITER_NONE:
  744:                     break;
  745:                 case XML_TEXTWRITER_NAME:
  746:                     /* Output namespace declarations */
  747:                     count = xmlTextWriterOutputNSDecl(writer);
  748:                     if (count < 0)
  749:                         return -1;
  750:                     sum += count;
  751:                     count = xmlOutputBufferWriteString(writer->out, ">");
  752:                     if (count < 0)
  753:                         return -1;
  754:                     sum += count;
  755:                     if (writer->indent) {
  756:                         count =
  757:                             xmlOutputBufferWriteString(writer->out, "\n");
  758:                         if (count < 0)
  759:                             return -1;
  760:                         sum += count;
  761:                     }
  762:                     p->state = XML_TEXTWRITER_TEXT;
  763:                     break;
  764:                 default:
  765:                     return -1;
  766:             }
  767:         }
  768:     }
  769: 
  770:     p = (xmlTextWriterStackEntry *)
  771:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
  772:     if (p == 0) {
  773:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
  774:                         "xmlTextWriterStartElement : out of memory!\n");
  775:         return -1;
  776:     }
  777: 
  778:     p->name = NULL;
  779:     p->state = XML_TEXTWRITER_COMMENT;
  780: 
  781:     xmlListPushFront(writer->nodes, p);
  782: 
  783:     if (writer->indent) {
  784:         count = xmlTextWriterWriteIndent(writer);
  785:         if (count < 0)
  786:             return -1;
  787:         sum += count;
  788:     }
  789: 
  790:     count = xmlOutputBufferWriteString(writer->out, "<!--");
  791:     if (count < 0)
  792:         return -1;
  793:     sum += count;
  794: 
  795:     return sum;
  796: }
  797: 
  798: /**
  799:  * xmlTextWriterEndComment:
  800:  * @writer:  the xmlTextWriterPtr
  801:  *
  802:  * End the current xml coment.
  803:  *
  804:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  805:  */
  806: int
  807: xmlTextWriterEndComment(xmlTextWriterPtr writer)
  808: {
  809:     int count;
  810:     int sum;
  811:     xmlLinkPtr lk;
  812:     xmlTextWriterStackEntry *p;
  813: 
  814:     if (writer == NULL) {
  815:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  816:                         "xmlTextWriterEndComment : invalid writer!\n");
  817:         return -1;
  818:     }
  819: 
  820:     lk = xmlListFront(writer->nodes);
  821:     if (lk == 0) {
  822:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  823:                         "xmlTextWriterEndComment : not allowed in this context!\n");
  824:         return -1;
  825:     }
  826: 
  827:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
  828:     if (p == 0)
  829:         return -1;
  830: 
  831:     sum = 0;
  832:     switch (p->state) {
  833:         case XML_TEXTWRITER_COMMENT:
  834:             count = xmlOutputBufferWriteString(writer->out, "-->");
  835:             if (count < 0)
  836:                 return -1;
  837:             sum += count;
  838:             break;
  839:         default:
  840:             return -1;
  841:     }
  842: 
  843:     if (writer->indent) {
  844:         count = xmlOutputBufferWriteString(writer->out, "\n");
  845:         if (count < 0)
  846:             return -1;
  847:         sum += count;
  848:     }
  849: 
  850:     xmlListPopFront(writer->nodes);
  851:     return sum;
  852: }
  853: 
  854: /**
  855:  * xmlTextWriterWriteFormatComment:
  856:  * @writer:  the xmlTextWriterPtr
  857:  * @format:  format string (see printf)
  858:  * @...:  extra parameters for the format
  859:  *
  860:  * Write an xml comment.
  861:  *
  862:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  863:  */
  864: int XMLCDECL
  865: xmlTextWriterWriteFormatComment(xmlTextWriterPtr writer,
  866:                                 const char *format, ...)
  867: {
  868:     int rc;
  869:     va_list ap;
  870: 
  871:     va_start(ap, format);
  872: 
  873:     rc = xmlTextWriterWriteVFormatComment(writer, format, ap);
  874: 
  875:     va_end(ap);
  876:     return rc;
  877: }
  878: 
  879: /**
  880:  * xmlTextWriterWriteVFormatComment:
  881:  * @writer:  the xmlTextWriterPtr
  882:  * @format:  format string (see printf)
  883:  * @argptr:  pointer to the first member of the variable argument list.
  884:  *
  885:  * Write an xml comment.
  886:  *
  887:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  888:  */
  889: int
  890: xmlTextWriterWriteVFormatComment(xmlTextWriterPtr writer,
  891:                                  const char *format, va_list argptr)
  892: {
  893:     int rc;
  894:     xmlChar *buf;
  895: 
  896:     if (writer == NULL) {
  897:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
  898:                         "xmlTextWriterWriteVFormatComment : invalid writer!\n");
  899:         return -1;
  900:     }
  901: 
  902:     buf = xmlTextWriterVSprintf(format, argptr);
  903:     if (buf == NULL)
  904:         return -1;
  905: 
  906:     rc = xmlTextWriterWriteComment(writer, buf);
  907: 
  908:     xmlFree(buf);
  909:     return rc;
  910: }
  911: 
  912: /**
  913:  * xmlTextWriterWriteComment:
  914:  * @writer:  the xmlTextWriterPtr
  915:  * @content:  comment string
  916:  *
  917:  * Write an xml comment.
  918:  *
  919:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  920:  */
  921: int
  922: xmlTextWriterWriteComment(xmlTextWriterPtr writer, const xmlChar * content)
  923: {
  924:     int count;
  925:     int sum;
  926: 
  927:     sum = 0;
  928:     count = xmlTextWriterStartComment(writer);
  929:     if (count < 0)
  930:         return -1;
  931:     sum += count;
  932:     count = xmlTextWriterWriteString(writer, content);
  933:     if (count < 0)
  934:         return -1;
  935:     sum += count;
  936:     count = xmlTextWriterEndComment(writer);
  937:     if (count < 0)
  938:         return -1;
  939:     sum += count;
  940: 
  941:     return sum;
  942: }
  943: 
  944: /**
  945:  * xmlTextWriterStartElement:
  946:  * @writer:  the xmlTextWriterPtr
  947:  * @name:  element name
  948:  *
  949:  * Start an xml element.
  950:  *
  951:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
  952:  */
  953: int
  954: xmlTextWriterStartElement(xmlTextWriterPtr writer, const xmlChar * name)
  955: {
  956:     int count;
  957:     int sum;
  958:     xmlLinkPtr lk;
  959:     xmlTextWriterStackEntry *p;
  960: 
  961:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
  962:         return -1;
  963: 
  964:     sum = 0;
  965:     lk = xmlListFront(writer->nodes);
  966:     if (lk != 0) {
  967:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
  968:         if (p != 0) {
  969:             switch (p->state) {
  970:                 case XML_TEXTWRITER_PI:
  971:                 case XML_TEXTWRITER_PI_TEXT:
  972:                     return -1;
  973:                 case XML_TEXTWRITER_NONE:
  974:                     break;
  975: 				case XML_TEXTWRITER_ATTRIBUTE:
  976: 					count = xmlTextWriterEndAttribute(writer);
  977: 					if (count < 0)
  978: 						return -1;
  979: 					sum += count;
  980: 					/* fallthrough */
  981:                 case XML_TEXTWRITER_NAME:
  982:                     /* Output namespace declarations */
  983:                     count = xmlTextWriterOutputNSDecl(writer);
  984:                     if (count < 0)
  985:                         return -1;
  986:                     sum += count;
  987:                     count = xmlOutputBufferWriteString(writer->out, ">");
  988:                     if (count < 0)
  989:                         return -1;
  990:                     sum += count;
  991:                     if (writer->indent)
  992:                         count =
  993:                             xmlOutputBufferWriteString(writer->out, "\n");
  994:                     p->state = XML_TEXTWRITER_TEXT;
  995:                     break;
  996:                 default:
  997:                     break;
  998:             }
  999:         }
 1000:     }
 1001: 
 1002:     p = (xmlTextWriterStackEntry *)
 1003:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 1004:     if (p == 0) {
 1005:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 1006:                         "xmlTextWriterStartElement : out of memory!\n");
 1007:         return -1;
 1008:     }
 1009: 
 1010:     p->name = xmlStrdup(name);
 1011:     if (p->name == 0) {
 1012:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 1013:                         "xmlTextWriterStartElement : out of memory!\n");
 1014:         xmlFree(p);
 1015:         return -1;
 1016:     }
 1017:     p->state = XML_TEXTWRITER_NAME;
 1018: 
 1019:     xmlListPushFront(writer->nodes, p);
 1020: 
 1021:     if (writer->indent) {
 1022:         count = xmlTextWriterWriteIndent(writer);
 1023:         sum += count;
 1024:     }
 1025: 
 1026:     count = xmlOutputBufferWriteString(writer->out, "<");
 1027:     if (count < 0)
 1028:         return -1;
 1029:     sum += count;
 1030:     count =
 1031:         xmlOutputBufferWriteString(writer->out, (const char *) p->name);
 1032:     if (count < 0)
 1033:         return -1;
 1034:     sum += count;
 1035: 
 1036:     return sum;
 1037: }
 1038: 
 1039: /**
 1040:  * xmlTextWriterStartElementNS:
 1041:  * @writer:  the xmlTextWriterPtr
 1042:  * @prefix:  namespace prefix or NULL
 1043:  * @name:  element local name
 1044:  * @namespaceURI:  namespace URI or NULL
 1045:  *
 1046:  * Start an xml element with namespace support.
 1047:  *
 1048:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1049:  */
 1050: int
 1051: xmlTextWriterStartElementNS(xmlTextWriterPtr writer,
 1052:                             const xmlChar * prefix, const xmlChar * name,
 1053:                             const xmlChar * namespaceURI)
 1054: {
 1055:     int count;
 1056:     int sum;
 1057:     xmlChar *buf;
 1058: 
 1059:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
 1060:         return -1;
 1061: 
 1062:     buf = NULL;
 1063:     if (prefix != 0) {
 1064:         buf = xmlStrdup(prefix);
 1065:         buf = xmlStrcat(buf, BAD_CAST ":");
 1066:     }
 1067:     buf = xmlStrcat(buf, name);
 1068: 
 1069:     sum = 0;
 1070:     count = xmlTextWriterStartElement(writer, buf);
 1071:     xmlFree(buf);
 1072:     if (count < 0)
 1073:         return -1;
 1074:     sum += count;
 1075: 
 1076:     if (namespaceURI != 0) {
 1077:         xmlTextWriterNsStackEntry *p = (xmlTextWriterNsStackEntry *) 
 1078:         xmlMalloc(sizeof(xmlTextWriterNsStackEntry));
 1079:         if (p == 0) {
 1080:             xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY, 
 1081:                             "xmlTextWriterStartElementNS : out of memory!\n");
 1082:             return -1;
 1083:         }
 1084: 
 1085:         buf = xmlStrdup(BAD_CAST "xmlns");
 1086:         if (prefix != 0) {
 1087:             buf = xmlStrcat(buf, BAD_CAST ":");
 1088:             buf = xmlStrcat(buf, prefix);
 1089:         }
 1090: 
 1091:         p->prefix = buf;
 1092:         p->uri = xmlStrdup(namespaceURI);
 1093:         if (p->uri == 0) {
 1094:             xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 1095:                             "xmlTextWriterStartElementNS : out of memory!\n");
 1096:             xmlFree(p);
 1097:             return -1;
 1098:         }
 1099:         p->elem = xmlListFront(writer->nodes);
 1100: 
 1101:         xmlListPushFront(writer->nsstack, p);
 1102:     }
 1103: 
 1104:     return sum;
 1105: }
 1106: 
 1107: /**
 1108:  * xmlTextWriterEndElement:
 1109:  * @writer:  the xmlTextWriterPtr
 1110:  *
 1111:  * End the current xml element.
 1112:  *
 1113:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1114:  */
 1115: int
 1116: xmlTextWriterEndElement(xmlTextWriterPtr writer)
 1117: {
 1118:     int count;
 1119:     int sum;
 1120:     xmlLinkPtr lk;
 1121:     xmlTextWriterStackEntry *p;
 1122: 
 1123:     if (writer == NULL)
 1124:         return -1;
 1125: 
 1126:     lk = xmlListFront(writer->nodes);
 1127:     if (lk == 0) {
 1128:         xmlListDelete(writer->nsstack);
 1129:         writer->nsstack = NULL;
 1130:         return -1;
 1131:     }
 1132: 
 1133:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1134:     if (p == 0) {
 1135:         xmlListDelete(writer->nsstack);
 1136:         writer->nsstack = NULL;
 1137:         return -1;
 1138:     }
 1139: 
 1140:     sum = 0;
 1141:     switch (p->state) {
 1142:         case XML_TEXTWRITER_ATTRIBUTE:
 1143:             count = xmlTextWriterEndAttribute(writer);
 1144:             if (count < 0) {
 1145:                 xmlListDelete(writer->nsstack);
 1146:                 writer->nsstack = NULL;
 1147:                 return -1;
 1148:             }
 1149:             sum += count;
 1150:             /* fallthrough */
 1151:         case XML_TEXTWRITER_NAME:
 1152:             /* Output namespace declarations */
 1153:             count = xmlTextWriterOutputNSDecl(writer);
 1154:             if (count < 0)
 1155:                 return -1;
 1156:             sum += count;
 1157: 
 1158:             if (writer->indent) /* next element needs indent */
 1159:                 writer->doindent = 1;
 1160:             count = xmlOutputBufferWriteString(writer->out, "/>");
 1161:             if (count < 0)
 1162:                 return -1;
 1163:             sum += count;
 1164:             break;
 1165:         case XML_TEXTWRITER_TEXT:
 1166:             if ((writer->indent) && (writer->doindent)) {
 1167:                 count = xmlTextWriterWriteIndent(writer);
 1168:                 sum += count;
 1169:                 writer->doindent = 1;
 1170:             } else
 1171:                 writer->doindent = 1;
 1172:             count = xmlOutputBufferWriteString(writer->out, "</");
 1173:             if (count < 0)
 1174:                 return -1;
 1175:             sum += count;
 1176:             count = xmlOutputBufferWriteString(writer->out,
 1177:                                                (const char *) p->name);
 1178:             if (count < 0)
 1179:                 return -1;
 1180:             sum += count;
 1181:             count = xmlOutputBufferWriteString(writer->out, ">");
 1182:             if (count < 0)
 1183:                 return -1;
 1184:             sum += count;
 1185:             break;
 1186:         default:
 1187:             return -1;
 1188:     }
 1189: 
 1190:     if (writer->indent) {
 1191:         count = xmlOutputBufferWriteString(writer->out, "\n");
 1192:         sum += count;
 1193:     }
 1194: 
 1195:     xmlListPopFront(writer->nodes);
 1196:     return sum;
 1197: }
 1198: 
 1199: /**
 1200:  * xmlTextWriterFullEndElement:
 1201:  * @writer:  the xmlTextWriterPtr
 1202:  *
 1203:  * End the current xml element. Writes an end tag even if the element is empty
 1204:  *
 1205:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1206:  */
 1207: int
 1208: xmlTextWriterFullEndElement(xmlTextWriterPtr writer)
 1209: {
 1210:     int count;
 1211:     int sum;
 1212:     xmlLinkPtr lk;
 1213:     xmlTextWriterStackEntry *p;
 1214: 
 1215:     if (writer == NULL)
 1216:         return -1;
 1217: 
 1218:     lk = xmlListFront(writer->nodes);
 1219:     if (lk == 0)
 1220:         return -1;
 1221: 
 1222:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1223:     if (p == 0)
 1224:         return -1;
 1225: 
 1226:     sum = 0;
 1227:     switch (p->state) {
 1228:         case XML_TEXTWRITER_ATTRIBUTE:
 1229:             count = xmlTextWriterEndAttribute(writer);
 1230:             if (count < 0)
 1231:                 return -1;
 1232:             sum += count;
 1233:             /* fallthrough */
 1234:         case XML_TEXTWRITER_NAME:
 1235:             /* Output namespace declarations */
 1236:             count = xmlTextWriterOutputNSDecl(writer);
 1237:             if (count < 0)
 1238:                 return -1;
 1239:             sum += count;
 1240: 
 1241:             count = xmlOutputBufferWriteString(writer->out, ">");
 1242:             if (count < 0)
 1243:                 return -1;
 1244:             sum += count;
 1245:             if (writer->indent)
 1246:                 writer->doindent = 0;
 1247:             /* fallthrough */
 1248:         case XML_TEXTWRITER_TEXT:
 1249:             if ((writer->indent) && (writer->doindent)) {
 1250:                 count = xmlTextWriterWriteIndent(writer);
 1251:                 sum += count;
 1252:                 writer->doindent = 1;
 1253:             } else
 1254:                 writer->doindent = 1;
 1255:             count = xmlOutputBufferWriteString(writer->out, "</");
 1256:             if (count < 0)
 1257:                 return -1;
 1258:             sum += count;
 1259:             count = xmlOutputBufferWriteString(writer->out,
 1260:                                                (const char *) p->name);
 1261:             if (count < 0)
 1262:                 return -1;
 1263:             sum += count;
 1264:             count = xmlOutputBufferWriteString(writer->out, ">");
 1265:             if (count < 0)
 1266:                 return -1;
 1267:             sum += count;
 1268:             break;
 1269:         default:
 1270:             return -1;
 1271:     }
 1272: 
 1273:     if (writer->indent) {
 1274:         count = xmlOutputBufferWriteString(writer->out, "\n");
 1275:         sum += count;
 1276:     }
 1277: 
 1278:     xmlListPopFront(writer->nodes);
 1279:     return sum;
 1280: }
 1281: 
 1282: /**
 1283:  * xmlTextWriterWriteFormatRaw:
 1284:  * @writer:  the xmlTextWriterPtr
 1285:  * @format:  format string (see printf)
 1286:  * @...:  extra parameters for the format
 1287:  *
 1288:  * Write a formatted raw xml text.
 1289:  *
 1290:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1291:  */
 1292: int XMLCDECL
 1293: xmlTextWriterWriteFormatRaw(xmlTextWriterPtr writer, const char *format,
 1294:                             ...)
 1295: {
 1296:     int rc;
 1297:     va_list ap;
 1298: 
 1299:     va_start(ap, format);
 1300: 
 1301:     rc = xmlTextWriterWriteVFormatRaw(writer, format, ap);
 1302: 
 1303:     va_end(ap);
 1304:     return rc;
 1305: }
 1306: 
 1307: /**
 1308:  * xmlTextWriterWriteVFormatRaw:
 1309:  * @writer:  the xmlTextWriterPtr
 1310:  * @format:  format string (see printf)
 1311:  * @argptr:  pointer to the first member of the variable argument list.
 1312:  *
 1313:  * Write a formatted raw xml text.
 1314:  *
 1315:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1316:  */
 1317: int
 1318: xmlTextWriterWriteVFormatRaw(xmlTextWriterPtr writer, const char *format,
 1319:                              va_list argptr)
 1320: {
 1321:     int rc;
 1322:     xmlChar *buf;
 1323: 
 1324:     if (writer == NULL)
 1325:         return -1;
 1326: 
 1327:     buf = xmlTextWriterVSprintf(format, argptr);
 1328:     if (buf == NULL)
 1329:         return -1;
 1330: 
 1331:     rc = xmlTextWriterWriteRaw(writer, buf);
 1332: 
 1333:     xmlFree(buf);
 1334:     return rc;
 1335: }
 1336: 
 1337: /**
 1338:  * xmlTextWriterWriteRawLen:
 1339:  * @writer:  the xmlTextWriterPtr
 1340:  * @content:  text string
 1341:  * @len:  length of the text string
 1342:  *
 1343:  * Write an xml text.
 1344:  * TODO: what about entities and special chars??
 1345:  *
 1346:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1347:  */
 1348: int
 1349: xmlTextWriterWriteRawLen(xmlTextWriterPtr writer, const xmlChar * content,
 1350:                          int len)
 1351: {
 1352:     int count;
 1353:     int sum;
 1354:     xmlLinkPtr lk;
 1355:     xmlTextWriterStackEntry *p;
 1356: 
 1357:     if (writer == NULL) {
 1358:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 1359:                         "xmlTextWriterWriteRawLen : invalid writer!\n");
 1360:         return -1;
 1361:     }
 1362: 
 1363:     if ((content == NULL) || (len < 0)) {
 1364:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 1365:                         "xmlTextWriterWriteRawLen : invalid content!\n");
 1366:         return -1;
 1367:     }
 1368: 
 1369:     sum = 0;
 1370:     lk = xmlListFront(writer->nodes);
 1371:     if (lk != 0) {
 1372:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1373:         count = xmlTextWriterHandleStateDependencies(writer, p);
 1374:         if (count < 0)
 1375:             return -1;
 1376:         sum += count;
 1377:     }
 1378: 
 1379:     if (writer->indent)
 1380:         writer->doindent = 0;
 1381: 
 1382:     if (content != NULL) {
 1383:         count =
 1384:             xmlOutputBufferWrite(writer->out, len, (const char *) content);
 1385:         if (count < 0)
 1386:             return -1;
 1387:         sum += count;
 1388:     }
 1389: 
 1390:     return sum;
 1391: }
 1392: 
 1393: /**
 1394:  * xmlTextWriterWriteRaw:
 1395:  * @writer:  the xmlTextWriterPtr
 1396:  * @content:  text string
 1397:  *
 1398:  * Write a raw xml text.
 1399:  *
 1400:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1401:  */
 1402: int
 1403: xmlTextWriterWriteRaw(xmlTextWriterPtr writer, const xmlChar * content)
 1404: {
 1405:     return xmlTextWriterWriteRawLen(writer, content, xmlStrlen(content));
 1406: }
 1407: 
 1408: /**
 1409:  * xmlTextWriterWriteFormatString:
 1410:  * @writer:  the xmlTextWriterPtr
 1411:  * @format:  format string (see printf)
 1412:  * @...:  extra parameters for the format
 1413:  *
 1414:  * Write a formatted xml text.
 1415:  *
 1416:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1417:  */
 1418: int XMLCDECL
 1419: xmlTextWriterWriteFormatString(xmlTextWriterPtr writer, const char *format,
 1420:                                ...)
 1421: {
 1422:     int rc;
 1423:     va_list ap;
 1424: 
 1425:     if ((writer == NULL) || (format == NULL))
 1426:         return -1;
 1427: 
 1428:     va_start(ap, format);
 1429: 
 1430:     rc = xmlTextWriterWriteVFormatString(writer, format, ap);
 1431: 
 1432:     va_end(ap);
 1433:     return rc;
 1434: }
 1435: 
 1436: /**
 1437:  * xmlTextWriterWriteVFormatString:
 1438:  * @writer:  the xmlTextWriterPtr
 1439:  * @format:  format string (see printf)
 1440:  * @argptr:  pointer to the first member of the variable argument list.
 1441:  *
 1442:  * Write a formatted xml text.
 1443:  *
 1444:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1445:  */
 1446: int
 1447: xmlTextWriterWriteVFormatString(xmlTextWriterPtr writer,
 1448:                                 const char *format, va_list argptr)
 1449: {
 1450:     int rc;
 1451:     xmlChar *buf;
 1452: 
 1453:     if ((writer == NULL) || (format == NULL))
 1454:         return -1;
 1455: 
 1456:     buf = xmlTextWriterVSprintf(format, argptr);
 1457:     if (buf == NULL)
 1458:         return -1;
 1459: 
 1460:     rc = xmlTextWriterWriteString(writer, buf);
 1461: 
 1462:     xmlFree(buf);
 1463:     return rc;
 1464: }
 1465: 
 1466: /**
 1467:  * xmlTextWriterWriteString:
 1468:  * @writer:  the xmlTextWriterPtr
 1469:  * @content:  text string
 1470:  *
 1471:  * Write an xml text.
 1472:  *
 1473:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1474:  */
 1475: int
 1476: xmlTextWriterWriteString(xmlTextWriterPtr writer, const xmlChar * content)
 1477: {
 1478:     int count;
 1479:     int sum;
 1480:     xmlLinkPtr lk;
 1481:     xmlTextWriterStackEntry *p;
 1482:     xmlChar *buf;
 1483: 
 1484:     if ((writer == NULL) || (content == NULL))
 1485:         return -1;
 1486: 
 1487:     sum = 0;
 1488:     buf = (xmlChar *) content;
 1489:     lk = xmlListFront(writer->nodes);
 1490:     if (lk != 0) {
 1491:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1492:         if (p != 0) {
 1493:             switch (p->state) {
 1494:                 case XML_TEXTWRITER_NAME:
 1495:                 case XML_TEXTWRITER_TEXT:
 1496: #if 0
 1497:                     buf = NULL;
 1498: 		    xmlOutputBufferWriteEscape(writer->out, content, NULL);
 1499: #endif
 1500:                     buf = xmlEncodeSpecialChars(NULL, content);
 1501:                     break;
 1502:                 case XML_TEXTWRITER_ATTRIBUTE:
 1503:                     buf = NULL;
 1504:                     xmlAttrSerializeTxtContent(writer->out->buffer, writer->doc,
 1505:                                                NULL, content);
 1506:                     break;
 1507: 		default:
 1508: 		    break;
 1509:             }
 1510:         }
 1511:     }
 1512: 
 1513:     if (buf != NULL) {
 1514:         count = xmlTextWriterWriteRaw(writer, buf);
 1515: 
 1516:         if (buf != content)     /* buf was allocated by us, so free it */
 1517:             xmlFree(buf);
 1518: 
 1519:         if (count < 0)
 1520:             return -1;
 1521:         sum += count;
 1522:     }
 1523: 
 1524:     return sum;
 1525: }
 1526: 
 1527: /**
 1528:  * xmlOutputBufferWriteBase64:
 1529:  * @out: the xmlOutputBufferPtr
 1530:  * @data:   binary data
 1531:  * @len:  the number of bytes to encode
 1532:  *
 1533:  * Write base64 encoded data to an xmlOutputBuffer.
 1534:  * Adapted from John Walker's base64.c (http://www.fourmilab.ch/).
 1535:  *
 1536:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1537:  */
 1538: static int
 1539: xmlOutputBufferWriteBase64(xmlOutputBufferPtr out, int len,
 1540:                            const unsigned char *data)
 1541: {
 1542:     static unsigned char dtable[64] =
 1543:             {'A','B','C','D','E','F','G','H','I','J','K','L','M',
 1544: 	     'N','O','P','Q','R','S','T','U','V','W','X','Y','Z',
 1545: 	     'a','b','c','d','e','f','g','h','i','j','k','l','m',
 1546: 	     'n','o','p','q','r','s','t','u','v','w','x','y','z',
 1547: 	     '0','1','2','3','4','5','6','7','8','9','+','/'};
 1548: 
 1549:     int i;
 1550:     int linelen;
 1551:     int count;
 1552:     int sum;
 1553: 
 1554:     if ((out == NULL) || (len < 0) || (data == NULL))
 1555:         return(-1);
 1556: 
 1557:     linelen = 0;
 1558:     sum = 0;
 1559: 
 1560:     i = 0;
 1561:     while (1) {
 1562:         unsigned char igroup[3];
 1563:         unsigned char ogroup[4];
 1564:         int c;
 1565:         int n;
 1566: 
 1567:         igroup[0] = igroup[1] = igroup[2] = 0;
 1568:         for (n = 0; n < 3 && i < len; n++, i++) {
 1569:             c = data[i];
 1570:             igroup[n] = (unsigned char) c;
 1571:         }
 1572: 
 1573:         if (n > 0) {
 1574:             ogroup[0] = dtable[igroup[0] >> 2];
 1575:             ogroup[1] = dtable[((igroup[0] & 3) << 4) | (igroup[1] >> 4)];
 1576:             ogroup[2] =
 1577:                 dtable[((igroup[1] & 0xF) << 2) | (igroup[2] >> 6)];
 1578:             ogroup[3] = dtable[igroup[2] & 0x3F];
 1579: 
 1580:             if (n < 3) {
 1581:                 ogroup[3] = '=';
 1582:                 if (n < 2) {
 1583:                     ogroup[2] = '=';
 1584:                 }
 1585:             }
 1586: 
 1587:             if (linelen >= B64LINELEN) {
 1588:                 count = xmlOutputBufferWrite(out, 2, B64CRLF);
 1589:                 if (count == -1)
 1590:                     return -1;
 1591:                 sum += count;
 1592:                 linelen = 0;
 1593:             }
 1594:             count = xmlOutputBufferWrite(out, 4, (const char *) ogroup);
 1595:             if (count == -1)
 1596:                 return -1;
 1597:             sum += count;
 1598: 
 1599:             linelen += 4;
 1600:         }
 1601: 
 1602:         if (i >= len)
 1603:             break;
 1604:     }
 1605: 
 1606:     return sum;
 1607: }
 1608: 
 1609: /**
 1610:  * xmlTextWriterWriteBase64:
 1611:  * @writer: the xmlTextWriterPtr
 1612:  * @data:   binary data
 1613:  * @start:  the position within the data of the first byte to encode
 1614:  * @len:  the number of bytes to encode
 1615:  *
 1616:  * Write an base64 encoded xml text.
 1617:  *
 1618:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1619:  */
 1620: int
 1621: xmlTextWriterWriteBase64(xmlTextWriterPtr writer, const char *data,
 1622:                          int start, int len)
 1623: {
 1624:     int count;
 1625:     int sum;
 1626:     xmlLinkPtr lk;
 1627:     xmlTextWriterStackEntry *p;
 1628: 
 1629:     if ((writer == NULL) || (data == NULL) || (start < 0) || (len < 0))
 1630:         return -1;
 1631: 
 1632:     sum = 0;
 1633:     lk = xmlListFront(writer->nodes);
 1634:     if (lk != 0) {
 1635:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1636:         if (p != 0) {
 1637:             count = xmlTextWriterHandleStateDependencies(writer, p);
 1638:             if (count < 0)
 1639:                 return -1;
 1640:             sum += count;
 1641:         }
 1642:     }
 1643: 
 1644:     if (writer->indent)
 1645:         writer->doindent = 0;
 1646: 
 1647:     count =
 1648:         xmlOutputBufferWriteBase64(writer->out, len,
 1649:                                    (unsigned char *) data + start);
 1650:     if (count < 0)
 1651:         return -1;
 1652:     sum += count;
 1653: 
 1654:     return sum;
 1655: }
 1656: 
 1657: /**
 1658:  * xmlOutputBufferWriteBinHex:
 1659:  * @out: the xmlOutputBufferPtr
 1660:  * @data:   binary data
 1661:  * @len:  the number of bytes to encode
 1662:  *
 1663:  * Write hqx encoded data to an xmlOutputBuffer.
 1664:  * ::todo
 1665:  *
 1666:  * Returns the bytes written (may be 0 because of buffering) 
 1667:  * or -1 in case of error
 1668:  */
 1669: static int
 1670: xmlOutputBufferWriteBinHex(xmlOutputBufferPtr out,
 1671:                            int len, const unsigned char *data)
 1672: {
 1673:     int count;
 1674:     int sum;
 1675:     static char hex[16] = 
 1676:     	{'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
 1677:     int i;
 1678: 
 1679:     if ((out == NULL) || (data == NULL) || (len < 0)) {
 1680:         return -1;
 1681:     }
 1682: 
 1683:     sum = 0;
 1684:     for (i = 0; i < len; i++) {
 1685:         count =
 1686:             xmlOutputBufferWrite(out, 1,
 1687:                                  (const char *) &hex[data[i] >> 4]);
 1688:         if (count == -1)
 1689:             return -1;
 1690:         sum += count;
 1691:         count =
 1692:             xmlOutputBufferWrite(out, 1,
 1693:                                  (const char *) &hex[data[i] & 0xF]);
 1694:         if (count == -1)
 1695:             return -1;
 1696:         sum += count;
 1697:     }
 1698: 
 1699:     return sum;
 1700: }
 1701: 
 1702: /**
 1703:  * xmlTextWriterWriteBinHex:
 1704:  * @writer: the xmlTextWriterPtr
 1705:  * @data:   binary data
 1706:  * @start:  the position within the data of the first byte to encode
 1707:  * @len:  the number of bytes to encode
 1708:  *
 1709:  * Write a BinHex encoded xml text.
 1710:  *
 1711:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1712:  */
 1713: int
 1714: xmlTextWriterWriteBinHex(xmlTextWriterPtr writer, const char *data,
 1715:                          int start, int len)
 1716: {
 1717:     int count;
 1718:     int sum;
 1719:     xmlLinkPtr lk;
 1720:     xmlTextWriterStackEntry *p;
 1721: 
 1722:     if ((writer == NULL) || (data == NULL) || (start < 0) || (len < 0))
 1723:         return -1;
 1724: 
 1725:     sum = 0;
 1726:     lk = xmlListFront(writer->nodes);
 1727:     if (lk != 0) {
 1728:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1729:         if (p != 0) {
 1730:             count = xmlTextWriterHandleStateDependencies(writer, p);
 1731:             if (count < 0)
 1732:                 return -1;
 1733:             sum += count;
 1734:         }
 1735:     }
 1736: 
 1737:     if (writer->indent)
 1738:         writer->doindent = 0;
 1739: 
 1740:     count =
 1741:         xmlOutputBufferWriteBinHex(writer->out, len,
 1742:                                    (unsigned char *) data + start);
 1743:     if (count < 0)
 1744:         return -1;
 1745:     sum += count;
 1746: 
 1747:     return sum;
 1748: }
 1749: 
 1750: /**
 1751:  * xmlTextWriterStartAttribute:
 1752:  * @writer:  the xmlTextWriterPtr
 1753:  * @name:  element name
 1754:  *
 1755:  * Start an xml attribute.
 1756:  *
 1757:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1758:  */
 1759: int
 1760: xmlTextWriterStartAttribute(xmlTextWriterPtr writer, const xmlChar * name)
 1761: {
 1762:     int count;
 1763:     int sum;
 1764:     xmlLinkPtr lk;
 1765:     xmlTextWriterStackEntry *p;
 1766: 
 1767:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
 1768:         return -1;
 1769: 
 1770:     sum = 0;
 1771:     lk = xmlListFront(writer->nodes);
 1772:     if (lk == 0)
 1773:         return -1;
 1774: 
 1775:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1776:     if (p == 0)
 1777:         return -1;
 1778: 
 1779:     switch (p->state) {
 1780:         case XML_TEXTWRITER_ATTRIBUTE:
 1781:             count = xmlTextWriterEndAttribute(writer);
 1782:             if (count < 0)
 1783:                 return -1;
 1784:             sum += count;
 1785:             /* fallthrough */
 1786:         case XML_TEXTWRITER_NAME:
 1787:             count = xmlOutputBufferWriteString(writer->out, " ");
 1788:             if (count < 0)
 1789:                 return -1;
 1790:             sum += count;
 1791:             count =
 1792:                 xmlOutputBufferWriteString(writer->out,
 1793:                                            (const char *) name);
 1794:             if (count < 0)
 1795:                 return -1;
 1796:             sum += count;
 1797:             count = xmlOutputBufferWriteString(writer->out, "=");
 1798:             if (count < 0)
 1799:                 return -1;
 1800:             sum += count;
 1801:             count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 1802:             if (count < 0)
 1803:                 return -1;
 1804:             sum += count;
 1805:             p->state = XML_TEXTWRITER_ATTRIBUTE;
 1806:             break;
 1807:         default:
 1808:             return -1;
 1809:     }
 1810: 
 1811:     return sum;
 1812: }
 1813: 
 1814: /**
 1815:  * xmlTextWriterStartAttributeNS:
 1816:  * @writer:  the xmlTextWriterPtr
 1817:  * @prefix:  namespace prefix or NULL
 1818:  * @name:  element local name
 1819:  * @namespaceURI:  namespace URI or NULL
 1820:  *
 1821:  * Start an xml attribute with namespace support.
 1822:  *
 1823:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1824:  */
 1825: int
 1826: xmlTextWriterStartAttributeNS(xmlTextWriterPtr writer,
 1827:                               const xmlChar * prefix, const xmlChar * name,
 1828:                               const xmlChar * namespaceURI)
 1829: {
 1830:     int count;
 1831:     int sum;
 1832:     xmlChar *buf;
 1833:     xmlTextWriterNsStackEntry *p;
 1834: 
 1835:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
 1836:         return -1;
 1837: 
 1838:     /* Handle namespace first in case of error */
 1839:     if (namespaceURI != 0) {
 1840:         xmlTextWriterNsStackEntry nsentry, *curns;
 1841: 
 1842:         buf = xmlStrdup(BAD_CAST "xmlns");
 1843:         if (prefix != 0) {
 1844:             buf = xmlStrcat(buf, BAD_CAST ":");
 1845:             buf = xmlStrcat(buf, prefix);
 1846:         }
 1847: 
 1848:         nsentry.prefix = buf;
 1849:         nsentry.uri = (xmlChar *)namespaceURI;
 1850:         nsentry.elem = xmlListFront(writer->nodes);
 1851: 
 1852:         curns = (xmlTextWriterNsStackEntry *)xmlListSearch(writer->nsstack, 
 1853:                                                            (void *)&nsentry);
 1854:         if ((curns != NULL)) {
 1855:             xmlFree(buf);
 1856:             if (xmlStrcmp(curns->uri, namespaceURI) == 0) {
 1857:                 /* Namespace already defined on element skip */
 1858:                 buf = NULL;
 1859:             } else {
 1860:                 /* Prefix mismatch so error out */
 1861:                 return -1;
 1862:             }
 1863:         }
 1864: 
 1865:         /* Do not add namespace decl to list - it is already there */
 1866:         if (buf != NULL) {
 1867:             p = (xmlTextWriterNsStackEntry *)
 1868:                 xmlMalloc(sizeof(xmlTextWriterNsStackEntry));
 1869:             if (p == 0) {
 1870:                 xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 1871: 								        "xmlTextWriterStartAttributeNS : out of memory!\n");
 1872:                 return -1;
 1873:             }
 1874: 
 1875:             p->prefix = buf;
 1876:             p->uri = xmlStrdup(namespaceURI);
 1877:             if (p->uri == 0) {
 1878:                 xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 1879:                         "xmlTextWriterStartAttributeNS : out of memory!\n");
 1880:                 xmlFree(p);
 1881:                 return -1;
 1882:             }
 1883:             p->elem = xmlListFront(writer->nodes);
 1884: 
 1885:             xmlListPushFront(writer->nsstack, p);
 1886:         }
 1887:     }
 1888: 
 1889:     buf = NULL;
 1890:     if (prefix != 0) {
 1891:         buf = xmlStrdup(prefix);
 1892:         buf = xmlStrcat(buf, BAD_CAST ":");
 1893:     }
 1894:     buf = xmlStrcat(buf, name);
 1895: 
 1896:     sum = 0;
 1897:     count = xmlTextWriterStartAttribute(writer, buf);
 1898:     xmlFree(buf);
 1899:     if (count < 0)
 1900:         return -1;
 1901:     sum += count;
 1902: 
 1903:     return sum;
 1904: }
 1905: 
 1906: /**
 1907:  * xmlTextWriterEndAttribute:
 1908:  * @writer:  the xmlTextWriterPtr
 1909:  *
 1910:  * End the current xml element.
 1911:  *
 1912:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1913:  */
 1914: int
 1915: xmlTextWriterEndAttribute(xmlTextWriterPtr writer)
 1916: {
 1917:     int count;
 1918:     int sum;
 1919:     xmlLinkPtr lk;
 1920:     xmlTextWriterStackEntry *p;
 1921: 
 1922:     if (writer == NULL)
 1923:         return -1;
 1924: 
 1925:     lk = xmlListFront(writer->nodes);
 1926:     if (lk == 0) {
 1927:         return -1;
 1928:     }
 1929: 
 1930:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 1931:     if (p == 0) {
 1932:         return -1;
 1933:     }
 1934: 
 1935:     sum = 0;
 1936:     switch (p->state) {
 1937:         case XML_TEXTWRITER_ATTRIBUTE:
 1938:             p->state = XML_TEXTWRITER_NAME;
 1939: 
 1940:             count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 1941:             if (count < 0) {
 1942:                 return -1;
 1943:             }
 1944:             sum += count;
 1945:             break;
 1946:         default:
 1947:             return -1;
 1948:     }
 1949: 
 1950:     return sum;
 1951: }
 1952: 
 1953: /**
 1954:  * xmlTextWriterWriteFormatAttribute:
 1955:  * @writer:  the xmlTextWriterPtr
 1956:  * @name:  attribute name
 1957:  * @format:  format string (see printf)
 1958:  * @...:  extra parameters for the format
 1959:  *
 1960:  * Write a formatted xml attribute.
 1961:  *
 1962:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1963:  */
 1964: int XMLCDECL
 1965: xmlTextWriterWriteFormatAttribute(xmlTextWriterPtr writer,
 1966:                                   const xmlChar * name, const char *format,
 1967:                                   ...)
 1968: {
 1969:     int rc;
 1970:     va_list ap;
 1971: 
 1972:     va_start(ap, format);
 1973: 
 1974:     rc = xmlTextWriterWriteVFormatAttribute(writer, name, format, ap);
 1975: 
 1976:     va_end(ap);
 1977:     return rc;
 1978: }
 1979: 
 1980: /**
 1981:  * xmlTextWriterWriteVFormatAttribute:
 1982:  * @writer:  the xmlTextWriterPtr
 1983:  * @name:  attribute name
 1984:  * @format:  format string (see printf)
 1985:  * @argptr:  pointer to the first member of the variable argument list.
 1986:  *
 1987:  * Write a formatted xml attribute.
 1988:  *
 1989:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 1990:  */
 1991: int
 1992: xmlTextWriterWriteVFormatAttribute(xmlTextWriterPtr writer,
 1993:                                    const xmlChar * name,
 1994:                                    const char *format, va_list argptr)
 1995: {
 1996:     int rc;
 1997:     xmlChar *buf;
 1998: 
 1999:     if (writer == NULL)
 2000:         return -1;
 2001: 
 2002:     buf = xmlTextWriterVSprintf(format, argptr);
 2003:     if (buf == NULL)
 2004:         return -1;
 2005: 
 2006:     rc = xmlTextWriterWriteAttribute(writer, name, buf);
 2007: 
 2008:     xmlFree(buf);
 2009:     return rc;
 2010: }
 2011: 
 2012: /**
 2013:  * xmlTextWriterWriteAttribute:
 2014:  * @writer:  the xmlTextWriterPtr
 2015:  * @name:  attribute name
 2016:  * @content:  attribute content
 2017:  *
 2018:  * Write an xml attribute.
 2019:  *
 2020:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2021:  */
 2022: int
 2023: xmlTextWriterWriteAttribute(xmlTextWriterPtr writer, const xmlChar * name,
 2024:                             const xmlChar * content)
 2025: {
 2026:     int count;
 2027:     int sum;
 2028: 
 2029:     sum = 0;
 2030:     count = xmlTextWriterStartAttribute(writer, name);
 2031:     if (count < 0)
 2032:         return -1;
 2033:     sum += count;
 2034:     count = xmlTextWriterWriteString(writer, content);
 2035:     if (count < 0)
 2036:         return -1;
 2037:     sum += count;
 2038:     count = xmlTextWriterEndAttribute(writer);
 2039:     if (count < 0)
 2040:         return -1;
 2041:     sum += count;
 2042: 
 2043:     return sum;
 2044: }
 2045: 
 2046: /**
 2047:  * xmlTextWriterWriteFormatAttributeNS:
 2048:  * @writer:  the xmlTextWriterPtr
 2049:  * @prefix:  namespace prefix
 2050:  * @name:  attribute local name
 2051:  * @namespaceURI:  namespace URI
 2052:  * @format:  format string (see printf)
 2053:  * @...:  extra parameters for the format
 2054:  *
 2055:  * Write a formatted xml attribute.with namespace support
 2056:  *
 2057:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2058:  */
 2059: int XMLCDECL
 2060: xmlTextWriterWriteFormatAttributeNS(xmlTextWriterPtr writer,
 2061:                                     const xmlChar * prefix,
 2062:                                     const xmlChar * name,
 2063:                                     const xmlChar * namespaceURI,
 2064:                                     const char *format, ...)
 2065: {
 2066:     int rc;
 2067:     va_list ap;
 2068: 
 2069:     va_start(ap, format);
 2070: 
 2071:     rc = xmlTextWriterWriteVFormatAttributeNS(writer, prefix, name,
 2072:                                               namespaceURI, format, ap);
 2073: 
 2074:     va_end(ap);
 2075:     return rc;
 2076: }
 2077: 
 2078: /**
 2079:  * xmlTextWriterWriteVFormatAttributeNS:
 2080:  * @writer:  the xmlTextWriterPtr
 2081:  * @prefix:  namespace prefix
 2082:  * @name:  attribute local name
 2083:  * @namespaceURI:  namespace URI
 2084:  * @format:  format string (see printf)
 2085:  * @argptr:  pointer to the first member of the variable argument list.
 2086:  *
 2087:  * Write a formatted xml attribute.with namespace support
 2088:  *
 2089:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2090:  */
 2091: int
 2092: xmlTextWriterWriteVFormatAttributeNS(xmlTextWriterPtr writer,
 2093:                                      const xmlChar * prefix,
 2094:                                      const xmlChar * name,
 2095:                                      const xmlChar * namespaceURI,
 2096:                                      const char *format, va_list argptr)
 2097: {
 2098:     int rc;
 2099:     xmlChar *buf;
 2100: 
 2101:     if (writer == NULL)
 2102:         return -1;
 2103: 
 2104:     buf = xmlTextWriterVSprintf(format, argptr);
 2105:     if (buf == NULL)
 2106:         return -1;
 2107: 
 2108:     rc = xmlTextWriterWriteAttributeNS(writer, prefix, name, namespaceURI,
 2109:                                        buf);
 2110: 
 2111:     xmlFree(buf);
 2112:     return rc;
 2113: }
 2114: 
 2115: /**
 2116:  * xmlTextWriterWriteAttributeNS:
 2117:  * @writer:  the xmlTextWriterPtr
 2118:  * @prefix:  namespace prefix
 2119:  * @name:  attribute local name
 2120:  * @namespaceURI:  namespace URI
 2121:  * @content:  attribute content
 2122:  *
 2123:  * Write an xml attribute.
 2124:  *
 2125:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2126:  */
 2127: int
 2128: xmlTextWriterWriteAttributeNS(xmlTextWriterPtr writer,
 2129:                               const xmlChar * prefix, const xmlChar * name,
 2130:                               const xmlChar * namespaceURI,
 2131:                               const xmlChar * content)
 2132: {
 2133:     int count;
 2134:     int sum;
 2135: 
 2136:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
 2137:         return -1;
 2138: 
 2139:     sum = 0;
 2140:     count = xmlTextWriterStartAttributeNS(writer, prefix, name, namespaceURI);
 2141:     if (count < 0)
 2142:         return -1;
 2143:     sum += count;
 2144:     count = xmlTextWriterWriteString(writer, content);
 2145:     if (count < 0)
 2146:         return -1;
 2147:     sum += count;
 2148:     count = xmlTextWriterEndAttribute(writer);
 2149:     if (count < 0)
 2150:         return -1;
 2151:     sum += count;
 2152: 
 2153:     return sum;
 2154: }
 2155: 
 2156: /**
 2157:  * xmlTextWriterWriteFormatElement:
 2158:  * @writer:  the xmlTextWriterPtr
 2159:  * @name:  element name
 2160:  * @format:  format string (see printf)
 2161:  * @...:  extra parameters for the format
 2162:  *
 2163:  * Write a formatted xml element.
 2164:  *
 2165:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2166:  */
 2167: int XMLCDECL
 2168: xmlTextWriterWriteFormatElement(xmlTextWriterPtr writer,
 2169:                                 const xmlChar * name, const char *format,
 2170:                                 ...)
 2171: {
 2172:     int rc;
 2173:     va_list ap;
 2174: 
 2175:     va_start(ap, format);
 2176: 
 2177:     rc = xmlTextWriterWriteVFormatElement(writer, name, format, ap);
 2178: 
 2179:     va_end(ap);
 2180:     return rc;
 2181: }
 2182: 
 2183: /**
 2184:  * xmlTextWriterWriteVFormatElement:
 2185:  * @writer:  the xmlTextWriterPtr
 2186:  * @name:  element name
 2187:  * @format:  format string (see printf)
 2188:  * @argptr:  pointer to the first member of the variable argument list.
 2189:  *
 2190:  * Write a formatted xml element.
 2191:  *
 2192:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2193:  */
 2194: int
 2195: xmlTextWriterWriteVFormatElement(xmlTextWriterPtr writer,
 2196:                                  const xmlChar * name, const char *format,
 2197:                                  va_list argptr)
 2198: {
 2199:     int rc;
 2200:     xmlChar *buf;
 2201: 
 2202:     if (writer == NULL)
 2203:         return -1;
 2204: 
 2205:     buf = xmlTextWriterVSprintf(format, argptr);
 2206:     if (buf == NULL)
 2207:         return -1;
 2208: 
 2209:     rc = xmlTextWriterWriteElement(writer, name, buf);
 2210: 
 2211:     xmlFree(buf);
 2212:     return rc;
 2213: }
 2214: 
 2215: /**
 2216:  * xmlTextWriterWriteElement:
 2217:  * @writer:  the xmlTextWriterPtr
 2218:  * @name:  element name
 2219:  * @content:  element content
 2220:  *
 2221:  * Write an xml element.
 2222:  *
 2223:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2224:  */
 2225: int
 2226: xmlTextWriterWriteElement(xmlTextWriterPtr writer, const xmlChar * name,
 2227:                           const xmlChar * content)
 2228: {
 2229:     int count;
 2230:     int sum;
 2231: 
 2232:     sum = 0;
 2233:     count = xmlTextWriterStartElement(writer, name);
 2234:     if (count == -1)
 2235:         return -1;
 2236:     sum += count;
 2237:     count = xmlTextWriterWriteString(writer, content);
 2238:     if (count == -1)
 2239:         return -1;
 2240:     sum += count;
 2241:     count = xmlTextWriterEndElement(writer);
 2242:     if (count == -1)
 2243:         return -1;
 2244:     sum += count;
 2245: 
 2246:     return sum;
 2247: }
 2248: 
 2249: /**
 2250:  * xmlTextWriterWriteFormatElementNS:
 2251:  * @writer:  the xmlTextWriterPtr
 2252:  * @prefix:  namespace prefix
 2253:  * @name:  element local name
 2254:  * @namespaceURI:  namespace URI
 2255:  * @format:  format string (see printf)
 2256:  * @...:  extra parameters for the format
 2257:  *
 2258:  * Write a formatted xml element with namespace support.
 2259:  *
 2260:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2261:  */
 2262: int XMLCDECL
 2263: xmlTextWriterWriteFormatElementNS(xmlTextWriterPtr writer,
 2264:                                   const xmlChar * prefix,
 2265:                                   const xmlChar * name,
 2266:                                   const xmlChar * namespaceURI,
 2267:                                   const char *format, ...)
 2268: {
 2269:     int rc;
 2270:     va_list ap;
 2271: 
 2272:     va_start(ap, format);
 2273: 
 2274:     rc = xmlTextWriterWriteVFormatElementNS(writer, prefix, name,
 2275:                                             namespaceURI, format, ap);
 2276: 
 2277:     va_end(ap);
 2278:     return rc;
 2279: }
 2280: 
 2281: /**
 2282:  * xmlTextWriterWriteVFormatElementNS:
 2283:  * @writer:  the xmlTextWriterPtr
 2284:  * @prefix:  namespace prefix
 2285:  * @name:  element local name
 2286:  * @namespaceURI:  namespace URI
 2287:  * @format:  format string (see printf)
 2288:  * @argptr:  pointer to the first member of the variable argument list.
 2289:  *
 2290:  * Write a formatted xml element with namespace support.
 2291:  *
 2292:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2293:  */
 2294: int
 2295: xmlTextWriterWriteVFormatElementNS(xmlTextWriterPtr writer,
 2296:                                    const xmlChar * prefix,
 2297:                                    const xmlChar * name,
 2298:                                    const xmlChar * namespaceURI,
 2299:                                    const char *format, va_list argptr)
 2300: {
 2301:     int rc;
 2302:     xmlChar *buf;
 2303: 
 2304:     if (writer == NULL)
 2305:         return -1;
 2306: 
 2307:     buf = xmlTextWriterVSprintf(format, argptr);
 2308:     if (buf == NULL)
 2309:         return -1;
 2310: 
 2311:     rc = xmlTextWriterWriteElementNS(writer, prefix, name, namespaceURI,
 2312:                                      buf);
 2313: 
 2314:     xmlFree(buf);
 2315:     return rc;
 2316: }
 2317: 
 2318: /**
 2319:  * xmlTextWriterWriteElementNS:
 2320:  * @writer:  the xmlTextWriterPtr
 2321:  * @prefix:  namespace prefix
 2322:  * @name:  element local name
 2323:  * @namespaceURI:  namespace URI
 2324:  * @content:  element content
 2325:  *
 2326:  * Write an xml element with namespace support.
 2327:  *
 2328:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2329:  */
 2330: int
 2331: xmlTextWriterWriteElementNS(xmlTextWriterPtr writer,
 2332:                             const xmlChar * prefix, const xmlChar * name,
 2333:                             const xmlChar * namespaceURI,
 2334:                             const xmlChar * content)
 2335: {
 2336:     int count;
 2337:     int sum;
 2338: 
 2339:     if ((writer == NULL) || (name == NULL) || (*name == '\0'))
 2340:         return -1;
 2341: 
 2342:     sum = 0;
 2343:     count =
 2344:         xmlTextWriterStartElementNS(writer, prefix, name, namespaceURI);
 2345:     if (count < 0)
 2346:         return -1;
 2347:     sum += count;
 2348:     count = xmlTextWriterWriteString(writer, content);
 2349:     if (count == -1)
 2350:         return -1;
 2351:     sum += count;
 2352:     count = xmlTextWriterEndElement(writer);
 2353:     if (count == -1)
 2354:         return -1;
 2355:     sum += count;
 2356: 
 2357:     return sum;
 2358: }
 2359: 
 2360: /**
 2361:  * xmlTextWriterStartPI:
 2362:  * @writer:  the xmlTextWriterPtr
 2363:  * @target:  PI target
 2364:  *
 2365:  * Start an xml PI.
 2366:  *
 2367:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2368:  */
 2369: int
 2370: xmlTextWriterStartPI(xmlTextWriterPtr writer, const xmlChar * target)
 2371: {
 2372:     int count;
 2373:     int sum;
 2374:     xmlLinkPtr lk;
 2375:     xmlTextWriterStackEntry *p;
 2376: 
 2377:     if ((writer == NULL) || (target == NULL) || (*target == '\0'))
 2378:         return -1;
 2379: 
 2380:     if (xmlStrcasecmp(target, (const xmlChar *) "xml") == 0) {
 2381:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 2382:                         "xmlTextWriterStartPI : target name [Xx][Mm][Ll] is reserved for xml standardization!\n");
 2383:         return -1;
 2384:     }
 2385: 
 2386:     sum = 0;
 2387:     lk = xmlListFront(writer->nodes);
 2388:     if (lk != 0) {
 2389:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 2390:         if (p != 0) {
 2391:             switch (p->state) {
 2392:                 case XML_TEXTWRITER_ATTRIBUTE:
 2393:                     count = xmlTextWriterEndAttribute(writer);
 2394:                     if (count < 0)
 2395:                         return -1;
 2396:                     sum += count;
 2397:                     /* fallthrough */
 2398:                 case XML_TEXTWRITER_NAME:
 2399:                     /* Output namespace declarations */
 2400:                     count = xmlTextWriterOutputNSDecl(writer);
 2401:                     if (count < 0)
 2402:                         return -1;
 2403:                     sum += count;
 2404:                     count = xmlOutputBufferWriteString(writer->out, ">");
 2405:                     if (count < 0)
 2406:                         return -1;
 2407:                     sum += count;
 2408:                     p->state = XML_TEXTWRITER_TEXT;
 2409:                     break;
 2410:                 case XML_TEXTWRITER_NONE:
 2411:                 case XML_TEXTWRITER_TEXT:
 2412:                 case XML_TEXTWRITER_DTD:
 2413:                     break;
 2414:                 case XML_TEXTWRITER_PI:
 2415:                 case XML_TEXTWRITER_PI_TEXT:
 2416:                     xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 2417:                                     "xmlTextWriterStartPI : nested PI!\n");
 2418:                     return -1;
 2419:                 default:
 2420:                     return -1;
 2421:             }
 2422:         }
 2423:     }
 2424: 
 2425:     p = (xmlTextWriterStackEntry *)
 2426:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 2427:     if (p == 0) {
 2428:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 2429:                         "xmlTextWriterStartPI : out of memory!\n");
 2430:         return -1;
 2431:     }
 2432: 
 2433:     p->name = xmlStrdup(target);
 2434:     if (p->name == 0) {
 2435:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 2436:                         "xmlTextWriterStartPI : out of memory!\n");
 2437:         xmlFree(p);
 2438:         return -1;
 2439:     }
 2440:     p->state = XML_TEXTWRITER_PI;
 2441: 
 2442:     xmlListPushFront(writer->nodes, p);
 2443: 
 2444:     count = xmlOutputBufferWriteString(writer->out, "<?");
 2445:     if (count < 0)
 2446:         return -1;
 2447:     sum += count;
 2448:     count =
 2449:         xmlOutputBufferWriteString(writer->out, (const char *) p->name);
 2450:     if (count < 0)
 2451:         return -1;
 2452:     sum += count;
 2453: 
 2454:     return sum;
 2455: }
 2456: 
 2457: /**
 2458:  * xmlTextWriterEndPI:
 2459:  * @writer:  the xmlTextWriterPtr
 2460:  *
 2461:  * End the current xml PI.
 2462:  *
 2463:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2464:  */
 2465: int
 2466: xmlTextWriterEndPI(xmlTextWriterPtr writer)
 2467: {
 2468:     int count;
 2469:     int sum;
 2470:     xmlLinkPtr lk;
 2471:     xmlTextWriterStackEntry *p;
 2472: 
 2473:     if (writer == NULL)
 2474:         return -1;
 2475: 
 2476:     lk = xmlListFront(writer->nodes);
 2477:     if (lk == 0)
 2478:         return 0;
 2479: 
 2480:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 2481:     if (p == 0)
 2482:         return 0;
 2483: 
 2484:     sum = 0;
 2485:     switch (p->state) {
 2486:         case XML_TEXTWRITER_PI:
 2487:         case XML_TEXTWRITER_PI_TEXT:
 2488:             count = xmlOutputBufferWriteString(writer->out, "?>");
 2489:             if (count < 0)
 2490:                 return -1;
 2491:             sum += count;
 2492:             break;
 2493:         default:
 2494:             return -1;
 2495:     }
 2496: 
 2497:     if (writer->indent) {
 2498:         count = xmlOutputBufferWriteString(writer->out, "\n");
 2499:       	if (count < 0)
 2500:        	return -1;
 2501:         sum += count;
 2502:     }
 2503: 
 2504:     xmlListPopFront(writer->nodes);
 2505:     return sum;
 2506: }
 2507: 
 2508: /**
 2509:  * xmlTextWriterWriteFormatPI:
 2510:  * @writer:  the xmlTextWriterPtr
 2511:  * @target:  PI target
 2512:  * @format:  format string (see printf)
 2513:  * @...:  extra parameters for the format
 2514:  *
 2515:  * Write a formatted PI.
 2516:  *
 2517:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2518:  */
 2519: int XMLCDECL
 2520: xmlTextWriterWriteFormatPI(xmlTextWriterPtr writer, const xmlChar * target,
 2521:                            const char *format, ...)
 2522: {
 2523:     int rc;
 2524:     va_list ap;
 2525: 
 2526:     va_start(ap, format);
 2527: 
 2528:     rc = xmlTextWriterWriteVFormatPI(writer, target, format, ap);
 2529: 
 2530:     va_end(ap);
 2531:     return rc;
 2532: }
 2533: 
 2534: /**
 2535:  * xmlTextWriterWriteVFormatPI:
 2536:  * @writer:  the xmlTextWriterPtr
 2537:  * @target:  PI target
 2538:  * @format:  format string (see printf)
 2539:  * @argptr:  pointer to the first member of the variable argument list.
 2540:  *
 2541:  * Write a formatted xml PI.
 2542:  *
 2543:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2544:  */
 2545: int
 2546: xmlTextWriterWriteVFormatPI(xmlTextWriterPtr writer,
 2547:                             const xmlChar * target, const char *format,
 2548:                             va_list argptr)
 2549: {
 2550:     int rc;
 2551:     xmlChar *buf;
 2552: 
 2553:     if (writer == NULL)
 2554:         return -1;
 2555: 
 2556:     buf = xmlTextWriterVSprintf(format, argptr);
 2557:     if (buf == NULL)
 2558:         return -1;
 2559: 
 2560:     rc = xmlTextWriterWritePI(writer, target, buf);
 2561: 
 2562:     xmlFree(buf);
 2563:     return rc;
 2564: }
 2565: 
 2566: /**
 2567:  * xmlTextWriterWritePI:
 2568:  * @writer:  the xmlTextWriterPtr
 2569:  * @target:  PI target
 2570:  * @content:  PI content
 2571:  *
 2572:  * Write an xml PI.
 2573:  *
 2574:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2575:  */
 2576: int
 2577: xmlTextWriterWritePI(xmlTextWriterPtr writer, const xmlChar * target,
 2578:                      const xmlChar * content)
 2579: {
 2580:     int count;
 2581:     int sum;
 2582: 
 2583:     sum = 0;
 2584:     count = xmlTextWriterStartPI(writer, target);
 2585:     if (count == -1)
 2586:         return -1;
 2587:     sum += count;
 2588:     if (content != 0) {
 2589:         count = xmlTextWriterWriteString(writer, content);
 2590:         if (count == -1)
 2591:             return -1;
 2592:         sum += count;
 2593:     }
 2594:     count = xmlTextWriterEndPI(writer);
 2595:     if (count == -1)
 2596:         return -1;
 2597:     sum += count;
 2598: 
 2599:     return sum;
 2600: }
 2601: 
 2602: /**
 2603:  * xmlTextWriterStartCDATA:
 2604:  * @writer:  the xmlTextWriterPtr
 2605:  *
 2606:  * Start an xml CDATA section.
 2607:  *
 2608:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2609:  */
 2610: int
 2611: xmlTextWriterStartCDATA(xmlTextWriterPtr writer)
 2612: {
 2613:     int count;
 2614:     int sum;
 2615:     xmlLinkPtr lk;
 2616:     xmlTextWriterStackEntry *p;
 2617: 
 2618:     if (writer == NULL)
 2619:         return -1;
 2620: 
 2621:     sum = 0;
 2622:     lk = xmlListFront(writer->nodes);
 2623:     if (lk != 0) {
 2624:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 2625:         if (p != 0) {
 2626:             switch (p->state) {
 2627:                 case XML_TEXTWRITER_NONE:
 2628: 		case XML_TEXTWRITER_TEXT:
 2629:                 case XML_TEXTWRITER_PI:
 2630:                 case XML_TEXTWRITER_PI_TEXT:
 2631:                     break;
 2632:                 case XML_TEXTWRITER_ATTRIBUTE:
 2633:                     count = xmlTextWriterEndAttribute(writer);
 2634:                     if (count < 0)
 2635:                         return -1;
 2636:                     sum += count;
 2637:                     /* fallthrough */
 2638:                 case XML_TEXTWRITER_NAME:
 2639:                     /* Output namespace declarations */
 2640:                     count = xmlTextWriterOutputNSDecl(writer);
 2641:                     if (count < 0)
 2642:                         return -1;
 2643:                     sum += count;
 2644:                     count = xmlOutputBufferWriteString(writer->out, ">");
 2645:                     if (count < 0)
 2646:                         return -1;
 2647:                     sum += count;
 2648:                     p->state = XML_TEXTWRITER_TEXT;
 2649:                     break;
 2650:                 case XML_TEXTWRITER_CDATA:
 2651:                     xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 2652:                                     "xmlTextWriterStartCDATA : CDATA not allowed in this context!\n");
 2653:                     return -1;
 2654:                 default:
 2655:                     return -1;
 2656:             }
 2657:         }
 2658:     }
 2659: 
 2660:     p = (xmlTextWriterStackEntry *)
 2661:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 2662:     if (p == 0) {
 2663:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 2664:                         "xmlTextWriterStartCDATA : out of memory!\n");
 2665:         return -1;
 2666:     }
 2667: 
 2668:     p->name = NULL;
 2669:     p->state = XML_TEXTWRITER_CDATA;
 2670: 
 2671:     xmlListPushFront(writer->nodes, p);
 2672: 
 2673:     count = xmlOutputBufferWriteString(writer->out, "<![CDATA[");
 2674:     if (count < 0)
 2675:         return -1;
 2676:     sum += count;
 2677: 
 2678:     return sum;
 2679: }
 2680: 
 2681: /**
 2682:  * xmlTextWriterEndCDATA:
 2683:  * @writer:  the xmlTextWriterPtr
 2684:  *
 2685:  * End an xml CDATA section.
 2686:  *
 2687:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2688:  */
 2689: int
 2690: xmlTextWriterEndCDATA(xmlTextWriterPtr writer)
 2691: {
 2692:     int count;
 2693:     int sum;
 2694:     xmlLinkPtr lk;
 2695:     xmlTextWriterStackEntry *p;
 2696: 
 2697:     if (writer == NULL)
 2698:         return -1;
 2699: 
 2700:     lk = xmlListFront(writer->nodes);
 2701:     if (lk == 0)
 2702:         return -1;
 2703: 
 2704:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 2705:     if (p == 0)
 2706:         return -1;
 2707: 
 2708:     sum = 0;
 2709:     switch (p->state) {
 2710:         case XML_TEXTWRITER_CDATA:
 2711:             count = xmlOutputBufferWriteString(writer->out, "]]>");
 2712:             if (count < 0)
 2713:                 return -1;
 2714:             sum += count;
 2715:             break;
 2716:         default:
 2717:             return -1;
 2718:     }
 2719: 
 2720:     xmlListPopFront(writer->nodes);
 2721:     return sum;
 2722: }
 2723: 
 2724: /**
 2725:  * xmlTextWriterWriteFormatCDATA:
 2726:  * @writer:  the xmlTextWriterPtr
 2727:  * @format:  format string (see printf)
 2728:  * @...:  extra parameters for the format
 2729:  *
 2730:  * Write a formatted xml CDATA.
 2731:  *
 2732:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2733:  */
 2734: int XMLCDECL
 2735: xmlTextWriterWriteFormatCDATA(xmlTextWriterPtr writer, const char *format,
 2736:                               ...)
 2737: {
 2738:     int rc;
 2739:     va_list ap;
 2740: 
 2741:     va_start(ap, format);
 2742: 
 2743:     rc = xmlTextWriterWriteVFormatCDATA(writer, format, ap);
 2744: 
 2745:     va_end(ap);
 2746:     return rc;
 2747: }
 2748: 
 2749: /**
 2750:  * xmlTextWriterWriteVFormatCDATA:
 2751:  * @writer:  the xmlTextWriterPtr
 2752:  * @format:  format string (see printf)
 2753:  * @argptr:  pointer to the first member of the variable argument list.
 2754:  *
 2755:  * Write a formatted xml CDATA.
 2756:  *
 2757:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2758:  */
 2759: int
 2760: xmlTextWriterWriteVFormatCDATA(xmlTextWriterPtr writer, const char *format,
 2761:                                va_list argptr)
 2762: {
 2763:     int rc;
 2764:     xmlChar *buf;
 2765: 
 2766:     if (writer == NULL)
 2767:         return -1;
 2768: 
 2769:     buf = xmlTextWriterVSprintf(format, argptr);
 2770:     if (buf == NULL)
 2771:         return -1;
 2772: 
 2773:     rc = xmlTextWriterWriteCDATA(writer, buf);
 2774: 
 2775:     xmlFree(buf);
 2776:     return rc;
 2777: }
 2778: 
 2779: /**
 2780:  * xmlTextWriterWriteCDATA:
 2781:  * @writer:  the xmlTextWriterPtr
 2782:  * @content:  CDATA content
 2783:  *
 2784:  * Write an xml CDATA.
 2785:  *
 2786:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2787:  */
 2788: int
 2789: xmlTextWriterWriteCDATA(xmlTextWriterPtr writer, const xmlChar * content)
 2790: {
 2791:     int count;
 2792:     int sum;
 2793: 
 2794:     sum = 0;
 2795:     count = xmlTextWriterStartCDATA(writer);
 2796:     if (count == -1)
 2797:         return -1;
 2798:     sum += count;
 2799:     if (content != 0) {
 2800:         count = xmlTextWriterWriteString(writer, content);
 2801:         if (count == -1)
 2802:             return -1;
 2803:         sum += count;
 2804:     }
 2805:     count = xmlTextWriterEndCDATA(writer);
 2806:     if (count == -1)
 2807:         return -1;
 2808:     sum += count;
 2809: 
 2810:     return sum;
 2811: }
 2812: 
 2813: /**
 2814:  * xmlTextWriterStartDTD:
 2815:  * @writer:  the xmlTextWriterPtr
 2816:  * @name:  the name of the DTD
 2817:  * @pubid:  the public identifier, which is an alternative to the system identifier
 2818:  * @sysid:  the system identifier, which is the URI of the DTD
 2819:  *
 2820:  * Start an xml DTD.
 2821:  *
 2822:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2823:  */
 2824: int
 2825: xmlTextWriterStartDTD(xmlTextWriterPtr writer,
 2826:                       const xmlChar * name,
 2827:                       const xmlChar * pubid, const xmlChar * sysid)
 2828: {
 2829:     int count;
 2830:     int sum;
 2831:     xmlLinkPtr lk;
 2832:     xmlTextWriterStackEntry *p;
 2833: 
 2834:     if (writer == NULL || name == NULL || *name == '\0')
 2835:         return -1;
 2836: 
 2837:     sum = 0;
 2838:     lk = xmlListFront(writer->nodes);
 2839:     if ((lk != NULL) && (xmlLinkGetData(lk) != NULL)) {
 2840:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 2841:                         "xmlTextWriterStartDTD : DTD allowed only in prolog!\n");
 2842:         return -1;
 2843:     }
 2844: 
 2845:     p = (xmlTextWriterStackEntry *)
 2846:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 2847:     if (p == 0) {
 2848:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 2849:                         "xmlTextWriterStartDTD : out of memory!\n");
 2850:         return -1;
 2851:     }
 2852: 
 2853:     p->name = xmlStrdup(name);
 2854:     if (p->name == 0) {
 2855:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 2856:                         "xmlTextWriterStartDTD : out of memory!\n");
 2857:         xmlFree(p);
 2858:         return -1;
 2859:     }
 2860:     p->state = XML_TEXTWRITER_DTD;
 2861: 
 2862:     xmlListPushFront(writer->nodes, p);
 2863: 
 2864:     count = xmlOutputBufferWriteString(writer->out, "<!DOCTYPE ");
 2865:     if (count < 0)
 2866:         return -1;
 2867:     sum += count;
 2868:     count = xmlOutputBufferWriteString(writer->out, (const char *) name);
 2869:     if (count < 0)
 2870:         return -1;
 2871:     sum += count;
 2872: 
 2873:     if (pubid != 0) {
 2874:         if (sysid == 0) {
 2875:             xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 2876:                             "xmlTextWriterStartDTD : system identifier needed!\n");
 2877:             return -1;
 2878:         }
 2879: 
 2880:         if (writer->indent)
 2881:             count = xmlOutputBufferWrite(writer->out, 1, "\n");
 2882:         else
 2883:             count = xmlOutputBufferWrite(writer->out, 1, " ");
 2884:         if (count < 0)
 2885:             return -1;
 2886:         sum += count;
 2887: 
 2888:         count = xmlOutputBufferWriteString(writer->out, "PUBLIC ");
 2889:         if (count < 0)
 2890:             return -1;
 2891:         sum += count;
 2892: 
 2893:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 2894:         if (count < 0)
 2895:             return -1;
 2896:         sum += count;
 2897: 
 2898:         count =
 2899:             xmlOutputBufferWriteString(writer->out, (const char *) pubid);
 2900:         if (count < 0)
 2901:             return -1;
 2902:         sum += count;
 2903: 
 2904:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 2905:         if (count < 0)
 2906:             return -1;
 2907:         sum += count;
 2908:     }
 2909: 
 2910:     if (sysid != 0) {
 2911:         if (pubid == 0) {
 2912:             if (writer->indent)
 2913:                 count = xmlOutputBufferWrite(writer->out, 1, "\n");
 2914:             else
 2915:                 count = xmlOutputBufferWrite(writer->out, 1, " ");
 2916:             if (count < 0)
 2917:                 return -1;
 2918:             sum += count;
 2919:             count = xmlOutputBufferWriteString(writer->out, "SYSTEM ");
 2920:             if (count < 0)
 2921:                 return -1;
 2922:             sum += count;
 2923:         } else {
 2924: 			if (writer->indent)
 2925:             count = xmlOutputBufferWriteString(writer->out, "\n       ");
 2926:             else
 2927:                 count = xmlOutputBufferWrite(writer->out, 1, " ");
 2928:             if (count < 0)
 2929:                 return -1;
 2930:             sum += count;
 2931:         }
 2932: 
 2933:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 2934:         if (count < 0)
 2935:             return -1;
 2936:         sum += count;
 2937: 
 2938:         count =
 2939:             xmlOutputBufferWriteString(writer->out, (const char *) sysid);
 2940:         if (count < 0)
 2941:             return -1;
 2942:         sum += count;
 2943: 
 2944:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 2945:         if (count < 0)
 2946:             return -1;
 2947:         sum += count;
 2948:     }
 2949: 
 2950:     return sum;
 2951: }
 2952: 
 2953: /**
 2954:  * xmlTextWriterEndDTD:
 2955:  * @writer:  the xmlTextWriterPtr
 2956:  *
 2957:  * End an xml DTD.
 2958:  *
 2959:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 2960:  */
 2961: int
 2962: xmlTextWriterEndDTD(xmlTextWriterPtr writer)
 2963: {
 2964:     int loop;
 2965:     int count;
 2966:     int sum;
 2967:     xmlLinkPtr lk;
 2968:     xmlTextWriterStackEntry *p;
 2969: 
 2970:     if (writer == NULL)
 2971:         return -1;
 2972: 
 2973:     sum = 0;
 2974:     loop = 1;
 2975:     while (loop) {
 2976:         lk = xmlListFront(writer->nodes);
 2977:         if (lk == NULL)
 2978:             break;
 2979:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 2980:         if (p == 0)
 2981:             break;
 2982:         switch (p->state) {
 2983:             case XML_TEXTWRITER_DTD_TEXT:
 2984:                 count = xmlOutputBufferWriteString(writer->out, "]");
 2985:                 if (count < 0)
 2986:                     return -1;
 2987:                 sum += count;
 2988:                 /* fallthrough */
 2989:             case XML_TEXTWRITER_DTD:
 2990:                 count = xmlOutputBufferWriteString(writer->out, ">");
 2991: 
 2992:                 if (writer->indent) {
 2993:                     if (count < 0)
 2994:                         return -1;
 2995:                     sum += count;
 2996:                     count = xmlOutputBufferWriteString(writer->out, "\n");
 2997:                 }
 2998: 
 2999:                 xmlListPopFront(writer->nodes);
 3000:                 break;
 3001:             case XML_TEXTWRITER_DTD_ELEM:
 3002:             case XML_TEXTWRITER_DTD_ELEM_TEXT:
 3003:                 count = xmlTextWriterEndDTDElement(writer);
 3004:                 break;
 3005:             case XML_TEXTWRITER_DTD_ATTL:
 3006:             case XML_TEXTWRITER_DTD_ATTL_TEXT:
 3007:                 count = xmlTextWriterEndDTDAttlist(writer);
 3008:                 break;
 3009:             case XML_TEXTWRITER_DTD_ENTY:
 3010:             case XML_TEXTWRITER_DTD_PENT:
 3011:             case XML_TEXTWRITER_DTD_ENTY_TEXT:
 3012:                 count = xmlTextWriterEndDTDEntity(writer);
 3013:                 break;
 3014:             case XML_TEXTWRITER_COMMENT:
 3015:                 count = xmlTextWriterEndComment(writer);
 3016:                 break;
 3017:             default:
 3018:                 loop = 0;
 3019:                 continue;
 3020:         }
 3021: 
 3022:         if (count < 0)
 3023:             return -1;
 3024:         sum += count;
 3025:     }
 3026: 
 3027:     return sum;
 3028: }
 3029: 
 3030: /**
 3031:  * xmlTextWriterWriteFormatDTD:
 3032:  * @writer:  the xmlTextWriterPtr
 3033:  * @name:  the name of the DTD
 3034:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3035:  * @sysid:  the system identifier, which is the URI of the DTD
 3036:  * @format:  format string (see printf)
 3037:  * @...:  extra parameters for the format
 3038:  *
 3039:  * Write a DTD with a formatted markup declarations part.
 3040:  *
 3041:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3042:  */
 3043: int XMLCDECL
 3044: xmlTextWriterWriteFormatDTD(xmlTextWriterPtr writer,
 3045:                             const xmlChar * name,
 3046:                             const xmlChar * pubid,
 3047:                             const xmlChar * sysid, const char *format, ...)
 3048: {
 3049:     int rc;
 3050:     va_list ap;
 3051: 
 3052:     va_start(ap, format);
 3053: 
 3054:     rc = xmlTextWriterWriteVFormatDTD(writer, name, pubid, sysid, format,
 3055:                                       ap);
 3056: 
 3057:     va_end(ap);
 3058:     return rc;
 3059: }
 3060: 
 3061: /**
 3062:  * xmlTextWriterWriteVFormatDTD:
 3063:  * @writer:  the xmlTextWriterPtr
 3064:  * @name:  the name of the DTD
 3065:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3066:  * @sysid:  the system identifier, which is the URI of the DTD
 3067:  * @format:  format string (see printf)
 3068:  * @argptr:  pointer to the first member of the variable argument list.
 3069:  *
 3070:  * Write a DTD with a formatted markup declarations part.
 3071:  *
 3072:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3073:  */
 3074: int
 3075: xmlTextWriterWriteVFormatDTD(xmlTextWriterPtr writer,
 3076:                              const xmlChar * name,
 3077:                              const xmlChar * pubid,
 3078:                              const xmlChar * sysid,
 3079:                              const char *format, va_list argptr)
 3080: {
 3081:     int rc;
 3082:     xmlChar *buf;
 3083: 
 3084:     if (writer == NULL)
 3085:         return -1;
 3086: 
 3087:     buf = xmlTextWriterVSprintf(format, argptr);
 3088:     if (buf == NULL)
 3089:         return -1;
 3090: 
 3091:     rc = xmlTextWriterWriteDTD(writer, name, pubid, sysid, buf);
 3092: 
 3093:     xmlFree(buf);
 3094:     return rc;
 3095: }
 3096: 
 3097: /**
 3098:  * xmlTextWriterWriteDTD:
 3099:  * @writer:  the xmlTextWriterPtr
 3100:  * @name:  the name of the DTD
 3101:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3102:  * @sysid:  the system identifier, which is the URI of the DTD
 3103:  * @subset:  string content of the DTD
 3104:  *
 3105:  * Write a DTD.
 3106:  *
 3107:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3108:  */
 3109: int
 3110: xmlTextWriterWriteDTD(xmlTextWriterPtr writer,
 3111:                       const xmlChar * name,
 3112:                       const xmlChar * pubid,
 3113:                       const xmlChar * sysid, const xmlChar * subset)
 3114: {
 3115:     int count;
 3116:     int sum;
 3117: 
 3118:     sum = 0;
 3119:     count = xmlTextWriterStartDTD(writer, name, pubid, sysid);
 3120:     if (count == -1)
 3121:         return -1;
 3122:     sum += count;
 3123:     if (subset != 0) {
 3124:         count = xmlTextWriterWriteString(writer, subset);
 3125:         if (count == -1)
 3126:             return -1;
 3127:         sum += count;
 3128:     }
 3129:     count = xmlTextWriterEndDTD(writer);
 3130:     if (count == -1)
 3131:         return -1;
 3132:     sum += count;
 3133: 
 3134:     return sum;
 3135: }
 3136: 
 3137: /**
 3138:  * xmlTextWriterStartDTDElement:
 3139:  * @writer:  the xmlTextWriterPtr
 3140:  * @name:  the name of the DTD element
 3141:  *
 3142:  * Start an xml DTD element.
 3143:  *
 3144:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3145:  */
 3146: int
 3147: xmlTextWriterStartDTDElement(xmlTextWriterPtr writer, const xmlChar * name)
 3148: {
 3149:     int count;
 3150:     int sum;
 3151:     xmlLinkPtr lk;
 3152:     xmlTextWriterStackEntry *p;
 3153: 
 3154:     if (writer == NULL || name == NULL || *name == '\0')
 3155:         return -1;
 3156: 
 3157:     sum = 0;
 3158:     lk = xmlListFront(writer->nodes);
 3159:     if (lk == 0) {
 3160:         return -1;
 3161:     }
 3162: 
 3163:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3164:     if (p != 0) {
 3165:         switch (p->state) {
 3166:             case XML_TEXTWRITER_DTD:
 3167:                 count = xmlOutputBufferWriteString(writer->out, " [");
 3168:                 if (count < 0)
 3169:                     return -1;
 3170:                 sum += count;
 3171:                 if (writer->indent) {
 3172:                     count = xmlOutputBufferWriteString(writer->out, "\n");
 3173:                     if (count < 0)
 3174:                         return -1;
 3175:                     sum += count;
 3176:                 }
 3177:                 p->state = XML_TEXTWRITER_DTD_TEXT;
 3178:                 /* fallthrough */
 3179:             case XML_TEXTWRITER_DTD_TEXT:
 3180:             case XML_TEXTWRITER_NONE:
 3181:                 break;
 3182:             default:
 3183:                 return -1;
 3184:         }
 3185:     }
 3186: 
 3187:     p = (xmlTextWriterStackEntry *)
 3188:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 3189:     if (p == 0) {
 3190:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3191:                         "xmlTextWriterStartDTDElement : out of memory!\n");
 3192:         return -1;
 3193:     }
 3194: 
 3195:     p->name = xmlStrdup(name);
 3196:     if (p->name == 0) {
 3197:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3198:                         "xmlTextWriterStartDTDElement : out of memory!\n");
 3199:         xmlFree(p);
 3200:         return -1;
 3201:     }
 3202:     p->state = XML_TEXTWRITER_DTD_ELEM;
 3203: 
 3204:     xmlListPushFront(writer->nodes, p);
 3205: 
 3206:     if (writer->indent) {
 3207:         count = xmlTextWriterWriteIndent(writer);
 3208:         if (count < 0)
 3209:             return -1;
 3210:         sum += count;
 3211:     }
 3212: 
 3213:     count = xmlOutputBufferWriteString(writer->out, "<!ELEMENT ");
 3214:     if (count < 0)
 3215:         return -1;
 3216:     sum += count;
 3217:     count = xmlOutputBufferWriteString(writer->out, (const char *) name);
 3218:     if (count < 0)
 3219:         return -1;
 3220:     sum += count;
 3221: 
 3222:     return sum;
 3223: }
 3224: 
 3225: /**
 3226:  * xmlTextWriterEndDTDElement:
 3227:  * @writer:  the xmlTextWriterPtr
 3228:  *
 3229:  * End an xml DTD element.
 3230:  *
 3231:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3232:  */
 3233: int
 3234: xmlTextWriterEndDTDElement(xmlTextWriterPtr writer)
 3235: {
 3236:     int count;
 3237:     int sum;
 3238:     xmlLinkPtr lk;
 3239:     xmlTextWriterStackEntry *p;
 3240: 
 3241:     if (writer == NULL)
 3242:         return -1;
 3243: 
 3244:     sum = 0;
 3245:     lk = xmlListFront(writer->nodes);
 3246:     if (lk == 0)
 3247:         return -1;
 3248: 
 3249:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3250:     if (p == 0)
 3251:         return -1;
 3252: 
 3253:     switch (p->state) {
 3254:         case XML_TEXTWRITER_DTD_ELEM:
 3255:         case XML_TEXTWRITER_DTD_ELEM_TEXT:
 3256:             count = xmlOutputBufferWriteString(writer->out, ">");
 3257:             if (count < 0)
 3258:                 return -1;
 3259:             sum += count;
 3260:             break;
 3261:         default:
 3262:             return -1;
 3263:     }
 3264: 
 3265:     if (writer->indent) {
 3266:         count = xmlOutputBufferWriteString(writer->out, "\n");
 3267:         if (count < 0)
 3268:             return -1;
 3269:         sum += count;
 3270:     }
 3271: 
 3272:     xmlListPopFront(writer->nodes);
 3273:     return sum;
 3274: }
 3275: 
 3276: /**
 3277:  * xmlTextWriterWriteFormatDTDElement:
 3278:  * @writer:  the xmlTextWriterPtr
 3279:  * @name:  the name of the DTD element
 3280:  * @format:  format string (see printf)
 3281:  * @...:  extra parameters for the format
 3282:  *
 3283:  * Write a formatted DTD element.
 3284:  *
 3285:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3286:  */
 3287: int XMLCDECL
 3288: xmlTextWriterWriteFormatDTDElement(xmlTextWriterPtr writer,
 3289:                                    const xmlChar * name,
 3290:                                    const char *format, ...)
 3291: {
 3292:     int rc;
 3293:     va_list ap;
 3294: 
 3295:     va_start(ap, format);
 3296: 
 3297:     rc = xmlTextWriterWriteVFormatDTDElement(writer, name, format, ap);
 3298: 
 3299:     va_end(ap);
 3300:     return rc;
 3301: }
 3302: 
 3303: /**
 3304:  * xmlTextWriterWriteVFormatDTDElement:
 3305:  * @writer:  the xmlTextWriterPtr
 3306:  * @name:  the name of the DTD element
 3307:  * @format:  format string (see printf)
 3308:  * @argptr:  pointer to the first member of the variable argument list.
 3309:  *
 3310:  * Write a formatted DTD element.
 3311:  *
 3312:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3313:  */
 3314: int
 3315: xmlTextWriterWriteVFormatDTDElement(xmlTextWriterPtr writer,
 3316:                                     const xmlChar * name,
 3317:                                     const char *format, va_list argptr)
 3318: {
 3319:     int rc;
 3320:     xmlChar *buf;
 3321: 
 3322:     if (writer == NULL)
 3323:         return -1;
 3324: 
 3325:     buf = xmlTextWriterVSprintf(format, argptr);
 3326:     if (buf == NULL)
 3327:         return -1;
 3328: 
 3329:     rc = xmlTextWriterWriteDTDElement(writer, name, buf);
 3330: 
 3331:     xmlFree(buf);
 3332:     return rc;
 3333: }
 3334: 
 3335: /**
 3336:  * xmlTextWriterWriteDTDElement:
 3337:  * @writer:  the xmlTextWriterPtr
 3338:  * @name:  the name of the DTD element
 3339:  * @content:  content of the element
 3340:  *
 3341:  * Write a DTD element.
 3342:  *
 3343:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3344:  */
 3345: int
 3346: xmlTextWriterWriteDTDElement(xmlTextWriterPtr writer,
 3347:                              const xmlChar * name, const xmlChar * content)
 3348: {
 3349:     int count;
 3350:     int sum;
 3351: 
 3352:     if (content == NULL)
 3353:         return -1;
 3354: 
 3355:     sum = 0;
 3356:     count = xmlTextWriterStartDTDElement(writer, name);
 3357:     if (count == -1)
 3358:         return -1;
 3359:     sum += count;
 3360: 
 3361:     count = xmlTextWriterWriteString(writer, content);
 3362:     if (count == -1)
 3363:         return -1;
 3364:     sum += count;
 3365: 
 3366:     count = xmlTextWriterEndDTDElement(writer);
 3367:     if (count == -1)
 3368:         return -1;
 3369:     sum += count;
 3370: 
 3371:     return sum;
 3372: }
 3373: 
 3374: /**
 3375:  * xmlTextWriterStartDTDAttlist:
 3376:  * @writer:  the xmlTextWriterPtr
 3377:  * @name:  the name of the DTD ATTLIST
 3378:  *
 3379:  * Start an xml DTD ATTLIST.
 3380:  *
 3381:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3382:  */
 3383: int
 3384: xmlTextWriterStartDTDAttlist(xmlTextWriterPtr writer, const xmlChar * name)
 3385: {
 3386:     int count;
 3387:     int sum;
 3388:     xmlLinkPtr lk;
 3389:     xmlTextWriterStackEntry *p;
 3390: 
 3391:     if (writer == NULL || name == NULL || *name == '\0')
 3392:         return -1;
 3393: 
 3394:     sum = 0;
 3395:     lk = xmlListFront(writer->nodes);
 3396:     if (lk == 0) {
 3397:         return -1;
 3398:     }
 3399: 
 3400:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3401:     if (p != 0) {
 3402:         switch (p->state) {
 3403:             case XML_TEXTWRITER_DTD:
 3404:                 count = xmlOutputBufferWriteString(writer->out, " [");
 3405:                 if (count < 0)
 3406:                     return -1;
 3407:                 sum += count;
 3408:                 if (writer->indent) {
 3409:                     count = xmlOutputBufferWriteString(writer->out, "\n");
 3410:                     if (count < 0)
 3411:                         return -1;
 3412:                     sum += count;
 3413:                 }
 3414:                 p->state = XML_TEXTWRITER_DTD_TEXT;
 3415:                 /* fallthrough */
 3416:             case XML_TEXTWRITER_DTD_TEXT:
 3417:             case XML_TEXTWRITER_NONE:
 3418:                 break;
 3419:             default:
 3420:                 return -1;
 3421:         }
 3422:     }
 3423: 
 3424:     p = (xmlTextWriterStackEntry *)
 3425:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 3426:     if (p == 0) {
 3427:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3428:                         "xmlTextWriterStartDTDAttlist : out of memory!\n");
 3429:         return -1;
 3430:     }
 3431: 
 3432:     p->name = xmlStrdup(name);
 3433:     if (p->name == 0) {
 3434:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3435:                         "xmlTextWriterStartDTDAttlist : out of memory!\n");
 3436:         xmlFree(p);
 3437:         return -1;
 3438:     }
 3439:     p->state = XML_TEXTWRITER_DTD_ATTL;
 3440: 
 3441:     xmlListPushFront(writer->nodes, p);
 3442: 
 3443:     if (writer->indent) {
 3444:         count = xmlTextWriterWriteIndent(writer);
 3445:         if (count < 0)
 3446:             return -1;
 3447:         sum += count;
 3448:     }
 3449: 
 3450:     count = xmlOutputBufferWriteString(writer->out, "<!ATTLIST ");
 3451:     if (count < 0)
 3452:         return -1;
 3453:     sum += count;
 3454:     count = xmlOutputBufferWriteString(writer->out, (const char *) name);
 3455:     if (count < 0)
 3456:         return -1;
 3457:     sum += count;
 3458: 
 3459:     return sum;
 3460: }
 3461: 
 3462: /**
 3463:  * xmlTextWriterEndDTDAttlist:
 3464:  * @writer:  the xmlTextWriterPtr
 3465:  *
 3466:  * End an xml DTD attribute list.
 3467:  *
 3468:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3469:  */
 3470: int
 3471: xmlTextWriterEndDTDAttlist(xmlTextWriterPtr writer)
 3472: {
 3473:     int count;
 3474:     int sum;
 3475:     xmlLinkPtr lk;
 3476:     xmlTextWriterStackEntry *p;
 3477: 
 3478:     if (writer == NULL)
 3479:         return -1;
 3480: 
 3481:     sum = 0;
 3482:     lk = xmlListFront(writer->nodes);
 3483:     if (lk == 0)
 3484:         return -1;
 3485: 
 3486:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3487:     if (p == 0)
 3488:         return -1;
 3489: 
 3490:     switch (p->state) {
 3491:         case XML_TEXTWRITER_DTD_ATTL:
 3492:         case XML_TEXTWRITER_DTD_ATTL_TEXT:
 3493:             count = xmlOutputBufferWriteString(writer->out, ">");
 3494:             if (count < 0)
 3495:                 return -1;
 3496:             sum += count;
 3497:             break;
 3498:         default:
 3499:             return -1;
 3500:     }
 3501: 
 3502:     if (writer->indent) {
 3503:         count = xmlOutputBufferWriteString(writer->out, "\n");
 3504:         if (count < 0)
 3505:             return -1;
 3506:         sum += count;
 3507:     }
 3508: 
 3509:     xmlListPopFront(writer->nodes);
 3510:     return sum;
 3511: }
 3512: 
 3513: /**
 3514:  * xmlTextWriterWriteFormatDTDAttlist:
 3515:  * @writer:  the xmlTextWriterPtr
 3516:  * @name:  the name of the DTD ATTLIST
 3517:  * @format:  format string (see printf)
 3518:  * @...:  extra parameters for the format
 3519:  *
 3520:  * Write a formatted DTD ATTLIST.
 3521:  *
 3522:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3523:  */
 3524: int XMLCDECL
 3525: xmlTextWriterWriteFormatDTDAttlist(xmlTextWriterPtr writer,
 3526:                                    const xmlChar * name,
 3527:                                    const char *format, ...)
 3528: {
 3529:     int rc;
 3530:     va_list ap;
 3531: 
 3532:     va_start(ap, format);
 3533: 
 3534:     rc = xmlTextWriterWriteVFormatDTDAttlist(writer, name, format, ap);
 3535: 
 3536:     va_end(ap);
 3537:     return rc;
 3538: }
 3539: 
 3540: /**
 3541:  * xmlTextWriterWriteVFormatDTDAttlist:
 3542:  * @writer:  the xmlTextWriterPtr
 3543:  * @name:  the name of the DTD ATTLIST
 3544:  * @format:  format string (see printf)
 3545:  * @argptr:  pointer to the first member of the variable argument list.
 3546:  *
 3547:  * Write a formatted DTD ATTLIST.
 3548:  *
 3549:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3550:  */
 3551: int
 3552: xmlTextWriterWriteVFormatDTDAttlist(xmlTextWriterPtr writer,
 3553:                                     const xmlChar * name,
 3554:                                     const char *format, va_list argptr)
 3555: {
 3556:     int rc;
 3557:     xmlChar *buf;
 3558: 
 3559:     if (writer == NULL)
 3560:         return -1;
 3561: 
 3562:     buf = xmlTextWriterVSprintf(format, argptr);
 3563:     if (buf == NULL)
 3564:         return -1;
 3565: 
 3566:     rc = xmlTextWriterWriteDTDAttlist(writer, name, buf);
 3567: 
 3568:     xmlFree(buf);
 3569:     return rc;
 3570: }
 3571: 
 3572: /**
 3573:  * xmlTextWriterWriteDTDAttlist:
 3574:  * @writer:  the xmlTextWriterPtr
 3575:  * @name:  the name of the DTD ATTLIST
 3576:  * @content:  content of the ATTLIST
 3577:  *
 3578:  * Write a DTD ATTLIST.
 3579:  *
 3580:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3581:  */
 3582: int
 3583: xmlTextWriterWriteDTDAttlist(xmlTextWriterPtr writer,
 3584:                              const xmlChar * name, const xmlChar * content)
 3585: {
 3586:     int count;
 3587:     int sum;
 3588: 
 3589:     if (content == NULL)
 3590:         return -1;
 3591: 
 3592:     sum = 0;
 3593:     count = xmlTextWriterStartDTDAttlist(writer, name);
 3594:     if (count == -1)
 3595:         return -1;
 3596:     sum += count;
 3597: 
 3598:     count = xmlTextWriterWriteString(writer, content);
 3599:     if (count == -1)
 3600:         return -1;
 3601:     sum += count;
 3602: 
 3603:     count = xmlTextWriterEndDTDAttlist(writer);
 3604:     if (count == -1)
 3605:         return -1;
 3606:     sum += count;
 3607: 
 3608:     return sum;
 3609: }
 3610: 
 3611: /**
 3612:  * xmlTextWriterStartDTDEntity:
 3613:  * @writer:  the xmlTextWriterPtr
 3614:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3615:  * @name:  the name of the DTD ATTLIST
 3616:  *
 3617:  * Start an xml DTD ATTLIST.
 3618:  *
 3619:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3620:  */
 3621: int
 3622: xmlTextWriterStartDTDEntity(xmlTextWriterPtr writer,
 3623:                             int pe, const xmlChar * name)
 3624: {
 3625:     int count;
 3626:     int sum;
 3627:     xmlLinkPtr lk;
 3628:     xmlTextWriterStackEntry *p;
 3629: 
 3630:     if (writer == NULL || name == NULL || *name == '\0')
 3631:         return -1;
 3632: 
 3633:     sum = 0;
 3634:     lk = xmlListFront(writer->nodes);
 3635:     if (lk != 0) {
 3636: 
 3637:         p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3638:         if (p != 0) {
 3639:             switch (p->state) {
 3640:                 case XML_TEXTWRITER_DTD:
 3641:                     count = xmlOutputBufferWriteString(writer->out, " [");
 3642:                     if (count < 0)
 3643:                         return -1;
 3644:                     sum += count;
 3645:                     if (writer->indent) {
 3646:                         count =
 3647:                             xmlOutputBufferWriteString(writer->out, "\n");
 3648:                         if (count < 0)
 3649:                             return -1;
 3650:                         sum += count;
 3651:                     }
 3652:                     p->state = XML_TEXTWRITER_DTD_TEXT;
 3653:                     /* fallthrough */
 3654:                 case XML_TEXTWRITER_DTD_TEXT:
 3655:                 case XML_TEXTWRITER_NONE:
 3656:                     break;
 3657:                 default:
 3658:                     return -1;
 3659:             }
 3660:         }
 3661:     }
 3662: 
 3663:     p = (xmlTextWriterStackEntry *)
 3664:         xmlMalloc(sizeof(xmlTextWriterStackEntry));
 3665:     if (p == 0) {
 3666:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3667:                         "xmlTextWriterStartDTDElement : out of memory!\n");
 3668:         return -1;
 3669:     }
 3670: 
 3671:     p->name = xmlStrdup(name);
 3672:     if (p->name == 0) {
 3673:         xmlWriterErrMsg(writer, XML_ERR_NO_MEMORY,
 3674:                         "xmlTextWriterStartDTDElement : out of memory!\n");
 3675:         xmlFree(p);
 3676:         return -1;
 3677:     }
 3678: 
 3679:     if (pe != 0)
 3680:         p->state = XML_TEXTWRITER_DTD_PENT;
 3681:     else
 3682:         p->state = XML_TEXTWRITER_DTD_ENTY;
 3683: 
 3684:     xmlListPushFront(writer->nodes, p);
 3685: 
 3686:     if (writer->indent) {
 3687:         count = xmlTextWriterWriteIndent(writer);
 3688:         if (count < 0)
 3689:             return -1;
 3690:         sum += count;
 3691:     }
 3692: 
 3693:     count = xmlOutputBufferWriteString(writer->out, "<!ENTITY ");
 3694:     if (count < 0)
 3695:         return -1;
 3696:     sum += count;
 3697: 
 3698:     if (pe != 0) {
 3699:         count = xmlOutputBufferWriteString(writer->out, "% ");
 3700:         if (count < 0)
 3701:             return -1;
 3702:         sum += count;
 3703:     }
 3704: 
 3705:     count = xmlOutputBufferWriteString(writer->out, (const char *) name);
 3706:     if (count < 0)
 3707:         return -1;
 3708:     sum += count;
 3709: 
 3710:     return sum;
 3711: }
 3712: 
 3713: /**
 3714:  * xmlTextWriterEndDTDEntity:
 3715:  * @writer:  the xmlTextWriterPtr
 3716:  *
 3717:  * End an xml DTD entity.
 3718:  *
 3719:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3720:  */
 3721: int
 3722: xmlTextWriterEndDTDEntity(xmlTextWriterPtr writer)
 3723: {
 3724:     int count;
 3725:     int sum;
 3726:     xmlLinkPtr lk;
 3727:     xmlTextWriterStackEntry *p;
 3728: 
 3729:     if (writer == NULL)
 3730:         return -1;
 3731: 
 3732:     sum = 0;
 3733:     lk = xmlListFront(writer->nodes);
 3734:     if (lk == 0)
 3735:         return -1;
 3736: 
 3737:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3738:     if (p == 0)
 3739:         return -1;
 3740: 
 3741:     switch (p->state) {
 3742:         case XML_TEXTWRITER_DTD_ENTY_TEXT:
 3743:             count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 3744:             if (count < 0)
 3745:                 return -1;
 3746:             sum += count;
 3747:         case XML_TEXTWRITER_DTD_ENTY:
 3748:         case XML_TEXTWRITER_DTD_PENT:
 3749:             count = xmlOutputBufferWriteString(writer->out, ">");
 3750:             if (count < 0)
 3751:                 return -1;
 3752:             sum += count;
 3753:             break;
 3754:         default:
 3755:             return -1;
 3756:     }
 3757: 
 3758:     if (writer->indent) {
 3759:         count = xmlOutputBufferWriteString(writer->out, "\n");
 3760:         if (count < 0)
 3761:             return -1;
 3762:         sum += count;
 3763:     }
 3764: 
 3765:     xmlListPopFront(writer->nodes);
 3766:     return sum;
 3767: }
 3768: 
 3769: /**
 3770:  * xmlTextWriterWriteFormatDTDInternalEntity:
 3771:  * @writer:  the xmlTextWriterPtr
 3772:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3773:  * @name:  the name of the DTD entity
 3774:  * @format:  format string (see printf)
 3775:  * @...:  extra parameters for the format
 3776:  *
 3777:  * Write a formatted DTD internal entity.
 3778:  *
 3779:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3780:  */
 3781: int XMLCDECL
 3782: xmlTextWriterWriteFormatDTDInternalEntity(xmlTextWriterPtr writer,
 3783:                                           int pe,
 3784:                                           const xmlChar * name,
 3785:                                           const char *format, ...)
 3786: {
 3787:     int rc;
 3788:     va_list ap;
 3789: 
 3790:     va_start(ap, format);
 3791: 
 3792:     rc = xmlTextWriterWriteVFormatDTDInternalEntity(writer, pe, name,
 3793:                                                     format, ap);
 3794: 
 3795:     va_end(ap);
 3796:     return rc;
 3797: }
 3798: 
 3799: /**
 3800:  * xmlTextWriterWriteVFormatDTDInternalEntity:
 3801:  * @writer:  the xmlTextWriterPtr
 3802:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3803:  * @name:  the name of the DTD entity
 3804:  * @format:  format string (see printf)
 3805:  * @argptr:  pointer to the first member of the variable argument list.
 3806:  *
 3807:  * Write a formatted DTD internal entity.
 3808:  *
 3809:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3810:  */
 3811: int
 3812: xmlTextWriterWriteVFormatDTDInternalEntity(xmlTextWriterPtr writer,
 3813:                                            int pe,
 3814:                                            const xmlChar * name,
 3815:                                            const char *format,
 3816:                                            va_list argptr)
 3817: {
 3818:     int rc;
 3819:     xmlChar *buf;
 3820: 
 3821:     if (writer == NULL)
 3822:         return -1;
 3823: 
 3824:     buf = xmlTextWriterVSprintf(format, argptr);
 3825:     if (buf == NULL)
 3826:         return -1;
 3827: 
 3828:     rc = xmlTextWriterWriteDTDInternalEntity(writer, pe, name, buf);
 3829: 
 3830:     xmlFree(buf);
 3831:     return rc;
 3832: }
 3833: 
 3834: /**
 3835:  * xmlTextWriterWriteDTDEntity:
 3836:  * @writer:  the xmlTextWriterPtr
 3837:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3838:  * @name:  the name of the DTD entity
 3839:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3840:  * @sysid:  the system identifier, which is the URI of the DTD
 3841:  * @ndataid:  the xml notation name.
 3842:  * @content:  content of the entity
 3843:  *
 3844:  * Write a DTD entity.
 3845:  *
 3846:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3847:  */
 3848: int
 3849: xmlTextWriterWriteDTDEntity(xmlTextWriterPtr writer,
 3850:                             int pe,
 3851:                             const xmlChar * name,
 3852:                             const xmlChar * pubid,
 3853:                             const xmlChar * sysid,
 3854:                             const xmlChar * ndataid,
 3855:                             const xmlChar * content)
 3856: {
 3857:     if ((content == NULL) && (pubid == NULL) && (sysid == NULL))
 3858:         return -1;
 3859:     if ((pe != 0) && (ndataid != NULL))
 3860:         return -1;
 3861: 
 3862:     if ((pubid == NULL) && (sysid == NULL))
 3863:         return xmlTextWriterWriteDTDInternalEntity(writer, pe, name,
 3864:                                                    content);
 3865: 
 3866:     return xmlTextWriterWriteDTDExternalEntity(writer, pe, name, pubid,
 3867:                                                sysid, ndataid);
 3868: }
 3869: 
 3870: /**
 3871:  * xmlTextWriterWriteDTDInternalEntity:
 3872:  * @writer:  the xmlTextWriterPtr
 3873:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3874:  * @name:  the name of the DTD entity
 3875:  * @content:  content of the entity
 3876:  *
 3877:  * Write a DTD internal entity.
 3878:  *
 3879:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3880:  */
 3881: int
 3882: xmlTextWriterWriteDTDInternalEntity(xmlTextWriterPtr writer,
 3883:                                     int pe,
 3884:                                     const xmlChar * name,
 3885:                                     const xmlChar * content)
 3886: {
 3887:     int count;
 3888:     int sum;
 3889: 
 3890:     if ((name == NULL) || (*name == '\0') || (content == NULL))
 3891:         return -1;
 3892: 
 3893:     sum = 0;
 3894:     count = xmlTextWriterStartDTDEntity(writer, pe, name);
 3895:     if (count == -1)
 3896:         return -1;
 3897:     sum += count;
 3898: 
 3899:     count = xmlTextWriterWriteString(writer, content);
 3900:     if (count == -1)
 3901:         return -1;
 3902:     sum += count;
 3903: 
 3904:     count = xmlTextWriterEndDTDEntity(writer);
 3905:     if (count == -1)
 3906:         return -1;
 3907:     sum += count;
 3908: 
 3909:     return sum;
 3910: }
 3911: 
 3912: /**
 3913:  * xmlTextWriterWriteDTDExternalEntity:
 3914:  * @writer:  the xmlTextWriterPtr
 3915:  * @pe:  TRUE if this is a parameter entity, FALSE if not
 3916:  * @name:  the name of the DTD entity
 3917:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3918:  * @sysid:  the system identifier, which is the URI of the DTD
 3919:  * @ndataid:  the xml notation name.
 3920:  *
 3921:  * Write a DTD external entity. The entity must have been started with xmlTextWriterStartDTDEntity
 3922:  *
 3923:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3924:  */
 3925: int
 3926: xmlTextWriterWriteDTDExternalEntity(xmlTextWriterPtr writer,
 3927:                                     int pe,
 3928:                                     const xmlChar * name,
 3929:                                     const xmlChar * pubid,
 3930:                                     const xmlChar * sysid,
 3931:                                     const xmlChar * ndataid)
 3932: {
 3933:     int count;
 3934:     int sum;
 3935: 
 3936:     if (((pubid == NULL) && (sysid == NULL)))
 3937:         return -1;
 3938:     if ((pe != 0) && (ndataid != NULL))
 3939:         return -1;
 3940: 
 3941:     sum = 0;
 3942:     count = xmlTextWriterStartDTDEntity(writer, pe, name);
 3943:     if (count == -1)
 3944:         return -1;
 3945:     sum += count;
 3946: 
 3947:     count =
 3948:         xmlTextWriterWriteDTDExternalEntityContents(writer, pubid, sysid,
 3949:                                                     ndataid);
 3950:     if (count < 0)
 3951:         return -1;
 3952:     sum += count;
 3953: 
 3954:     count = xmlTextWriterEndDTDEntity(writer);
 3955:     if (count == -1)
 3956:         return -1;
 3957:     sum += count;
 3958: 
 3959:     return sum;
 3960: }
 3961: 
 3962: /**
 3963:  * xmlTextWriterWriteDTDExternalEntityContents:
 3964:  * @writer:  the xmlTextWriterPtr
 3965:  * @pubid:  the public identifier, which is an alternative to the system identifier
 3966:  * @sysid:  the system identifier, which is the URI of the DTD
 3967:  * @ndataid:  the xml notation name.
 3968:  *
 3969:  * Write the contents of a DTD external entity.
 3970:  *
 3971:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 3972:  */
 3973: int
 3974: xmlTextWriterWriteDTDExternalEntityContents(xmlTextWriterPtr writer,
 3975:                                             const xmlChar * pubid,
 3976:                                             const xmlChar * sysid,
 3977:                                             const xmlChar * ndataid)
 3978: {
 3979:     int count;
 3980:     int sum;
 3981:     xmlLinkPtr lk;
 3982:     xmlTextWriterStackEntry *p;
 3983: 
 3984:     if (writer == NULL) {
 3985:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 3986:                         "xmlTextWriterWriteDTDExternalEntityContents: xmlTextWriterPtr invalid!\n");
 3987:         return -1;
 3988:     }
 3989: 
 3990:     sum = 0;
 3991:     lk = xmlListFront(writer->nodes);
 3992:     if (lk == 0) {
 3993:         xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 3994:                         "xmlTextWriterWriteDTDExternalEntityContents: you must call xmlTextWriterStartDTDEntity before the call to this function!\n");
 3995:         return -1;
 3996:     }
 3997: 
 3998:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 3999:     if (p == 0)
 4000:         return -1;
 4001: 
 4002:     switch (p->state) {
 4003:         case XML_TEXTWRITER_DTD_ENTY:
 4004:             break;
 4005:         case XML_TEXTWRITER_DTD_PENT:
 4006:             if (ndataid != NULL) {
 4007:                 xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 4008:                                 "xmlTextWriterWriteDTDExternalEntityContents: notation not allowed with parameter entities!\n");
 4009:                 return -1;
 4010:             }
 4011:             break;
 4012:         default:
 4013:             xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 4014:                             "xmlTextWriterWriteDTDExternalEntityContents: you must call xmlTextWriterStartDTDEntity before the call to this function!\n");
 4015:             return -1;
 4016:     }
 4017: 
 4018:     if (pubid != 0) {
 4019:         if (sysid == 0) {
 4020:             xmlWriterErrMsg(writer, XML_ERR_INTERNAL_ERROR,
 4021:                             "xmlTextWriterWriteDTDExternalEntityContents: system identifier needed!\n");
 4022:             return -1;
 4023:         }
 4024: 
 4025:         count = xmlOutputBufferWriteString(writer->out, " PUBLIC ");
 4026:         if (count < 0)
 4027:             return -1;
 4028:         sum += count;
 4029: 
 4030:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4031:         if (count < 0)
 4032:             return -1;
 4033:         sum += count;
 4034: 
 4035:         count =
 4036:             xmlOutputBufferWriteString(writer->out, (const char *) pubid);
 4037:         if (count < 0)
 4038:             return -1;
 4039:         sum += count;
 4040: 
 4041:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4042:         if (count < 0)
 4043:             return -1;
 4044:         sum += count;
 4045:     }
 4046: 
 4047:     if (sysid != 0) {
 4048:         if (pubid == 0) {
 4049:             count = xmlOutputBufferWriteString(writer->out, " SYSTEM");
 4050:             if (count < 0)
 4051:                 return -1;
 4052:             sum += count;
 4053:         }
 4054: 
 4055:         count = xmlOutputBufferWriteString(writer->out, " ");
 4056:         if (count < 0)
 4057:             return -1;
 4058:         sum += count;
 4059: 
 4060:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4061:         if (count < 0)
 4062:             return -1;
 4063:         sum += count;
 4064: 
 4065:         count =
 4066:             xmlOutputBufferWriteString(writer->out, (const char *) sysid);
 4067:         if (count < 0)
 4068:             return -1;
 4069:         sum += count;
 4070: 
 4071:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4072:         if (count < 0)
 4073:             return -1;
 4074:         sum += count;
 4075:     }
 4076: 
 4077:     if (ndataid != NULL) {
 4078:         count = xmlOutputBufferWriteString(writer->out, " NDATA ");
 4079:         if (count < 0)
 4080:             return -1;
 4081:         sum += count;
 4082: 
 4083:         count =
 4084:             xmlOutputBufferWriteString(writer->out,
 4085:                                        (const char *) ndataid);
 4086:         if (count < 0)
 4087:             return -1;
 4088:         sum += count;
 4089:     }
 4090: 
 4091:     return sum;
 4092: }
 4093: 
 4094: /**
 4095:  * xmlTextWriterWriteDTDNotation:
 4096:  * @writer:  the xmlTextWriterPtr
 4097:  * @name:  the name of the xml notation
 4098:  * @pubid:  the public identifier, which is an alternative to the system identifier
 4099:  * @sysid:  the system identifier, which is the URI of the DTD
 4100:  *
 4101:  * Write a DTD entity.
 4102:  *
 4103:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 4104:  */
 4105: int
 4106: xmlTextWriterWriteDTDNotation(xmlTextWriterPtr writer,
 4107:                               const xmlChar * name,
 4108:                               const xmlChar * pubid, const xmlChar * sysid)
 4109: {
 4110:     int count;
 4111:     int sum;
 4112:     xmlLinkPtr lk;
 4113:     xmlTextWriterStackEntry *p;
 4114: 
 4115:     if (writer == NULL || name == NULL || *name == '\0')
 4116:         return -1;
 4117: 
 4118:     sum = 0;
 4119:     lk = xmlListFront(writer->nodes);
 4120:     if (lk == 0) {
 4121:         return -1;
 4122:     }
 4123: 
 4124:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 4125:     if (p != 0) {
 4126:         switch (p->state) {
 4127:             case XML_TEXTWRITER_DTD:
 4128:                 count = xmlOutputBufferWriteString(writer->out, " [");
 4129:                 if (count < 0)
 4130:                     return -1;
 4131:                 sum += count;
 4132:                 if (writer->indent) {
 4133:                     count = xmlOutputBufferWriteString(writer->out, "\n");
 4134:                     if (count < 0)
 4135:                         return -1;
 4136:                     sum += count;
 4137:                 }
 4138:                 p->state = XML_TEXTWRITER_DTD_TEXT;
 4139:                 /* fallthrough */
 4140:             case XML_TEXTWRITER_DTD_TEXT:
 4141:                 break;
 4142:             default:
 4143:                 return -1;
 4144:         }
 4145:     }
 4146: 
 4147:     if (writer->indent) {
 4148:         count = xmlTextWriterWriteIndent(writer);
 4149:         if (count < 0)
 4150:             return -1;
 4151:         sum += count;
 4152:     }
 4153: 
 4154:     count = xmlOutputBufferWriteString(writer->out, "<!NOTATION ");
 4155:     if (count < 0)
 4156:         return -1;
 4157:     sum += count;
 4158:     count = xmlOutputBufferWriteString(writer->out, (const char *) name);
 4159:     if (count < 0)
 4160:         return -1;
 4161:     sum += count;
 4162: 
 4163:     if (pubid != 0) {
 4164:         count = xmlOutputBufferWriteString(writer->out, " PUBLIC ");
 4165:         if (count < 0)
 4166:             return -1;
 4167:         sum += count;
 4168:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4169:         if (count < 0)
 4170:             return -1;
 4171:         sum += count;
 4172:         count =
 4173:             xmlOutputBufferWriteString(writer->out, (const char *) pubid);
 4174:         if (count < 0)
 4175:             return -1;
 4176:         sum += count;
 4177:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4178:         if (count < 0)
 4179:             return -1;
 4180:         sum += count;
 4181:     }
 4182: 
 4183:     if (sysid != 0) {
 4184:         if (pubid == 0) {
 4185:             count = xmlOutputBufferWriteString(writer->out, " SYSTEM");
 4186:             if (count < 0)
 4187:                 return -1;
 4188:             sum += count;
 4189:         }
 4190:         count = xmlOutputBufferWriteString(writer->out, " ");
 4191:         if (count < 0)
 4192:             return -1;
 4193:         sum += count;
 4194:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4195:         if (count < 0)
 4196:             return -1;
 4197:         sum += count;
 4198:         count =
 4199:             xmlOutputBufferWriteString(writer->out, (const char *) sysid);
 4200:         if (count < 0)
 4201:             return -1;
 4202:         sum += count;
 4203:         count = xmlOutputBufferWrite(writer->out, 1, &writer->qchar);
 4204:         if (count < 0)
 4205:             return -1;
 4206:         sum += count;
 4207:     }
 4208: 
 4209:     count = xmlOutputBufferWriteString(writer->out, ">");
 4210:     if (count < 0)
 4211:         return -1;
 4212:     sum += count;
 4213: 
 4214:     return sum;
 4215: }
 4216: 
 4217: /**
 4218:  * xmlTextWriterFlush:
 4219:  * @writer:  the xmlTextWriterPtr
 4220:  *
 4221:  * Flush the output buffer.
 4222:  *
 4223:  * Returns the bytes written (may be 0 because of buffering) or -1 in case of error
 4224:  */
 4225: int
 4226: xmlTextWriterFlush(xmlTextWriterPtr writer)
 4227: {
 4228:     int count;
 4229: 
 4230:     if (writer == NULL)
 4231:         return -1;
 4232: 
 4233:     if (writer->out == NULL)
 4234:         count = 0;
 4235:     else
 4236:         count = xmlOutputBufferFlush(writer->out);
 4237: 
 4238:     return count;
 4239: }
 4240: 
 4241: /**
 4242:  * misc
 4243:  */
 4244: 
 4245: /**
 4246:  * xmlFreeTextWriterStackEntry:
 4247:  * @lk:  the xmlLinkPtr
 4248:  *
 4249:  * Free callback for the xmlList.
 4250:  */
 4251: static void
 4252: xmlFreeTextWriterStackEntry(xmlLinkPtr lk)
 4253: {
 4254:     xmlTextWriterStackEntry *p;
 4255: 
 4256:     p = (xmlTextWriterStackEntry *) xmlLinkGetData(lk);
 4257:     if (p == 0)
 4258:         return;
 4259: 
 4260:     if (p->name != 0)
 4261:         xmlFree(p->name);
 4262:     xmlFree(p);
 4263: }
 4264: 
 4265: /**
 4266:  * xmlCmpTextWriterStackEntry:
 4267:  * @data0:  the first data
 4268:  * @data1:  the second data
 4269:  *
 4270:  * Compare callback for the xmlList.
 4271:  *
 4272:  * Returns -1, 0, 1
 4273:  */
 4274: static int
 4275: xmlCmpTextWriterStackEntry(const void *data0, const void *data1)
 4276: {
 4277:     xmlTextWriterStackEntry *p0;
 4278:     xmlTextWriterStackEntry *p1;
 4279: 
 4280:     if (data0 == data1)
 4281:         return 0;
 4282: 
 4283:     if (data0 == 0)
 4284:         return -1;
 4285: 
 4286:     if (data1 == 0)
 4287:         return 1;
 4288: 
 4289:     p0 = (xmlTextWriterStackEntry *) data0;
 4290:     p1 = (xmlTextWriterStackEntry *) data1;
 4291: 
 4292:     return xmlStrcmp(p0->name, p1->name);
 4293: }
 4294: 
 4295: /**
 4296:  * misc
 4297:  */
 4298: 
 4299: /**
 4300:  * xmlTextWriterOutputNSDecl:
 4301:  * @writer:  the xmlTextWriterPtr
 4302:  *
 4303:  * Output the current namespace declarations.
 4304:  */
 4305: static int
 4306: xmlTextWriterOutputNSDecl(xmlTextWriterPtr writer)
 4307: {
 4308:     xmlLinkPtr lk;
 4309:     xmlTextWriterNsStackEntry *np;
 4310:     int count;
 4311:     int sum;
 4312: 
 4313:     sum = 0;
 4314:     while (!xmlListEmpty(writer->nsstack)) {
 4315:         xmlChar *namespaceURI = NULL;
 4316:         xmlChar *prefix = NULL;
 4317: 
 4318:         lk = xmlListFront(writer->nsstack);
 4319:         np = (xmlTextWriterNsStackEntry *) xmlLinkGetData(lk);
 4320: 
 4321:         if (np != 0) {
 4322:             namespaceURI = xmlStrdup(np->uri);
 4323:             prefix = xmlStrdup(np->prefix);
 4324:         }
 4325: 
 4326:         xmlListPopFront(writer->nsstack);
 4327: 
 4328:         if (np != 0) {
 4329:             count = xmlTextWriterWriteAttribute(writer, prefix, namespaceURI);
 4330:             xmlFree(namespaceURI);
 4331:             xmlFree(prefix);
 4332: 
 4333:             if (count < 0) {
 4334:                 xmlListDelete(writer->nsstack);
 4335:                 writer->nsstack = NULL;
 4336:                 return -1;
 4337:             }
 4338:             sum += count;
 4339:         }
 4340:     }
 4341:     return sum;
 4342: }
 4343: 
 4344: /**
 4345:  * xmlFreeTextWriterNsStackEntry:
 4346:  * @lk:  the xmlLinkPtr
 4347:  *
 4348:  * Free callback for the xmlList.
 4349:  */
 4350: static void
 4351: xmlFreeTextWriterNsStackEntry(xmlLinkPtr lk)
 4352: {
 4353:     xmlTextWriterNsStackEntry *p;
 4354: 
 4355:     p = (xmlTextWriterNsStackEntry *) xmlLinkGetData(lk);
 4356:     if (p == 0)
 4357:         return;
 4358: 
 4359:     if (p->prefix != 0)
 4360:         xmlFree(p->prefix);
 4361:     if (p->uri != 0)
 4362:         xmlFree(p->uri);
 4363: 
 4364:     xmlFree(p);
 4365: }
 4366: 
 4367: /**
 4368:  * xmlCmpTextWriterNsStackEntry:
 4369:  * @data0:  the first data
 4370:  * @data1:  the second data
 4371:  *
 4372:  * Compare callback for the xmlList.
 4373:  *
 4374:  * Returns -1, 0, 1
 4375:  */
 4376: static int
 4377: xmlCmpTextWriterNsStackEntry(const void *data0, const void *data1)
 4378: {
 4379:     xmlTextWriterNsStackEntry *p0;
 4380:     xmlTextWriterNsStackEntry *p1;
 4381:     int rc;
 4382: 
 4383:     if (data0 == data1)
 4384:         return 0;
 4385: 
 4386:     if (data0 == 0)
 4387:         return -1;
 4388: 
 4389:     if (data1 == 0)
 4390:         return 1;
 4391: 
 4392:     p0 = (xmlTextWriterNsStackEntry *) data0;
 4393:     p1 = (xmlTextWriterNsStackEntry *) data1;
 4394: 
 4395:     rc = xmlStrcmp(p0->prefix, p1->prefix);
 4396: 
 4397:     if ((rc != 0) || (p0->elem != p1->elem))
 4398:         rc = -1;
 4399: 
 4400:     return rc;
 4401: }
 4402: 
 4403: /**
 4404:  * xmlTextWriterWriteDocCallback:
 4405:  * @context:  the xmlBufferPtr
 4406:  * @str:  the data to write
 4407:  * @len:  the length of the data
 4408:  *
 4409:  * Write callback for the xmlOutputBuffer with target xmlBuffer
 4410:  *
 4411:  * Returns -1, 0, 1
 4412:  */
 4413: static int
 4414: xmlTextWriterWriteDocCallback(void *context, const xmlChar * str, int len)
 4415: {
 4416:     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) context;
 4417:     int rc;
 4418: 
 4419:     if ((rc = xmlParseChunk(ctxt, (const char *) str, len, 0)) != 0) {
 4420:         xmlWriterErrMsgInt(NULL, XML_ERR_INTERNAL_ERROR,
 4421:                         "xmlTextWriterWriteDocCallback : XML error %d !\n",
 4422:                         rc);
 4423:         return -1;
 4424:     }
 4425: 
 4426:     return len;
 4427: }
 4428: 
 4429: /**
 4430:  * xmlTextWriterCloseDocCallback:
 4431:  * @context:  the xmlBufferPtr
 4432:  *
 4433:  * Close callback for the xmlOutputBuffer with target xmlBuffer
 4434:  *
 4435:  * Returns -1, 0, 1
 4436:  */
 4437: static int
 4438: xmlTextWriterCloseDocCallback(void *context)
 4439: {
 4440:     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) context;
 4441:     int rc;
 4442: 
 4443:     if ((rc = xmlParseChunk(ctxt, NULL, 0, 1)) != 0) {
 4444:         xmlWriterErrMsgInt(NULL, XML_ERR_INTERNAL_ERROR,
 4445:                         "xmlTextWriterWriteDocCallback : XML error %d !\n",
 4446:                         rc);
 4447:         return -1;
 4448:     }
 4449: 
 4450:     return 0;
 4451: }
 4452: 
 4453: /**
 4454:  * xmlTextWriterVSprintf:
 4455:  * @format:  see printf
 4456:  * @argptr:  pointer to the first member of the variable argument list.
 4457:  *
 4458:  * Utility function for formatted output
 4459:  *
 4460:  * Returns a new xmlChar buffer with the data or NULL on error. This buffer must be freed.
 4461:  */
 4462: static xmlChar *
 4463: xmlTextWriterVSprintf(const char *format, va_list argptr)
 4464: {
 4465:     int size;
 4466:     int count;
 4467:     xmlChar *buf;
 4468:     va_list locarg;
 4469: 
 4470:     size = BUFSIZ;
 4471:     buf = (xmlChar *) xmlMalloc(size);
 4472:     if (buf == NULL) {
 4473:         xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
 4474:                         "xmlTextWriterVSprintf : out of memory!\n");
 4475:         return NULL;
 4476:     }
 4477: 
 4478:     VA_COPY(locarg, argptr);
 4479:     while (((count = vsnprintf((char *) buf, size, format, locarg)) < 0)
 4480:            || (count == size - 1) || (count == size) || (count > size)) {
 4481: 	va_end(locarg);
 4482:         xmlFree(buf);
 4483:         size += BUFSIZ;
 4484:         buf = (xmlChar *) xmlMalloc(size);
 4485:         if (buf == NULL) {
 4486:             xmlWriterErrMsg(NULL, XML_ERR_NO_MEMORY,
 4487:                             "xmlTextWriterVSprintf : out of memory!\n");
 4488:             return NULL;
 4489:         }
 4490: 	VA_COPY(locarg, argptr);
 4491:     }
 4492:     va_end(locarg);
 4493: 
 4494:     return buf;
 4495: }
 4496: 
 4497: /**
 4498:  * xmlTextWriterStartDocumentCallback:
 4499:  * @ctx: the user data (XML parser context)
 4500:  *
 4501:  * called at the start of document processing.
 4502:  */
 4503: static void
 4504: xmlTextWriterStartDocumentCallback(void *ctx)
 4505: {
 4506:     xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
 4507:     xmlDocPtr doc;
 4508: 
 4509:     if (ctxt->html) {
 4510: #ifdef LIBXML_HTML_ENABLED
 4511:         if (ctxt->myDoc == NULL)
 4512:             ctxt->myDoc = htmlNewDocNoDtD(NULL, NULL);
 4513:         if (ctxt->myDoc == NULL) {
 4514:             if ((ctxt->sax != NULL) && (ctxt->sax->error != NULL))
 4515:                 ctxt->sax->error(ctxt->userData,
 4516:                                  "SAX.startDocument(): out of memory\n");
 4517:             ctxt->errNo = XML_ERR_NO_MEMORY;
 4518:             ctxt->instate = XML_PARSER_EOF;
 4519:             ctxt->disableSAX = 1;
 4520:             return;
 4521:         }
 4522: #else
 4523:         xmlWriterErrMsg(NULL, XML_ERR_INTERNAL_ERROR,
 4524:                         "libxml2 built without HTML support\n");
 4525:         ctxt->errNo = XML_ERR_INTERNAL_ERROR;
 4526:         ctxt->instate = XML_PARSER_EOF;
 4527:         ctxt->disableSAX = 1;
 4528:         return;
 4529: #endif
 4530:     } else {
 4531:         doc = ctxt->myDoc;
 4532:         if (doc == NULL)
 4533:             doc = ctxt->myDoc = xmlNewDoc(ctxt->version);
 4534:         if (doc != NULL) {
 4535:             if (doc->children == NULL) {
 4536:                 if (ctxt->encoding != NULL)
 4537:                     doc->encoding = xmlStrdup(ctxt->encoding);
 4538:                 else
 4539:                     doc->encoding = NULL;
 4540:                 doc->standalone = ctxt->standalone;
 4541:             }
 4542:         } else {
 4543:             if ((ctxt->sax != NULL) && (ctxt->sax->error != NULL))
 4544:                 ctxt->sax->error(ctxt->userData,
 4545:                                  "SAX.startDocument(): out of memory\n");
 4546:             ctxt->errNo = XML_ERR_NO_MEMORY;
 4547:             ctxt->instate = XML_PARSER_EOF;
 4548:             ctxt->disableSAX = 1;
 4549:             return;
 4550:         }
 4551:     }
 4552:     if ((ctxt->myDoc != NULL) && (ctxt->myDoc->URL == NULL) &&
 4553:         (ctxt->input != NULL) && (ctxt->input->filename != NULL)) {
 4554:         ctxt->myDoc->URL =
 4555:             xmlCanonicPath((const xmlChar *) ctxt->input->filename);
 4556:         if (ctxt->myDoc->URL == NULL)
 4557:             ctxt->myDoc->URL =
 4558:                 xmlStrdup((const xmlChar *) ctxt->input->filename);
 4559:     }
 4560: }
 4561: 
 4562: /**
 4563:  * xmlTextWriterSetIndent:
 4564:  * @writer:  the xmlTextWriterPtr
 4565:  * @indent:  do indentation?
 4566:  *
 4567:  * Set indentation output. indent = 0 do not indentation. indent > 0 do indentation.
 4568:  *
 4569:  * Returns -1 on error or 0 otherwise.
 4570:  */
 4571: int
 4572: xmlTextWriterSetIndent(xmlTextWriterPtr writer, int indent)
 4573: {
 4574:     if ((writer == NULL) || (indent < 0))
 4575:         return -1;
 4576: 
 4577:     writer->indent = indent;
 4578:     writer->doindent = 1;
 4579: 
 4580:     return 0;
 4581: }
 4582: 
 4583: /**
 4584:  * xmlTextWriterSetIndentString:
 4585:  * @writer:  the xmlTextWriterPtr
 4586:  * @str:  the xmlChar string
 4587:  *
 4588:  * Set string indentation.
 4589:  *
 4590:  * Returns -1 on error or 0 otherwise.
 4591:  */
 4592: int
 4593: xmlTextWriterSetIndentString(xmlTextWriterPtr writer, const xmlChar * str)
 4594: {
 4595:     if ((writer == NULL) || (!str))
 4596:         return -1;
 4597: 
 4598:     if (writer->ichar != NULL)
 4599:         xmlFree(writer->ichar);
 4600:     writer->ichar = xmlStrdup(str);
 4601: 
 4602:     if (!writer->ichar)
 4603:         return -1;
 4604:     else
 4605:         return 0;
 4606: }
 4607: 
 4608: /**
 4609:  * xmlTextWriterWriteIndent:
 4610:  * @writer:  the xmlTextWriterPtr
 4611:  *
 4612:  * Write indent string.
 4613:  *
 4614:  * Returns -1 on error or the number of strings written.
 4615:  */
 4616: static int
 4617: xmlTextWriterWriteIndent(xmlTextWriterPtr writer)
 4618: {
 4619:     int lksize;
 4620:     int i;
 4621:     int ret;
 4622: 
 4623:     lksize = xmlListSize(writer->nodes);
 4624:     if (lksize < 1)
 4625:         return (-1);            /* list is empty */
 4626:     for (i = 0; i < (lksize - 1); i++) {
 4627:         ret = xmlOutputBufferWriteString(writer->out,
 4628:                                          (const char *) writer->ichar);
 4629:         if (ret == -1)
 4630:             return (-1);
 4631:     }
 4632: 
 4633:     return (lksize - 1);
 4634: }
 4635: 
 4636: /**
 4637:  * xmlTextWriterHandleStateDependencies:
 4638:  * @writer:  the xmlTextWriterPtr
 4639:  * @p:  the xmlTextWriterStackEntry
 4640:  *
 4641:  * Write state dependent strings.
 4642:  *
 4643:  * Returns -1 on error or the number of characters written.
 4644:  */
 4645: static int
 4646: xmlTextWriterHandleStateDependencies(xmlTextWriterPtr writer,
 4647:                                      xmlTextWriterStackEntry * p)
 4648: {
 4649:     int count;
 4650:     int sum;
 4651:     char extra[3];
 4652: 
 4653:     if (writer == NULL)
 4654:         return -1;
 4655: 
 4656:     if (p == NULL)
 4657:         return 0;
 4658: 
 4659:     sum = 0;
 4660:     extra[0] = extra[1] = extra[2] = '\0';
 4661:     if (p != 0) {
 4662:         sum = 0;
 4663:         switch (p->state) {
 4664:             case XML_TEXTWRITER_NAME:
 4665:                 /* Output namespace declarations */
 4666:                 count = xmlTextWriterOutputNSDecl(writer);
 4667:                 if (count < 0)
 4668:                     return -1;
 4669:                 sum += count;
 4670:                 extra[0] = '>';
 4671:                 p->state = XML_TEXTWRITER_TEXT;
 4672:                 break;
 4673:             case XML_TEXTWRITER_PI:
 4674:                 extra[0] = ' ';
 4675:                 p->state = XML_TEXTWRITER_PI_TEXT;
 4676:                 break;
 4677:             case XML_TEXTWRITER_DTD:
 4678:                 extra[0] = ' ';
 4679:                 extra[1] = '[';
 4680:                 p->state = XML_TEXTWRITER_DTD_TEXT;
 4681:                 break;
 4682:             case XML_TEXTWRITER_DTD_ELEM:
 4683:                 extra[0] = ' ';
 4684:                 p->state = XML_TEXTWRITER_DTD_ELEM_TEXT;
 4685:                 break;
 4686:             case XML_TEXTWRITER_DTD_ATTL:
 4687:                 extra[0] = ' ';
 4688:                 p->state = XML_TEXTWRITER_DTD_ATTL_TEXT;
 4689:                 break;
 4690:             case XML_TEXTWRITER_DTD_ENTY:
 4691:             case XML_TEXTWRITER_DTD_PENT:
 4692:                 extra[0] = ' ';
 4693:                 extra[1] = writer->qchar;
 4694:                 p->state = XML_TEXTWRITER_DTD_ENTY_TEXT;
 4695:                 break;
 4696:             default:
 4697:                 break;
 4698:         }
 4699:     }
 4700: 
 4701:     if (*extra != '\0') {
 4702:         count = xmlOutputBufferWriteString(writer->out, extra);
 4703:         if (count < 0)
 4704:             return -1;
 4705:         sum += count;
 4706:     }
 4707: 
 4708:     return sum;
 4709: }
 4710: 
 4711: #define bottom_xmlwriter
 4712: #include "elfgcchack.h"
 4713: #endif

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>