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