| [898] | 1 |  | 
|---|
 | 2 | #include "yaml_private.h" | 
|---|
 | 3 |  | 
|---|
 | 4 | /* | 
|---|
 | 5 |  * Declarations. | 
|---|
 | 6 |  */ | 
|---|
 | 7 |  | 
|---|
 | 8 | static int | 
|---|
 | 9 | yaml_emitter_set_writer_error(yaml_emitter_t *emitter, const char *problem); | 
|---|
 | 10 |  | 
|---|
 | 11 | YAML_DECLARE(int) | 
|---|
 | 12 | yaml_emitter_flush(yaml_emitter_t *emitter); | 
|---|
 | 13 |  | 
|---|
 | 14 | /* | 
|---|
 | 15 |  * Set the writer error and return 0. | 
|---|
 | 16 |  */ | 
|---|
 | 17 |  | 
|---|
 | 18 | static int | 
|---|
 | 19 | yaml_emitter_set_writer_error(yaml_emitter_t *emitter, const char *problem) | 
|---|
 | 20 | { | 
|---|
 | 21 |     emitter->error = YAML_WRITER_ERROR; | 
|---|
 | 22 |     emitter->problem = problem; | 
|---|
 | 23 |  | 
|---|
 | 24 |     return 0; | 
|---|
 | 25 | } | 
|---|
 | 26 |  | 
|---|
 | 27 | /* | 
|---|
 | 28 |  * Flush the output buffer. | 
|---|
 | 29 |  */ | 
|---|
 | 30 |  | 
|---|
 | 31 | YAML_DECLARE(int) | 
|---|
 | 32 | yaml_emitter_flush(yaml_emitter_t *emitter) | 
|---|
 | 33 | { | 
|---|
 | 34 |     int low, high; | 
|---|
 | 35 |  | 
|---|
 | 36 |     assert(emitter);    /* Non-NULL emitter object is expected. */ | 
|---|
 | 37 |     assert(emitter->write_handler); /* Write handler must be set. */ | 
|---|
 | 38 |     assert(emitter->encoding);  /* Output encoding must be set. */ | 
|---|
 | 39 |  | 
|---|
 | 40 |     emitter->buffer.last = emitter->buffer.pointer; | 
|---|
 | 41 |     emitter->buffer.pointer = emitter->buffer.start; | 
|---|
 | 42 |  | 
|---|
 | 43 |     /* Check if the buffer is empty. */ | 
|---|
 | 44 |  | 
|---|
 | 45 |     if (emitter->buffer.start == emitter->buffer.last) { | 
|---|
 | 46 |         return 1; | 
|---|
 | 47 |     } | 
|---|
 | 48 |  | 
|---|
 | 49 |     /* If the output encoding is UTF-8, we don't need to recode the buffer. */ | 
|---|
 | 50 |  | 
|---|
 | 51 |     if (emitter->encoding == YAML_UTF8_ENCODING) | 
|---|
 | 52 |     { | 
|---|
 | 53 |         if (emitter->write_handler(emitter->write_handler_data, | 
|---|
 | 54 |                     emitter->buffer.start, | 
|---|
 | 55 |                     emitter->buffer.last - emitter->buffer.start)) { | 
|---|
 | 56 |             emitter->buffer.last = emitter->buffer.start; | 
|---|
 | 57 |             emitter->buffer.pointer = emitter->buffer.start; | 
|---|
 | 58 |             return 1; | 
|---|
 | 59 |         } | 
|---|
 | 60 |         else { | 
|---|
 | 61 |             return yaml_emitter_set_writer_error(emitter, "Write error"); | 
|---|
 | 62 |         } | 
|---|
 | 63 |     } | 
|---|
 | 64 |  | 
|---|
 | 65 |     /* Recode the buffer into the raw buffer. */ | 
|---|
 | 66 |  | 
|---|
 | 67 |     low = (emitter->encoding == YAML_UTF16LE_ENCODING ? 0 : 1); | 
|---|
 | 68 |     high = (emitter->encoding == YAML_UTF16LE_ENCODING ? 1 : 0); | 
|---|
 | 69 |  | 
|---|
 | 70 |     while (emitter->buffer.pointer != emitter->buffer.last) | 
|---|
 | 71 |     { | 
|---|
 | 72 |         unsigned char octet; | 
|---|
 | 73 |         unsigned int width; | 
|---|
 | 74 |         unsigned int value; | 
|---|
 | 75 |         size_t k; | 
|---|
 | 76 |  | 
|---|
 | 77 |         /*  | 
|---|
 | 78 |          * See the "reader.c" code for more details on UTF-8 encoding.  Note | 
|---|
 | 79 |          * that we assume that the buffer contains a valid UTF-8 sequence. | 
|---|
 | 80 |          */ | 
|---|
 | 81 |  | 
|---|
 | 82 |         /* Read the next UTF-8 character. */ | 
|---|
 | 83 |  | 
|---|
 | 84 |         octet = emitter->buffer.pointer[0]; | 
|---|
 | 85 |  | 
|---|
 | 86 |         width = (octet & 0x80) == 0x00 ? 1 : | 
|---|
 | 87 |                 (octet & 0xE0) == 0xC0 ? 2 : | 
|---|
 | 88 |                 (octet & 0xF0) == 0xE0 ? 3 : | 
|---|
 | 89 |                 (octet & 0xF8) == 0xF0 ? 4 : 0; | 
|---|
 | 90 |  | 
|---|
 | 91 |         value = (octet & 0x80) == 0x00 ? octet & 0x7F : | 
|---|
 | 92 |                 (octet & 0xE0) == 0xC0 ? octet & 0x1F : | 
|---|
 | 93 |                 (octet & 0xF0) == 0xE0 ? octet & 0x0F : | 
|---|
 | 94 |                 (octet & 0xF8) == 0xF0 ? octet & 0x07 : 0; | 
|---|
 | 95 |  | 
|---|
 | 96 |         for (k = 1; k < width; k ++) { | 
|---|
 | 97 |             octet = emitter->buffer.pointer[k]; | 
|---|
 | 98 |             value = (value << 6) + (octet & 0x3F); | 
|---|
 | 99 |         } | 
|---|
 | 100 |  | 
|---|
 | 101 |         emitter->buffer.pointer += width; | 
|---|
 | 102 |  | 
|---|
 | 103 |         /* Write the character. */ | 
|---|
 | 104 |  | 
|---|
 | 105 |         if (value < 0x10000) | 
|---|
 | 106 |         { | 
|---|
 | 107 |             emitter->raw_buffer.last[high] = value >> 8; | 
|---|
 | 108 |             emitter->raw_buffer.last[low] = value & 0xFF; | 
|---|
 | 109 |  | 
|---|
 | 110 |             emitter->raw_buffer.last += 2; | 
|---|
 | 111 |         } | 
|---|
 | 112 |         else | 
|---|
 | 113 |         { | 
|---|
 | 114 |             /* Write the character using a surrogate pair (check "reader.c"). */ | 
|---|
 | 115 |  | 
|---|
 | 116 |             value -= 0x10000; | 
|---|
 | 117 |             emitter->raw_buffer.last[high] = 0xD8 + (value >> 18); | 
|---|
 | 118 |             emitter->raw_buffer.last[low] = (value >> 10) & 0xFF; | 
|---|
 | 119 |             emitter->raw_buffer.last[high+2] = 0xDC + ((value >> 8) & 0xFF); | 
|---|
 | 120 |             emitter->raw_buffer.last[low+2] = value & 0xFF; | 
|---|
 | 121 |  | 
|---|
 | 122 |             emitter->raw_buffer.last += 4; | 
|---|
 | 123 |         } | 
|---|
 | 124 |     } | 
|---|
 | 125 |  | 
|---|
 | 126 |     /* Write the raw buffer. */ | 
|---|
 | 127 |  | 
|---|
 | 128 |     if (emitter->write_handler(emitter->write_handler_data, | 
|---|
 | 129 |                 emitter->raw_buffer.start, | 
|---|
 | 130 |                 emitter->raw_buffer.last - emitter->raw_buffer.start)) { | 
|---|
 | 131 |         emitter->buffer.last = emitter->buffer.start; | 
|---|
 | 132 |         emitter->buffer.pointer = emitter->buffer.start; | 
|---|
 | 133 |         emitter->raw_buffer.last = emitter->raw_buffer.start; | 
|---|
 | 134 |         emitter->raw_buffer.pointer = emitter->raw_buffer.start; | 
|---|
 | 135 |         return 1; | 
|---|
 | 136 |     } | 
|---|
 | 137 |     else { | 
|---|
 | 138 |         return yaml_emitter_set_writer_error(emitter, "Write error"); | 
|---|
 | 139 |     } | 
|---|
 | 140 | } | 
|---|
 | 141 |  | 
|---|