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