2 * Copyright 2015, 2016 Hans Leidekker for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
28 #include "webservices.h"
30 #include "wine/debug.h"
31 #include "wine/heap.h"
32 #include "wine/list.h"
33 #include "wine/unicode.h"
34 #include "webservices_private.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(webservices
);
38 static const struct prop_desc writer_props
[] =
40 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_MAX_DEPTH */
41 { sizeof(BOOL
), FALSE
}, /* WS_XML_WRITER_PROPERTY_ALLOW_FRAGMENT */
42 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_MAX_ATTRIBUTES */
43 { sizeof(BOOL
), FALSE
}, /* WS_XML_WRITER_PROPERTY_WRITE_DECLARATION */
44 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_INDENT */
45 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_BUFFER_TRIM_SIZE */
46 { sizeof(WS_CHARSET
), FALSE
}, /* WS_XML_WRITER_PROPERTY_CHARSET */
47 { sizeof(WS_BUFFERS
), FALSE
}, /* WS_XML_WRITER_PROPERTY_BUFFERS */
48 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_BUFFER_MAX_SIZE */
49 { sizeof(WS_BYTES
), FALSE
}, /* WS_XML_WRITER_PROPERTY_BYTES */
50 { sizeof(BOOL
), TRUE
}, /* WS_XML_WRITER_PROPERTY_IN_ATTRIBUTE */
51 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_MAX_MIME_PARTS_BUFFER_SIZE */
52 { sizeof(WS_BYTES
), FALSE
}, /* WS_XML_WRITER_PROPERTY_INITIAL_BUFFER */
53 { sizeof(BOOL
), FALSE
}, /* WS_XML_WRITER_PROPERTY_ALLOW_INVALID_CHARACTER_REFERENCES */
54 { sizeof(ULONG
), FALSE
}, /* WS_XML_WRITER_PROPERTY_MAX_NAMESPACES */
55 { sizeof(ULONG
), TRUE
}, /* WS_XML_WRITER_PROPERTY_BYTES_WRITTEN */
56 { sizeof(ULONG
), TRUE
}, /* WS_XML_WRITER_PROPERTY_BYTES_TO_CLOSE */
57 { sizeof(BOOL
), FALSE
}, /* WS_XML_WRITER_PROPERTY_COMPRESS_EMPTY_ELEMENTS */
58 { sizeof(BOOL
), FALSE
} /* WS_XML_WRITER_PROPERTY_EMIT_UNCOMPRESSED_EMPTY_ELEMENTS */
64 WRITER_STATE_STARTELEMENT
,
65 WRITER_STATE_STARTATTRIBUTE
,
66 WRITER_STATE_STARTCDATA
,
67 WRITER_STATE_ENDSTARTELEMENT
,
70 WRITER_STATE_ENDELEMENT
,
79 unsigned char *write_bufptr
;
80 enum writer_state state
;
83 WS_XML_STRING
*current_ns
;
84 WS_XML_WRITER_ENCODING_TYPE output_enc
;
85 WS_CHARSET output_charset
;
86 WS_XML_WRITER_OUTPUT_TYPE output_type
;
87 struct xmlbuf
*output_buf
;
90 const WS_XML_DICTIONARY
*dict
;
92 WS_DYNAMIC_STRING_CALLBACK dict_cb
;
95 struct prop prop
[sizeof(writer_props
)/sizeof(writer_props
[0])];
98 #define WRITER_MAGIC (('W' << 24) | ('R' << 16) | ('I' << 8) | 'T')
100 static struct writer
*alloc_writer(void)
102 static const ULONG count
= sizeof(writer_props
)/sizeof(writer_props
[0]);
104 ULONG size
= sizeof(*ret
) + prop_size( writer_props
, count
);
106 if (!(ret
= heap_alloc_zero( size
))) return NULL
;
108 ret
->magic
= WRITER_MAGIC
;
109 InitializeCriticalSection( &ret
->cs
);
110 ret
->cs
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": writer.cs");
112 prop_init( writer_props
, count
, ret
->prop
, &ret
[1] );
113 ret
->prop_count
= count
;
117 static void free_writer( struct writer
*writer
)
119 destroy_nodes( writer
->root
);
120 free_xml_string( writer
->current_ns
);
121 WsFreeHeap( writer
->output_heap
);
123 writer
->cs
.DebugInfo
->Spare
[0] = 0;
124 DeleteCriticalSection( &writer
->cs
);
128 static void write_insert_eof( struct writer
*writer
, struct node
*eof
)
130 if (!writer
->root
) writer
->root
= eof
;
133 eof
->parent
= writer
->root
;
134 list_add_tail( &writer
->root
->children
, &eof
->entry
);
136 writer
->current
= eof
;
139 static void write_insert_bof( struct writer
*writer
, struct node
*bof
)
141 writer
->root
->parent
= bof
;
142 list_add_tail( &bof
->children
, &writer
->root
->entry
);
143 writer
->current
= writer
->root
= bof
;
146 static void write_insert_node( struct writer
*writer
, struct node
*parent
, struct node
*node
)
148 node
->parent
= parent
;
149 list_add_before( list_tail( &parent
->children
), &node
->entry
);
150 writer
->current
= node
;
153 static struct node
*find_parent( struct writer
*writer
)
155 if (is_valid_parent( writer
->current
)) return writer
->current
;
156 if (is_valid_parent( writer
->current
->parent
)) return writer
->current
->parent
;
160 static HRESULT
init_writer( struct writer
*writer
)
164 writer
->write_pos
= 0;
165 writer
->write_bufptr
= NULL
;
166 destroy_nodes( writer
->root
);
167 writer
->root
= writer
->current
= NULL
;
168 free_xml_string( writer
->current_ns
);
169 writer
->current_ns
= NULL
;
171 if (!(node
= alloc_node( WS_XML_NODE_TYPE_EOF
))) return E_OUTOFMEMORY
;
172 write_insert_eof( writer
, node
);
173 writer
->state
= WRITER_STATE_INITIAL
;
174 writer
->output_enc
= WS_XML_WRITER_ENCODING_TYPE_TEXT
;
175 writer
->output_charset
= WS_CHARSET_UTF8
;
177 writer
->dict_do_lookup
= FALSE
;
178 writer
->dict_cb
= NULL
;
179 writer
->dict_cb_state
= NULL
;
183 /**************************************************************************
184 * WsCreateWriter [webservices.@]
186 HRESULT WINAPI
WsCreateWriter( const WS_XML_WRITER_PROPERTY
*properties
, ULONG count
,
187 WS_XML_WRITER
**handle
, WS_ERROR
*error
)
189 struct writer
*writer
;
190 ULONG i
, max_depth
= 32, max_attrs
= 128, trim_size
= 4096, max_size
= 65536, max_ns
= 32;
191 WS_CHARSET charset
= WS_CHARSET_UTF8
;
194 TRACE( "%p %u %p %p\n", properties
, count
, handle
, error
);
195 if (error
) FIXME( "ignoring error parameter\n" );
197 if (!handle
) return E_INVALIDARG
;
198 if (!(writer
= alloc_writer())) return E_OUTOFMEMORY
;
200 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_MAX_DEPTH
, &max_depth
, sizeof(max_depth
) );
201 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_MAX_ATTRIBUTES
, &max_attrs
, sizeof(max_attrs
) );
202 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_BUFFER_TRIM_SIZE
, &trim_size
, sizeof(trim_size
) );
203 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_CHARSET
, &charset
, sizeof(charset
) );
204 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_BUFFER_MAX_SIZE
, &max_size
, sizeof(max_size
) );
205 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_MAX_MIME_PARTS_BUFFER_SIZE
, &max_size
, sizeof(max_size
) );
206 prop_set( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_MAX_NAMESPACES
, &max_ns
, sizeof(max_ns
) );
208 for (i
= 0; i
< count
; i
++)
210 hr
= prop_set( writer
->prop
, writer
->prop_count
, properties
[i
].id
, properties
[i
].value
,
211 properties
[i
].valueSize
);
214 free_writer( writer
);
219 hr
= prop_get( writer
->prop
, writer
->prop_count
, WS_XML_WRITER_PROPERTY_BUFFER_MAX_SIZE
,
220 &max_size
, sizeof(max_size
) );
223 free_writer( writer
);
227 hr
= WsCreateHeap( max_size
, 0, NULL
, 0, &writer
->output_heap
, NULL
);
230 free_writer( writer
);
234 hr
= init_writer( writer
);
237 free_writer( writer
);
241 TRACE( "created %p\n", writer
);
242 *handle
= (WS_XML_WRITER
*)writer
;
246 /**************************************************************************
247 * WsFreeWriter [webservices.@]
249 void WINAPI
WsFreeWriter( WS_XML_WRITER
*handle
)
251 struct writer
*writer
= (struct writer
*)handle
;
253 TRACE( "%p\n", handle
);
257 EnterCriticalSection( &writer
->cs
);
259 if (writer
->magic
!= WRITER_MAGIC
)
261 LeaveCriticalSection( &writer
->cs
);
267 LeaveCriticalSection( &writer
->cs
);
268 free_writer( writer
);
271 /**************************************************************************
272 * WsGetWriterProperty [webservices.@]
274 HRESULT WINAPI
WsGetWriterProperty( WS_XML_WRITER
*handle
, WS_XML_WRITER_PROPERTY_ID id
,
275 void *buf
, ULONG size
, WS_ERROR
*error
)
277 struct writer
*writer
= (struct writer
*)handle
;
280 TRACE( "%p %u %p %u %p\n", handle
, id
, buf
, size
, error
);
281 if (error
) FIXME( "ignoring error parameter\n" );
283 if (!writer
) return E_INVALIDARG
;
285 EnterCriticalSection( &writer
->cs
);
287 if (writer
->magic
!= WRITER_MAGIC
)
289 LeaveCriticalSection( &writer
->cs
);
293 if (!writer
->output_type
)
295 LeaveCriticalSection( &writer
->cs
);
296 return WS_E_INVALID_OPERATION
;
301 case WS_XML_WRITER_PROPERTY_BYTES
:
303 WS_BYTES
*bytes
= buf
;
304 if (size
!= sizeof(*bytes
)) hr
= E_INVALIDARG
;
307 bytes
->bytes
= writer
->output_buf
->bytes
.bytes
;
308 bytes
->length
= writer
->output_buf
->bytes
.length
;
312 case WS_XML_WRITER_PROPERTY_BUFFERS
:
313 if (writer
->output_buf
->bytes
.length
)
315 WS_BUFFERS
*buffers
= buf
;
316 if (size
!= sizeof(*buffers
)) hr
= E_INVALIDARG
;
319 buffers
->bufferCount
= 1;
320 buffers
->buffers
= &writer
->output_buf
->bytes
;
326 hr
= prop_get( writer
->prop
, writer
->prop_count
, id
, buf
, size
);
329 LeaveCriticalSection( &writer
->cs
);
333 static void set_output_buffer( struct writer
*writer
, struct xmlbuf
*xmlbuf
)
335 /* free current buffer if it's ours */
336 if (writer
->output_buf
&& !writer
->output_buf_user
)
338 free_xmlbuf( writer
->output_buf
);
340 writer
->output_buf
= xmlbuf
;
341 writer
->output_type
= WS_XML_WRITER_OUTPUT_TYPE_BUFFER
;
342 writer
->write_bufptr
= xmlbuf
->bytes
.bytes
;
343 writer
->write_pos
= 0;
346 /**************************************************************************
347 * WsSetOutput [webservices.@]
349 HRESULT WINAPI
WsSetOutput( WS_XML_WRITER
*handle
, const WS_XML_WRITER_ENCODING
*encoding
,
350 const WS_XML_WRITER_OUTPUT
*output
, const WS_XML_WRITER_PROPERTY
*properties
,
351 ULONG count
, WS_ERROR
*error
)
353 struct writer
*writer
= (struct writer
*)handle
;
358 TRACE( "%p %p %p %p %u %p\n", handle
, encoding
, output
, properties
, count
, error
);
359 if (error
) FIXME( "ignoring error parameter\n" );
361 if (!writer
) return E_INVALIDARG
;
363 EnterCriticalSection( &writer
->cs
);
365 if (writer
->magic
!= WRITER_MAGIC
)
367 LeaveCriticalSection( &writer
->cs
);
371 for (i
= 0; i
< count
; i
++)
373 hr
= prop_set( writer
->prop
, writer
->prop_count
, properties
[i
].id
, properties
[i
].value
,
374 properties
[i
].valueSize
);
375 if (hr
!= S_OK
) goto done
;
378 if ((hr
= init_writer( writer
)) != S_OK
) goto done
;
380 switch (encoding
->encodingType
)
382 case WS_XML_WRITER_ENCODING_TYPE_TEXT
:
384 WS_XML_WRITER_TEXT_ENCODING
*text
= (WS_XML_WRITER_TEXT_ENCODING
*)encoding
;
385 if (text
->charSet
!= WS_CHARSET_UTF8
)
387 FIXME( "charset %u not supported\n", text
->charSet
);
391 writer
->output_enc
= WS_XML_WRITER_ENCODING_TYPE_TEXT
;
392 writer
->output_charset
= WS_CHARSET_UTF8
;
395 case WS_XML_WRITER_ENCODING_TYPE_BINARY
:
397 WS_XML_WRITER_BINARY_ENCODING
*bin
= (WS_XML_WRITER_BINARY_ENCODING
*)encoding
;
398 writer
->output_enc
= WS_XML_WRITER_ENCODING_TYPE_BINARY
;
399 writer
->output_charset
= 0;
400 writer
->dict
= bin
->staticDictionary
;
401 writer
->dict_cb
= bin
->dynamicStringCallback
;
402 writer
->dict_cb_state
= bin
->dynamicStringCallbackState
;
406 FIXME( "encoding type %u not supported\n", encoding
->encodingType
);
411 switch (output
->outputType
)
413 case WS_XML_WRITER_OUTPUT_TYPE_BUFFER
:
415 struct xmlbuf
*xmlbuf
;
416 if (!(xmlbuf
= alloc_xmlbuf( writer
->output_heap
, 0, writer
->output_enc
, writer
->output_charset
,
417 writer
->dict
, NULL
)))
419 hr
= WS_E_QUOTA_EXCEEDED
;
422 set_output_buffer( writer
, xmlbuf
);
423 writer
->output_buf_user
= FALSE
;
427 FIXME( "output type %u not supported\n", output
->outputType
);
432 if (!(node
= alloc_node( WS_XML_NODE_TYPE_BOF
))) hr
= E_OUTOFMEMORY
;
433 else write_insert_bof( writer
, node
);
436 LeaveCriticalSection( &writer
->cs
);
440 /**************************************************************************
441 * WsSetOutputToBuffer [webservices.@]
443 HRESULT WINAPI
WsSetOutputToBuffer( WS_XML_WRITER
*handle
, WS_XML_BUFFER
*buffer
,
444 const WS_XML_WRITER_PROPERTY
*properties
, ULONG count
,
447 struct writer
*writer
= (struct writer
*)handle
;
448 struct xmlbuf
*xmlbuf
= (struct xmlbuf
*)buffer
;
453 TRACE( "%p %p %p %u %p\n", handle
, buffer
, properties
, count
, error
);
454 if (error
) FIXME( "ignoring error parameter\n" );
456 if (!writer
|| !xmlbuf
) return E_INVALIDARG
;
458 EnterCriticalSection( &writer
->cs
);
460 if (writer
->magic
!= WRITER_MAGIC
)
462 LeaveCriticalSection( &writer
->cs
);
466 for (i
= 0; i
< count
; i
++)
468 hr
= prop_set( writer
->prop
, writer
->prop_count
, properties
[i
].id
, properties
[i
].value
,
469 properties
[i
].valueSize
);
470 if (hr
!= S_OK
) goto done
;
473 if ((hr
= init_writer( writer
)) != S_OK
) goto done
;
474 writer
->output_enc
= xmlbuf
->encoding
;
475 writer
->output_charset
= xmlbuf
->charset
;
476 set_output_buffer( writer
, xmlbuf
);
477 writer
->output_buf_user
= TRUE
;
479 if (!(node
= alloc_node( WS_XML_NODE_TYPE_BOF
))) hr
= E_OUTOFMEMORY
;
480 else write_insert_bof( writer
, node
);
483 LeaveCriticalSection( &writer
->cs
);
487 static HRESULT
write_grow_buffer( struct writer
*writer
, ULONG size
)
489 struct xmlbuf
*buf
= writer
->output_buf
;
493 if (buf
->size
>= writer
->write_pos
+ size
)
495 buf
->bytes
.length
= writer
->write_pos
+ size
;
498 new_size
= max( buf
->size
* 2, writer
->write_pos
+ size
);
499 if (!(tmp
= ws_realloc( buf
->heap
, buf
->bytes
.bytes
, buf
->size
, new_size
))) return WS_E_QUOTA_EXCEEDED
;
500 writer
->write_bufptr
= buf
->bytes
.bytes
= tmp
;
501 buf
->size
= new_size
;
502 buf
->bytes
.length
= writer
->write_pos
+ size
;
506 static inline void write_char( struct writer
*writer
, unsigned char ch
)
508 writer
->write_bufptr
[writer
->write_pos
++] = ch
;
511 static inline void write_bytes( struct writer
*writer
, const BYTE
*bytes
, ULONG len
)
513 memcpy( writer
->write_bufptr
+ writer
->write_pos
, bytes
, len
);
514 writer
->write_pos
+= len
;
523 static const struct escape escape_lt
= { '<', "<", 4 };
524 static const struct escape escape_gt
= { '>', ">", 4 };
525 static const struct escape escape_amp
= { '&', "&", 5 };
526 static const struct escape escape_apos
= { '\'', "'", 6 };
527 static const struct escape escape_quot
= { '"', """, 6 };
529 static HRESULT
write_bytes_escape( struct writer
*writer
, const BYTE
*bytes
, ULONG len
,
530 const struct escape
**escapes
, ULONG nb_escapes
)
536 for (i
= 0; i
< len
; i
++)
540 for (j
= 0; j
< nb_escapes
; j
++)
542 if (bytes
[i
] == escapes
[j
]->ch
)
544 ptr
= (const BYTE
*)escapes
[j
]->entity
;
545 size
= escapes
[j
]->len
;
549 if ((hr
= write_grow_buffer( writer
, size
)) != S_OK
) return hr
;
550 write_bytes( writer
, ptr
, size
);
556 static HRESULT
write_attribute_value_text( struct writer
*writer
, const WS_XML_TEXT
*text
, BOOL single
)
558 WS_XML_UTF8_TEXT
*utf8
= (WS_XML_UTF8_TEXT
*)text
;
559 const struct escape
*escapes
[3];
561 escapes
[0] = single
? &escape_apos
: &escape_quot
;
562 escapes
[1] = &escape_lt
;
563 escapes
[2] = &escape_amp
;
564 return write_bytes_escape( writer
, utf8
->value
.bytes
, utf8
->value
.length
, escapes
, 3 );
567 static HRESULT
write_attribute_text( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
569 unsigned char quote
= attr
->singleQuote
? '\'' : '"';
570 const WS_XML_STRING
*prefix
= NULL
;
574 if (attr
->prefix
) prefix
= attr
->prefix
;
576 /* ' prefix:attr="value"' */
578 size
= attr
->localName
->length
+ 4 /* ' =""' */;
579 if (prefix
&& prefix
->length
) size
+= prefix
->length
+ 1 /* ':' */;
580 if ((hr
= write_grow_buffer( writer
, size
)) != S_OK
) return hr
;
582 write_char( writer
, ' ' );
583 if (prefix
&& prefix
->length
)
585 write_bytes( writer
, prefix
->bytes
, prefix
->length
);
586 write_char( writer
, ':' );
588 write_bytes( writer
, attr
->localName
->bytes
, attr
->localName
->length
);
589 write_char( writer
, '=' );
590 write_char( writer
, quote
);
591 if (attr
->value
) hr
= write_attribute_value_text( writer
, attr
->value
, attr
->singleQuote
);
592 write_char( writer
, quote
);
597 static HRESULT
write_int31( struct writer
*writer
, ULONG len
)
601 if (len
> 0x7fffffff) return E_INVALIDARG
;
603 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
606 write_char( writer
, len
);
609 write_char( writer
, (len
& 0x7f) | 0x80 );
611 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
612 if ((len
>>= 7) < 0x80)
614 write_char( writer
, len
);
617 write_char( writer
, (len
& 0x7f) | 0x80 );
619 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
620 if ((len
>>= 7) < 0x80)
622 write_char( writer
, len
);
625 write_char( writer
, (len
& 0x7f) | 0x80 );
627 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
628 if ((len
>>= 7) < 0x80)
630 write_char( writer
, len
);
633 write_char( writer
, (len
& 0x7f) | 0x80 );
635 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
636 if ((len
>>= 7) < 0x08)
638 write_char( writer
, len
);
641 return WS_E_INVALID_FORMAT
;
644 static HRESULT
write_string( struct writer
*writer
, const BYTE
*bytes
, ULONG len
)
647 if ((hr
= write_int31( writer
, len
)) != S_OK
) return hr
;
648 if ((hr
= write_grow_buffer( writer
, len
)) != S_OK
) return hr
;
649 write_bytes( writer
, bytes
, len
);
653 static HRESULT
write_dict_string( struct writer
*writer
, ULONG id
)
655 if (id
> 0x7fffffff) return E_INVALIDARG
;
656 return write_int31( writer
, id
);
659 static enum record_type
get_attr_text_record_type( const WS_XML_TEXT
*text
, BOOL use_dict
)
661 if (!text
) return RECORD_CHARS8_TEXT
;
662 switch (text
->textType
)
664 case WS_XML_TEXT_TYPE_UTF8
:
666 const WS_XML_UTF8_TEXT
*text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
667 if (use_dict
) return RECORD_DICTIONARY_TEXT
;
668 if (text_utf8
->value
.length
<= MAX_UINT8
) return RECORD_CHARS8_TEXT
;
669 if (text_utf8
->value
.length
<= MAX_UINT16
) return RECORD_CHARS16_TEXT
;
670 return RECORD_CHARS32_TEXT
;
672 case WS_XML_TEXT_TYPE_UTF16
:
674 const WS_XML_UTF16_TEXT
*text_utf16
= (const WS_XML_UTF16_TEXT
*)text
;
675 int len
= text_utf16
->byteCount
/ sizeof(WCHAR
);
676 int len_utf8
= WideCharToMultiByte( CP_UTF8
, 0, (const WCHAR
*)text_utf16
->bytes
, len
, NULL
, 0, NULL
, NULL
);
677 if (len_utf8
<= MAX_UINT8
) return RECORD_CHARS8_TEXT
;
678 if (len_utf8
<= MAX_UINT16
) return RECORD_CHARS16_TEXT
;
679 return RECORD_CHARS32_TEXT
;
681 case WS_XML_TEXT_TYPE_BASE64
:
683 const WS_XML_BASE64_TEXT
*text_base64
= (const WS_XML_BASE64_TEXT
*)text
;
684 if (text_base64
->length
<= MAX_UINT8
) return RECORD_BYTES8_TEXT
;
685 if (text_base64
->length
<= MAX_UINT16
) return RECORD_BYTES16_TEXT
;
686 return RECORD_BYTES32_TEXT
;
688 case WS_XML_TEXT_TYPE_BOOL
:
690 const WS_XML_BOOL_TEXT
*text_bool
= (const WS_XML_BOOL_TEXT
*)text
;
691 return text_bool
->value
? RECORD_TRUE_TEXT
: RECORD_FALSE_TEXT
;
693 case WS_XML_TEXT_TYPE_INT32
:
695 const WS_XML_INT32_TEXT
*text_int32
= (const WS_XML_INT32_TEXT
*)text
;
696 if (!text_int32
->value
) return RECORD_ZERO_TEXT
;
697 if (text_int32
->value
== 1) return RECORD_ONE_TEXT
;
698 if (text_int32
->value
>= MIN_INT8
&& text_int32
->value
<= MAX_INT8
) return RECORD_INT8_TEXT
;
699 if (text_int32
->value
>= MIN_INT16
&& text_int32
->value
<= MAX_INT16
) return RECORD_INT16_TEXT
;
700 return RECORD_INT32_TEXT
;
702 case WS_XML_TEXT_TYPE_INT64
:
704 const WS_XML_INT64_TEXT
*text_int64
= (const WS_XML_INT64_TEXT
*)text
;
705 if (!text_int64
->value
) return RECORD_ZERO_TEXT
;
706 if (text_int64
->value
== 1) return RECORD_ONE_TEXT
;
707 if (text_int64
->value
>= MIN_INT8
&& text_int64
->value
<= MAX_INT8
) return RECORD_INT8_TEXT
;
708 if (text_int64
->value
>= MIN_INT16
&& text_int64
->value
<= MAX_INT16
) return RECORD_INT16_TEXT
;
709 if (text_int64
->value
>= MIN_INT32
&& text_int64
->value
<= MAX_INT32
) return RECORD_INT32_TEXT
;
710 return RECORD_INT64_TEXT
;
712 case WS_XML_TEXT_TYPE_UINT64
:
714 const WS_XML_UINT64_TEXT
*text_uint64
= (const WS_XML_UINT64_TEXT
*)text
;
715 if (!text_uint64
->value
) return RECORD_ZERO_TEXT
;
716 if (text_uint64
->value
== 1) return RECORD_ONE_TEXT
;
717 if (text_uint64
->value
<= MAX_INT8
) return RECORD_INT8_TEXT
;
718 if (text_uint64
->value
<= MAX_INT16
) return RECORD_INT16_TEXT
;
719 if (text_uint64
->value
<= MAX_INT32
) return RECORD_INT32_TEXT
;
720 if (text_uint64
->value
<= MAX_INT64
) return RECORD_INT64_TEXT
;
721 return RECORD_UINT64_TEXT
;
723 case WS_XML_TEXT_TYPE_DOUBLE
:
725 const WS_XML_DOUBLE_TEXT
*text_double
= (const WS_XML_DOUBLE_TEXT
*)text
;
726 if (!text_double
->value
) return RECORD_ZERO_TEXT
;
727 if (text_double
->value
== 1) return RECORD_ONE_TEXT
;
728 if (isinf( text_double
->value
) || (INT64
)text_double
->value
!= text_double
->value
)
729 return RECORD_DOUBLE_TEXT
;
730 if (text_double
->value
<= MAX_INT8
) return RECORD_INT8_TEXT
;
731 if (text_double
->value
<= MAX_INT16
) return RECORD_INT16_TEXT
;
732 if (text_double
->value
<= MAX_INT32
) return RECORD_INT32_TEXT
;
733 return RECORD_INT64_TEXT
;
735 case WS_XML_TEXT_TYPE_GUID
:
736 return RECORD_GUID_TEXT
;
738 case WS_XML_TEXT_TYPE_UNIQUE_ID
:
739 return RECORD_UNIQUE_ID_TEXT
;
741 case WS_XML_TEXT_TYPE_DATETIME
:
742 return RECORD_DATETIME_TEXT
;
745 FIXME( "unhandled text type %u\n", text
->textType
);
750 static INT64
get_text_value_int( const WS_XML_TEXT
*text
)
752 switch (text
->textType
)
754 case WS_XML_TEXT_TYPE_INT32
:
756 const WS_XML_INT32_TEXT
*text_int32
= (const WS_XML_INT32_TEXT
*)text
;
757 return text_int32
->value
;
759 case WS_XML_TEXT_TYPE_INT64
:
761 const WS_XML_INT64_TEXT
*text_int64
= (const WS_XML_INT64_TEXT
*)text
;
762 return text_int64
->value
;
764 case WS_XML_TEXT_TYPE_UINT64
:
766 const WS_XML_UINT64_TEXT
*text_uint64
= (const WS_XML_UINT64_TEXT
*)text
;
767 return text_uint64
->value
;
769 case WS_XML_TEXT_TYPE_DOUBLE
:
771 const WS_XML_DOUBLE_TEXT
*text_double
= (const WS_XML_DOUBLE_TEXT
*)text
;
772 return text_double
->value
;
775 ERR( "unhandled text type %u\n", text
->textType
);
781 static BOOL
get_string_id( struct writer
*writer
, const WS_XML_STRING
*str
, ULONG
*id
)
783 if (writer
->dict
&& str
->dictionary
== writer
->dict
)
791 writer
->dict_cb( writer
->dict_cb_state
, str
, &found
, id
, NULL
);
792 if (found
) *id
= (*id
<< 1) | 1;
798 static ULONG
format_bool( const BOOL
*ptr
, unsigned char *buf
)
800 static const unsigned char bool_true
[] = {'t','r','u','e'}, bool_false
[] = {'f','a','l','s','e'};
803 memcpy( buf
, bool_true
, sizeof(bool_true
) );
804 return sizeof(bool_true
);
806 memcpy( buf
, bool_false
, sizeof(bool_false
) );
807 return sizeof(bool_false
);
810 static ULONG
format_int32( const INT32
*ptr
, unsigned char *buf
)
812 return wsprintfA( (char *)buf
, "%d", *ptr
);
815 static ULONG
format_int64( const INT64
*ptr
, unsigned char *buf
)
817 return wsprintfA( (char *)buf
, "%I64d", *ptr
);
820 static ULONG
format_uint64( const UINT64
*ptr
, unsigned char *buf
)
822 return wsprintfA( (char *)buf
, "%I64u", *ptr
);
825 static ULONG
format_double( const double *ptr
, unsigned char *buf
)
828 static const long double precision
= 0.0000000000000001;
829 unsigned char *p
= buf
;
830 long double val
= *ptr
;
831 int neg
, mag
, mag2
, use_exp
;
835 memcpy( buf
, "NaN", 3 );
842 memcpy( buf
, "-INF", 4 );
845 memcpy( buf
, "INF", 3 );
861 use_exp
= (mag
>= 15 || (neg
&& mag
>= 1) || mag
<= -1);
864 if (mag
< 0) mag
-= 1;
865 val
= val
/ powl( 10.0, mag
);
869 else if (mag
< 1) mag
= 0;
871 while (val
> precision
|| mag
>= 0)
873 long double weight
= powl( 10.0, mag
);
874 if (weight
> 0 && !isinf( weight
))
876 int digit
= floorl( val
/ weight
);
877 val
-= digit
* weight
;
878 *(p
++) = '0' + digit
;
880 if (!mag
&& val
> precision
) *(p
++) = '.';
888 if (mag2
> 0) *(p
++) = '+';
897 *(p
++) = '0' + mag2
% 10;
901 for (i
= -mag
, j
= -1; i
< j
; i
++, j
--)
911 FIXME( "powl not found at build time\n" );
916 static inline int year_size( int year
)
918 return leap_year( year
) ? 366 : 365;
922 static ULONG
format_datetime( const WS_DATETIME
*ptr
, unsigned char *buf
)
924 static const char fmt
[] = "%04u-%02u-%02uT%02u:%02u:%02u";
925 int day
, hour
, min
, sec
, sec_frac
, month
= 0, year
= 1, tz_hour
;
926 unsigned __int64 ticks
, day_ticks
;
929 if (ptr
->format
== WS_DATETIME_FORMAT_LOCAL
&&
930 ptr
->ticks
>= TICKS_1601_01_01
+ TZ_OFFSET
* TICKS_PER_HOUR
)
932 ticks
= ptr
->ticks
- TZ_OFFSET
* TICKS_PER_HOUR
;
940 day
= ticks
/ TICKS_PER_DAY
;
941 day_ticks
= ticks
% TICKS_PER_DAY
;
942 hour
= day_ticks
/ TICKS_PER_HOUR
;
943 min
= (day_ticks
% TICKS_PER_HOUR
) / TICKS_PER_MIN
;
944 sec
= (day_ticks
% TICKS_PER_MIN
) / TICKS_PER_SEC
;
945 sec_frac
= day_ticks
% TICKS_PER_SEC
;
947 while (day
>= year_size( year
))
949 day
-= year_size( year
);
952 while (day
>= month_days
[leap_year( year
)][month
])
954 day
-= month_days
[leap_year( year
)][month
];
958 len
= sprintf( (char *)buf
, fmt
, year
, month
+ 1, day
+ 1, hour
, min
, sec
);
961 static const char fmt_frac
[] = ".%07u";
962 len
+= sprintf( (char *)buf
+ len
, fmt_frac
, sec_frac
);
963 while (buf
[len
- 1] == '0') len
--;
965 if (ptr
->format
== WS_DATETIME_FORMAT_UTC
)
969 else if (ptr
->format
== WS_DATETIME_FORMAT_LOCAL
)
971 static const char fmt_tz
[] = "%c%02u:00";
972 len
+= sprintf( (char *)buf
+ len
, fmt_tz
, tz_hour
? '-' : '+', tz_hour
);
978 static ULONG
format_guid( const GUID
*ptr
, unsigned char *buf
)
980 static const char fmt
[] = "%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x";
981 return sprintf( (char *)buf
, fmt
, ptr
->Data1
, ptr
->Data2
, ptr
->Data3
,
982 ptr
->Data4
[0], ptr
->Data4
[1], ptr
->Data4
[2], ptr
->Data4
[3],
983 ptr
->Data4
[4], ptr
->Data4
[5], ptr
->Data4
[6], ptr
->Data4
[7] );
986 static ULONG
format_urn( const GUID
*ptr
, unsigned char *buf
)
988 static const char fmt
[] = "urn:uuid:%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x";
989 return sprintf( (char *)buf
, fmt
, ptr
->Data1
, ptr
->Data2
, ptr
->Data3
,
990 ptr
->Data4
[0], ptr
->Data4
[1], ptr
->Data4
[2], ptr
->Data4
[3],
991 ptr
->Data4
[4], ptr
->Data4
[5], ptr
->Data4
[6], ptr
->Data4
[7] );
994 static ULONG
format_qname( const WS_XML_STRING
*prefix
, const WS_XML_STRING
*localname
, unsigned char *buf
)
997 if (prefix
&& prefix
->length
)
999 memcpy( buf
, prefix
->bytes
, prefix
->length
);
1000 len
+= prefix
->length
;
1003 memcpy( buf
+ len
, localname
->bytes
, localname
->length
);
1004 return len
+ localname
->length
;
1007 static ULONG
encode_base64( const unsigned char *bin
, ULONG len
, unsigned char *buf
)
1009 static const char base64
[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
1014 buf
[i
++] = base64
[(bin
[0] & 0xfc) >> 2];
1015 x
= (bin
[0] & 3) << 4;
1018 buf
[i
++] = base64
[x
];
1023 buf
[i
++] = base64
[x
| ((bin
[1] & 0xf0) >> 4)];
1024 x
= (bin
[1] & 0x0f) << 2;
1027 buf
[i
++] = base64
[x
];
1031 buf
[i
++] = base64
[x
| ((bin
[2] & 0xc0) >> 6)];
1032 buf
[i
++] = base64
[bin
[2] & 0x3f];
1039 HRESULT
text_to_utf8text( const WS_XML_TEXT
*text
, const WS_XML_UTF8_TEXT
*old
, ULONG
*offset
,
1040 WS_XML_UTF8_TEXT
**ret
)
1042 ULONG len_old
= old
? old
->value
.length
: 0;
1043 if (offset
) *offset
= len_old
;
1045 switch (text
->textType
)
1047 case WS_XML_TEXT_TYPE_UTF8
:
1049 const WS_XML_UTF8_TEXT
*src
= (const WS_XML_UTF8_TEXT
*)text
;
1051 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ src
->value
.length
))) return E_OUTOFMEMORY
;
1052 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1053 memcpy( (*ret
)->value
.bytes
+ len_old
, src
->value
.bytes
, src
->value
.length
);
1056 case WS_XML_TEXT_TYPE_UTF16
:
1058 const WS_XML_UTF16_TEXT
*src
= (const WS_XML_UTF16_TEXT
*)text
;
1059 const WCHAR
*str
= (const WCHAR
*)src
->bytes
;
1060 ULONG len
= src
->byteCount
/ sizeof(WCHAR
), len_utf8
;
1062 if (src
->byteCount
% sizeof(WCHAR
)) return E_INVALIDARG
;
1063 len_utf8
= WideCharToMultiByte( CP_UTF8
, 0, str
, len
, NULL
, 0, NULL
, NULL
);
1064 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len_utf8
))) return E_OUTOFMEMORY
;
1065 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1066 WideCharToMultiByte( CP_UTF8
, 0, str
, len
, (char *)(*ret
)->value
.bytes
+ len_old
, len_utf8
, NULL
, NULL
);
1069 case WS_XML_TEXT_TYPE_BASE64
:
1071 const WS_XML_BASE64_TEXT
*base64
= (const WS_XML_BASE64_TEXT
*)text
;
1072 ULONG len
= ((4 * base64
->length
/ 3) + 3) & ~3;
1074 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1075 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1076 (*ret
)->value
.length
= encode_base64( base64
->bytes
, base64
->length
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1079 case WS_XML_TEXT_TYPE_BOOL
:
1081 const WS_XML_BOOL_TEXT
*bool_text
= (const WS_XML_BOOL_TEXT
*)text
;
1083 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ 5 ))) return E_OUTOFMEMORY
;
1084 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1085 (*ret
)->value
.length
= format_bool( &bool_text
->value
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1088 case WS_XML_TEXT_TYPE_INT32
:
1090 const WS_XML_INT32_TEXT
*int32_text
= (const WS_XML_INT32_TEXT
*)text
;
1091 unsigned char buf
[12]; /* "-2147483648" */
1092 ULONG len
= format_int32( &int32_text
->value
, buf
);
1094 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1095 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1096 memcpy( (*ret
)->value
.bytes
+ len_old
, buf
, len
);
1099 case WS_XML_TEXT_TYPE_INT64
:
1101 const WS_XML_INT64_TEXT
*int64_text
= (const WS_XML_INT64_TEXT
*)text
;
1102 unsigned char buf
[21]; /* "-9223372036854775808" */
1103 ULONG len
= format_int64( &int64_text
->value
, buf
);
1105 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1106 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1107 memcpy( (*ret
)->value
.bytes
+ len_old
, buf
, len
);
1110 case WS_XML_TEXT_TYPE_UINT64
:
1112 const WS_XML_UINT64_TEXT
*uint64_text
= (const WS_XML_UINT64_TEXT
*)text
;
1113 unsigned char buf
[21]; /* "18446744073709551615" */
1114 ULONG len
= format_uint64( &uint64_text
->value
, buf
);
1116 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1117 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1118 memcpy( (*ret
)->value
.bytes
+ len_old
, buf
, len
);
1121 case WS_XML_TEXT_TYPE_DOUBLE
:
1123 const WS_XML_DOUBLE_TEXT
*double_text
= (const WS_XML_DOUBLE_TEXT
*)text
;
1124 unsigned char buf
[32]; /* "-1.1111111111111111E-308", oversized to address Valgrind limitations */
1125 unsigned short fpword
;
1128 if (!set_fpword( 0x37f, &fpword
)) return E_NOTIMPL
;
1129 len
= format_double( &double_text
->value
, buf
);
1130 restore_fpword( fpword
);
1131 if (!len
) return E_NOTIMPL
;
1133 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1134 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1135 memcpy( (*ret
)->value
.bytes
+ len_old
, buf
, len
);
1138 case WS_XML_TEXT_TYPE_GUID
:
1140 const WS_XML_GUID_TEXT
*id
= (const WS_XML_GUID_TEXT
*)text
;
1142 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ 37 ))) return E_OUTOFMEMORY
;
1143 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1144 (*ret
)->value
.length
= format_guid( &id
->value
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1147 case WS_XML_TEXT_TYPE_UNIQUE_ID
:
1149 const WS_XML_UNIQUE_ID_TEXT
*id
= (const WS_XML_UNIQUE_ID_TEXT
*)text
;
1151 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ 46 ))) return E_OUTOFMEMORY
;
1152 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1153 (*ret
)->value
.length
= format_urn( &id
->value
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1156 case WS_XML_TEXT_TYPE_DATETIME
:
1158 const WS_XML_DATETIME_TEXT
*dt
= (const WS_XML_DATETIME_TEXT
*)text
;
1160 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ 34 ))) return E_OUTOFMEMORY
;
1161 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1162 (*ret
)->value
.length
= format_datetime( &dt
->value
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1165 case WS_XML_TEXT_TYPE_QNAME
:
1167 const WS_XML_QNAME_TEXT
*qn
= (const WS_XML_QNAME_TEXT
*)text
;
1168 ULONG len
= qn
->localName
->length
;
1170 if (qn
->prefix
&& qn
->prefix
->length
) len
+= qn
->prefix
->length
+ 1;
1171 if (!(*ret
= alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
1172 if (old
) memcpy( (*ret
)->value
.bytes
, old
->value
.bytes
, len_old
);
1173 (*ret
)->value
.length
= format_qname( qn
->prefix
, qn
->localName
, (*ret
)->value
.bytes
+ len_old
) + len_old
;
1177 FIXME( "unhandled text type %u\n", text
->textType
);
1182 static HRESULT
write_attribute_value_bin( struct writer
*writer
, const WS_XML_TEXT
*text
)
1184 enum record_type type
;
1185 BOOL use_dict
= FALSE
;
1189 if (text
&& text
->textType
== WS_XML_TEXT_TYPE_UTF8
)
1191 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
;
1192 use_dict
= get_string_id( writer
, &utf8
->value
, &id
);
1194 type
= get_attr_text_record_type( text
, use_dict
);
1196 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1197 write_char( writer
, type
);
1201 case RECORD_CHARS8_TEXT
:
1203 const WS_XML_UTF8_TEXT
*text_utf8
;
1204 WS_XML_UTF8_TEXT
*new = NULL
;
1209 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1210 write_char( writer
, 0 );
1213 if (text
->textType
== WS_XML_TEXT_TYPE_UTF8
) text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
1216 if ((hr
= text_to_utf8text( text
, NULL
, NULL
, &new )) != S_OK
) return hr
;
1219 len
= text_utf8
->value
.length
;
1220 if ((hr
= write_grow_buffer( writer
, sizeof(len
) + len
)) != S_OK
)
1225 write_char( writer
, len
);
1226 write_bytes( writer
, text_utf8
->value
.bytes
, len
);
1230 case RECORD_CHARS16_TEXT
:
1232 const WS_XML_UTF8_TEXT
*text_utf8
;
1233 WS_XML_UTF8_TEXT
*new = NULL
;
1236 if (text
->textType
== WS_XML_TEXT_TYPE_UTF8
) text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
1239 if ((hr
= text_to_utf8text( text
, NULL
, NULL
, &new )) != S_OK
) return hr
;
1242 len
= text_utf8
->value
.length
;
1243 if ((hr
= write_grow_buffer( writer
, sizeof(len
) + len
)) != S_OK
)
1248 write_bytes( writer
, (const BYTE
*)&len
, sizeof(len
) );
1249 write_bytes( writer
, text_utf8
->value
.bytes
, len
);
1253 case RECORD_BYTES8_TEXT
:
1255 WS_XML_BASE64_TEXT
*text_base64
= (WS_XML_BASE64_TEXT
*)text
;
1256 if ((hr
= write_grow_buffer( writer
, 1 + text_base64
->length
)) != S_OK
) return hr
;
1257 write_char( writer
, text_base64
->length
);
1258 write_bytes( writer
, text_base64
->bytes
, text_base64
->length
);
1261 case RECORD_BYTES16_TEXT
:
1263 WS_XML_BASE64_TEXT
*text_base64
= (WS_XML_BASE64_TEXT
*)text
;
1264 UINT16 len
= text_base64
->length
;
1265 if ((hr
= write_grow_buffer( writer
, sizeof(len
) + len
)) != S_OK
) return hr
;
1266 write_bytes( writer
, (const BYTE
*)&len
, sizeof(len
) );
1267 write_bytes( writer
, text_base64
->bytes
, len
);
1270 case RECORD_ZERO_TEXT
:
1271 case RECORD_ONE_TEXT
:
1272 case RECORD_FALSE_TEXT
:
1273 case RECORD_TRUE_TEXT
:
1276 case RECORD_INT8_TEXT
:
1278 INT8 val
= get_text_value_int( text
);
1279 if ((hr
= write_grow_buffer( writer
, sizeof(val
) )) != S_OK
) return hr
;
1280 write_char( writer
, val
);
1283 case RECORD_INT16_TEXT
:
1285 INT16 val
= get_text_value_int( text
);
1286 if ((hr
= write_grow_buffer( writer
, sizeof(val
) )) != S_OK
) return hr
;
1287 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
1290 case RECORD_INT32_TEXT
:
1292 INT32 val
= get_text_value_int( text
);
1293 if ((hr
= write_grow_buffer( writer
, sizeof(val
) )) != S_OK
) return hr
;
1294 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
1297 case RECORD_INT64_TEXT
:
1299 INT64 val
= get_text_value_int( text
);
1300 if ((hr
= write_grow_buffer( writer
, sizeof(val
) )) != S_OK
) return hr
;
1301 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
1304 case RECORD_UINT64_TEXT
:
1306 WS_XML_UINT64_TEXT
*text_uint64
= (WS_XML_UINT64_TEXT
*)text
;
1307 if ((hr
= write_grow_buffer( writer
, sizeof(text_uint64
->value
) )) != S_OK
) return hr
;
1308 write_bytes( writer
, (const BYTE
*)&text_uint64
->value
, sizeof(text_uint64
->value
) );
1311 case RECORD_DOUBLE_TEXT
:
1313 WS_XML_DOUBLE_TEXT
*text_double
= (WS_XML_DOUBLE_TEXT
*)text
;
1314 if ((hr
= write_grow_buffer( writer
, sizeof(text_double
->value
) )) != S_OK
) return hr
;
1315 write_bytes( writer
, (const BYTE
*)&text_double
->value
, sizeof(text_double
->value
) );
1318 case RECORD_GUID_TEXT
:
1320 WS_XML_GUID_TEXT
*text_guid
= (WS_XML_GUID_TEXT
*)text
;
1321 if ((hr
= write_grow_buffer( writer
, sizeof(text_guid
->value
) )) != S_OK
) return hr
;
1322 write_bytes( writer
, (const BYTE
*)&text_guid
->value
, sizeof(text_guid
->value
) );
1325 case RECORD_UNIQUE_ID_TEXT
:
1327 WS_XML_UNIQUE_ID_TEXT
*text_unique_id
= (WS_XML_UNIQUE_ID_TEXT
*)text
;
1328 if ((hr
= write_grow_buffer( writer
, sizeof(text_unique_id
->value
) )) != S_OK
) return hr
;
1329 write_bytes( writer
, (const BYTE
*)&text_unique_id
->value
, sizeof(text_unique_id
->value
) );
1332 case RECORD_DATETIME_TEXT
:
1334 WS_XML_DATETIME_TEXT
*text_datetime
= (WS_XML_DATETIME_TEXT
*)text
;
1335 UINT64 val
= text_datetime
->value
.ticks
;
1337 assert( val
<= TICKS_MAX
);
1338 if (text_datetime
->value
.format
== WS_DATETIME_FORMAT_UTC
) val
|= (UINT64
)1 << 62;
1339 else if (text_datetime
->value
.format
== WS_DATETIME_FORMAT_LOCAL
) val
|= (UINT64
)1 << 63;
1341 if ((hr
= write_grow_buffer( writer
, sizeof(val
) )) != S_OK
) return hr
;
1342 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
1346 FIXME( "unhandled record type %02x\n", type
);
1351 static enum record_type
get_attr_record_type( const WS_XML_ATTRIBUTE
*attr
, BOOL use_dict
)
1353 if (!attr
->prefix
|| !attr
->prefix
->length
)
1355 if (use_dict
) return RECORD_SHORT_DICTIONARY_ATTRIBUTE
;
1356 return RECORD_SHORT_ATTRIBUTE
;
1358 if (attr
->prefix
->length
== 1 && attr
->prefix
->bytes
[0] >= 'a' && attr
->prefix
->bytes
[0] <= 'z')
1360 if (use_dict
) return RECORD_PREFIX_DICTIONARY_ATTRIBUTE_A
+ attr
->prefix
->bytes
[0] - 'a';
1361 return RECORD_PREFIX_ATTRIBUTE_A
+ attr
->prefix
->bytes
[0] - 'a';
1363 if (use_dict
) return RECORD_DICTIONARY_ATTRIBUTE
;
1364 return RECORD_ATTRIBUTE
;
1367 static HRESULT
write_attribute_bin( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
1370 enum record_type type
= get_attr_record_type( attr
, get_string_id(writer
, attr
->localName
, &id
) );
1373 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1374 write_char( writer
, type
);
1376 if (type
>= RECORD_PREFIX_ATTRIBUTE_A
&& type
<= RECORD_PREFIX_ATTRIBUTE_Z
)
1378 if ((hr
= write_string( writer
, attr
->localName
->bytes
, attr
->localName
->length
)) != S_OK
) return hr
;
1379 return write_attribute_value_bin( writer
, attr
->value
);
1381 if (type
>= RECORD_PREFIX_DICTIONARY_ATTRIBUTE_A
&& type
<= RECORD_PREFIX_DICTIONARY_ATTRIBUTE_Z
)
1383 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1384 return write_attribute_value_bin( writer
, attr
->value
);
1389 case RECORD_SHORT_ATTRIBUTE
:
1390 if ((hr
= write_string( writer
, attr
->localName
->bytes
, attr
->localName
->length
)) != S_OK
) return hr
;
1393 case RECORD_ATTRIBUTE
:
1394 if ((hr
= write_string( writer
, attr
->prefix
->bytes
, attr
->prefix
->length
)) != S_OK
) return hr
;
1395 if ((hr
= write_string( writer
, attr
->localName
->bytes
, attr
->localName
->length
)) != S_OK
) return hr
;
1398 case RECORD_SHORT_DICTIONARY_ATTRIBUTE
:
1399 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1402 case RECORD_DICTIONARY_ATTRIBUTE
:
1403 if ((hr
= write_string( writer
, attr
->prefix
->bytes
, attr
->prefix
->length
)) != S_OK
) return hr
;
1404 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1408 ERR( "unhandled record type %02x\n", type
);
1409 return WS_E_NOT_SUPPORTED
;
1412 return write_attribute_value_bin( writer
, attr
->value
);
1415 static HRESULT
write_attribute( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
1417 switch (writer
->output_enc
)
1419 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_attribute_text( writer
, attr
);
1420 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_attribute_bin( writer
, attr
);
1422 ERR( "unhandled encoding %u\n", writer
->output_enc
);
1423 return WS_E_NOT_SUPPORTED
;
1427 static inline BOOL
is_current_namespace( struct writer
*writer
, const WS_XML_STRING
*ns
)
1429 return (WsXmlStringEquals( writer
->current_ns
, ns
, NULL
) == S_OK
);
1432 /**************************************************************************
1433 * WsGetPrefixFromNamespace [webservices.@]
1435 HRESULT WINAPI
WsGetPrefixFromNamespace( WS_XML_WRITER
*handle
, const WS_XML_STRING
*ns
,
1436 BOOL required
, const WS_XML_STRING
**prefix
,
1439 struct writer
*writer
= (struct writer
*)handle
;
1440 WS_XML_ELEMENT_NODE
*elem
;
1444 TRACE( "%p %s %d %p %p\n", handle
, debugstr_xmlstr(ns
), required
, prefix
, error
);
1445 if (error
) FIXME( "ignoring error parameter\n" );
1447 if (!writer
|| !ns
|| !prefix
) return E_INVALIDARG
;
1449 EnterCriticalSection( &writer
->cs
);
1451 if (writer
->magic
!= WRITER_MAGIC
)
1453 LeaveCriticalSection( &writer
->cs
);
1454 return E_INVALIDARG
;
1457 elem
= &writer
->current
->hdr
;
1458 if (elem
->prefix
&& is_current_namespace( writer
, ns
))
1460 *prefix
= elem
->prefix
;
1466 if (required
) hr
= WS_E_INVALID_FORMAT
;
1474 LeaveCriticalSection( &writer
->cs
);
1478 static HRESULT
write_namespace_attribute_text( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
1480 unsigned char quote
= attr
->singleQuote
? '\'' : '"';
1484 /* ' xmlns:prefix="namespace"' */
1486 size
= attr
->ns
->length
+ 9 /* ' xmlns=""' */;
1487 if (attr
->prefix
&& attr
->prefix
->length
) size
+= attr
->prefix
->length
+ 1 /* ':' */;
1488 if ((hr
= write_grow_buffer( writer
, size
)) != S_OK
) return hr
;
1490 write_bytes( writer
, (const BYTE
*)" xmlns", 6 );
1491 if (attr
->prefix
&& attr
->prefix
->length
)
1493 write_char( writer
, ':' );
1494 write_bytes( writer
, attr
->prefix
->bytes
, attr
->prefix
->length
);
1496 write_char( writer
, '=' );
1497 write_char( writer
, quote
);
1498 write_bytes( writer
, attr
->ns
->bytes
, attr
->ns
->length
);
1499 write_char( writer
, quote
);
1504 static enum record_type
get_xmlns_record_type( const WS_XML_ATTRIBUTE
*attr
, BOOL use_dict
)
1506 if (!attr
->prefix
|| !attr
->prefix
->length
)
1508 if (use_dict
) return RECORD_SHORT_DICTIONARY_XMLNS_ATTRIBUTE
;
1509 return RECORD_SHORT_XMLNS_ATTRIBUTE
;
1511 if (use_dict
) return RECORD_DICTIONARY_XMLNS_ATTRIBUTE
;
1512 return RECORD_XMLNS_ATTRIBUTE
;
1515 static HRESULT
write_namespace_attribute_bin( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
1518 enum record_type type
= get_xmlns_record_type( attr
, get_string_id(writer
, attr
->ns
, &id
) );
1521 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1522 write_char( writer
, type
);
1526 case RECORD_SHORT_XMLNS_ATTRIBUTE
:
1529 case RECORD_XMLNS_ATTRIBUTE
:
1530 if ((hr
= write_string( writer
, attr
->prefix
->bytes
, attr
->prefix
->length
)) != S_OK
) return hr
;
1533 case RECORD_SHORT_DICTIONARY_XMLNS_ATTRIBUTE
:
1534 return write_dict_string( writer
, id
);
1536 case RECORD_DICTIONARY_XMLNS_ATTRIBUTE
:
1537 if ((hr
= write_string( writer
, attr
->prefix
->bytes
, attr
->prefix
->length
)) != S_OK
) return hr
;
1538 return write_dict_string( writer
, id
);
1541 ERR( "unhandled record type %02x\n", type
);
1542 return WS_E_NOT_SUPPORTED
;
1545 return write_string( writer
, attr
->ns
->bytes
, attr
->ns
->length
);
1548 static HRESULT
write_namespace_attribute( struct writer
*writer
, const WS_XML_ATTRIBUTE
*attr
)
1550 switch (writer
->output_enc
)
1552 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_namespace_attribute_text( writer
, attr
);
1553 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_namespace_attribute_bin( writer
, attr
);
1555 ERR( "unhandled encoding %u\n", writer
->output_enc
);
1556 return WS_E_NOT_SUPPORTED
;
1560 static HRESULT
add_namespace_attribute( struct writer
*writer
, const WS_XML_STRING
*prefix
,
1561 const WS_XML_STRING
*ns
, BOOL single
)
1563 WS_XML_ATTRIBUTE
*attr
;
1564 WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
1567 if (!(attr
= heap_alloc_zero( sizeof(*attr
) ))) return E_OUTOFMEMORY
;
1569 attr
->singleQuote
= !!single
;
1571 if (prefix
&& !(attr
->prefix
= dup_xml_string( prefix
, writer
->dict_do_lookup
)))
1573 free_attribute( attr
);
1574 return E_OUTOFMEMORY
;
1576 if (!(attr
->ns
= dup_xml_string( ns
, writer
->dict_do_lookup
)))
1578 free_attribute( attr
);
1579 return E_OUTOFMEMORY
;
1581 if ((hr
= append_attribute( elem
, attr
)) != S_OK
)
1583 free_attribute( attr
);
1589 static inline BOOL
str_equal( const WS_XML_STRING
*str1
, const WS_XML_STRING
*str2
)
1591 if (!str1
&& !str2
) return TRUE
;
1592 return WsXmlStringEquals( str1
, str2
, NULL
) == S_OK
;
1595 static BOOL
namespace_in_scope( const WS_XML_ELEMENT_NODE
*elem
, const WS_XML_STRING
*prefix
,
1596 const WS_XML_STRING
*ns
)
1599 const struct node
*node
;
1601 for (node
= (const struct node
*)elem
; node
; node
= node
->parent
)
1603 if (node_type( node
) != WS_XML_NODE_TYPE_ELEMENT
) break;
1606 for (i
= 0; i
< elem
->attributeCount
; i
++)
1608 if (!elem
->attributes
[i
]->isXmlNs
) continue;
1609 if (str_equal( elem
->attributes
[i
]->prefix
, prefix
) &&
1610 str_equal( elem
->attributes
[i
]->ns
, ns
)) return TRUE
;
1616 static HRESULT
set_current_namespace( struct writer
*writer
, const WS_XML_STRING
*ns
)
1619 if (!(str
= dup_xml_string( ns
, writer
->dict_do_lookup
))) return E_OUTOFMEMORY
;
1620 free_xml_string( writer
->current_ns
);
1621 writer
->current_ns
= str
;
1625 static HRESULT
set_namespaces( struct writer
*writer
)
1627 WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
1631 if (elem
->ns
->length
&& !namespace_in_scope( elem
, elem
->prefix
, elem
->ns
))
1633 if ((hr
= add_namespace_attribute( writer
, elem
->prefix
, elem
->ns
, FALSE
)) != S_OK
) return hr
;
1634 if ((hr
= set_current_namespace( writer
, elem
->ns
)) != S_OK
) return hr
;
1637 for (i
= 0; i
< elem
->attributeCount
; i
++)
1639 const WS_XML_ATTRIBUTE
*attr
= elem
->attributes
[i
];
1640 if (!attr
->ns
->length
|| namespace_in_scope( elem
, attr
->prefix
, attr
->ns
)) continue;
1641 if ((hr
= add_namespace_attribute( writer
, attr
->prefix
, attr
->ns
, FALSE
)) != S_OK
) return hr
;
1647 /**************************************************************************
1648 * WsWriteEndAttribute [webservices.@]
1650 HRESULT WINAPI
WsWriteEndAttribute( WS_XML_WRITER
*handle
, WS_ERROR
*error
)
1652 struct writer
*writer
= (struct writer
*)handle
;
1654 TRACE( "%p %p\n", handle
, error
);
1655 if (error
) FIXME( "ignoring error parameter\n" );
1657 if (!writer
) return E_INVALIDARG
;
1659 EnterCriticalSection( &writer
->cs
);
1661 if (writer
->magic
!= WRITER_MAGIC
)
1663 LeaveCriticalSection( &writer
->cs
);
1664 return E_INVALIDARG
;
1667 writer
->state
= WRITER_STATE_STARTELEMENT
;
1669 LeaveCriticalSection( &writer
->cs
);
1673 static HRESULT
write_attributes( struct writer
*writer
, const WS_XML_ELEMENT_NODE
*elem
)
1677 for (i
= 0; i
< elem
->attributeCount
; i
++)
1679 if (elem
->attributes
[i
]->isXmlNs
) continue;
1680 if ((hr
= write_attribute( writer
, elem
->attributes
[i
] )) != S_OK
) return hr
;
1682 for (i
= 0; i
< elem
->attributeCount
; i
++)
1684 if (!elem
->attributes
[i
]->isXmlNs
|| !elem
->attributes
[i
]->prefix
) continue;
1685 if ((hr
= write_namespace_attribute( writer
, elem
->attributes
[i
] )) != S_OK
) return hr
;
1687 for (i
= 0; i
< elem
->attributeCount
; i
++)
1689 if (!elem
->attributes
[i
]->isXmlNs
|| elem
->attributes
[i
]->prefix
) continue;
1690 if ((hr
= write_namespace_attribute( writer
, elem
->attributes
[i
] )) != S_OK
) return hr
;
1695 static HRESULT
write_startelement_text( struct writer
*writer
)
1697 const WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
1701 /* '<prefix:localname prefix:attr="value"... xmlns:prefix="ns"'... */
1703 size
= elem
->localName
->length
+ 1 /* '<' */;
1704 if (elem
->prefix
&& elem
->prefix
->length
) size
+= elem
->prefix
->length
+ 1 /* ':' */;
1705 if ((hr
= write_grow_buffer( writer
, size
)) != S_OK
) return hr
;
1707 write_char( writer
, '<' );
1708 if (elem
->prefix
&& elem
->prefix
->length
)
1710 write_bytes( writer
, elem
->prefix
->bytes
, elem
->prefix
->length
);
1711 write_char( writer
, ':' );
1713 write_bytes( writer
, elem
->localName
->bytes
, elem
->localName
->length
);
1714 return write_attributes( writer
, elem
);
1717 static enum record_type
get_elem_record_type( const WS_XML_ELEMENT_NODE
*elem
, BOOL use_dict
)
1719 if (!elem
->prefix
|| !elem
->prefix
->length
)
1721 if (use_dict
) return RECORD_SHORT_DICTIONARY_ELEMENT
;
1722 return RECORD_SHORT_ELEMENT
;
1724 if (elem
->prefix
->length
== 1 && elem
->prefix
->bytes
[0] >= 'a' && elem
->prefix
->bytes
[0] <= 'z')
1726 if (use_dict
) return RECORD_PREFIX_DICTIONARY_ELEMENT_A
+ elem
->prefix
->bytes
[0] - 'a';
1727 return RECORD_PREFIX_ELEMENT_A
+ elem
->prefix
->bytes
[0] - 'a';
1729 if (use_dict
) return RECORD_DICTIONARY_ELEMENT
;
1730 return RECORD_ELEMENT
;
1733 static HRESULT
write_startelement_bin( struct writer
*writer
)
1735 const WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
1737 enum record_type type
= get_elem_record_type( elem
, get_string_id(writer
, elem
->localName
, &id
) );
1740 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1741 write_char( writer
, type
);
1743 if (type
>= RECORD_PREFIX_ELEMENT_A
&& type
<= RECORD_PREFIX_ELEMENT_Z
)
1745 if ((hr
= write_string( writer
, elem
->localName
->bytes
, elem
->localName
->length
)) != S_OK
) return hr
;
1746 return write_attributes( writer
, elem
);
1748 if (type
>= RECORD_PREFIX_DICTIONARY_ELEMENT_A
&& type
<= RECORD_PREFIX_DICTIONARY_ELEMENT_Z
)
1750 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1751 return write_attributes( writer
, elem
);
1756 case RECORD_SHORT_ELEMENT
:
1757 if ((hr
= write_string( writer
, elem
->localName
->bytes
, elem
->localName
->length
)) != S_OK
) return hr
;
1760 case RECORD_ELEMENT
:
1761 if ((hr
= write_string( writer
, elem
->prefix
->bytes
, elem
->prefix
->length
)) != S_OK
) return hr
;
1762 if ((hr
= write_string( writer
, elem
->localName
->bytes
, elem
->localName
->length
)) != S_OK
) return hr
;
1765 case RECORD_SHORT_DICTIONARY_ELEMENT
:
1766 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1769 case RECORD_DICTIONARY_ELEMENT
:
1770 if ((hr
= write_string( writer
, elem
->prefix
->bytes
, elem
->prefix
->length
)) != S_OK
) return hr
;
1771 if ((hr
= write_dict_string( writer
, id
)) != S_OK
) return hr
;
1775 ERR( "unhandled record type %02x\n", type
);
1776 return WS_E_NOT_SUPPORTED
;
1779 return write_attributes( writer
, elem
);
1782 static HRESULT
write_startelement( struct writer
*writer
)
1784 switch (writer
->output_enc
)
1786 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_startelement_text( writer
);
1787 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_startelement_bin( writer
);
1789 ERR( "unhandled encoding %u\n", writer
->output_enc
);
1790 return WS_E_NOT_SUPPORTED
;
1794 static struct node
*write_find_startelement( struct writer
*writer
)
1797 for (node
= writer
->current
; node
; node
= node
->parent
)
1799 if (node_type( node
) == WS_XML_NODE_TYPE_ELEMENT
) return node
;
1804 static inline BOOL
is_empty_element( const struct node
*node
)
1806 const struct node
*head
= LIST_ENTRY( list_head( &node
->children
), struct node
, entry
);
1807 return node_type( head
) == WS_XML_NODE_TYPE_END_ELEMENT
;
1810 static HRESULT
write_endelement_text( struct writer
*writer
, const WS_XML_ELEMENT_NODE
*elem
)
1817 if (elem
->isEmpty
&& writer
->state
!= WRITER_STATE_ENDSTARTELEMENT
)
1819 if ((hr
= write_grow_buffer( writer
, 2 )) != S_OK
) return hr
;
1820 write_char( writer
, '/' );
1821 write_char( writer
, '>' );
1825 /* '</prefix:localname>' */
1827 size
= elem
->localName
->length
+ 3 /* '</>' */;
1828 if (elem
->prefix
&& elem
->prefix
->length
) size
+= elem
->prefix
->length
+ 1 /* ':' */;
1829 if ((hr
= write_grow_buffer( writer
, size
)) != S_OK
) return hr
;
1831 write_char( writer
, '<' );
1832 write_char( writer
, '/' );
1833 if (elem
->prefix
&& elem
->prefix
->length
)
1835 write_bytes( writer
, elem
->prefix
->bytes
, elem
->prefix
->length
);
1836 write_char( writer
, ':' );
1838 write_bytes( writer
, elem
->localName
->bytes
, elem
->localName
->length
);
1839 write_char( writer
, '>' );
1843 static HRESULT
write_endelement_bin( struct writer
*writer
)
1846 if (node_type( writer
->current
) == WS_XML_NODE_TYPE_TEXT
) return S_OK
;
1847 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1848 write_char( writer
, RECORD_ENDELEMENT
);
1852 static HRESULT
write_endelement( struct writer
*writer
, const WS_XML_ELEMENT_NODE
*elem
)
1854 switch (writer
->output_enc
)
1856 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_endelement_text( writer
, elem
);
1857 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_endelement_bin( writer
);
1859 ERR( "unhandled encoding %u\n", writer
->output_enc
);
1860 return WS_E_NOT_SUPPORTED
;
1864 static HRESULT
write_close_element( struct writer
*writer
, struct node
*node
)
1866 WS_XML_ELEMENT_NODE
*elem
= &node
->hdr
;
1867 elem
->isEmpty
= is_empty_element( node
);
1868 return write_endelement( writer
, elem
);
1871 static HRESULT
write_endelement_node( struct writer
*writer
)
1876 if (!(node
= write_find_startelement( writer
))) return WS_E_INVALID_FORMAT
;
1877 if (writer
->state
== WRITER_STATE_STARTELEMENT
)
1879 if ((hr
= set_namespaces( writer
)) != S_OK
) return hr
;
1880 if ((hr
= write_startelement( writer
)) != S_OK
) return hr
;
1882 if ((hr
= write_close_element( writer
, node
)) != S_OK
) return hr
;
1883 writer
->current
= node
->parent
;
1884 writer
->state
= WRITER_STATE_ENDELEMENT
;
1888 /**************************************************************************
1889 * WsWriteEndElement [webservices.@]
1891 HRESULT WINAPI
WsWriteEndElement( WS_XML_WRITER
*handle
, WS_ERROR
*error
)
1893 struct writer
*writer
= (struct writer
*)handle
;
1896 TRACE( "%p %p\n", handle
, error
);
1897 if (error
) FIXME( "ignoring error parameter\n" );
1899 if (!writer
) return E_INVALIDARG
;
1901 EnterCriticalSection( &writer
->cs
);
1903 if (writer
->magic
!= WRITER_MAGIC
)
1905 LeaveCriticalSection( &writer
->cs
);
1906 return E_INVALIDARG
;
1909 hr
= write_endelement_node( writer
);
1911 LeaveCriticalSection( &writer
->cs
);
1915 static HRESULT
write_endstartelement_text( struct writer
*writer
)
1918 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
1919 write_char( writer
, '>' );
1923 static HRESULT
write_endstartelement( struct writer
*writer
)
1925 switch (writer
->output_enc
)
1927 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_endstartelement_text( writer
);
1928 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return S_OK
;
1930 ERR( "unhandled encoding %u\n", writer
->output_enc
);
1931 return WS_E_NOT_SUPPORTED
;
1935 /**************************************************************************
1936 * WsWriteEndStartElement [webservices.@]
1938 HRESULT WINAPI
WsWriteEndStartElement( WS_XML_WRITER
*handle
, WS_ERROR
*error
)
1940 struct writer
*writer
= (struct writer
*)handle
;
1943 TRACE( "%p %p\n", handle
, error
);
1944 if (error
) FIXME( "ignoring error parameter\n" );
1946 if (!writer
) return E_INVALIDARG
;
1948 EnterCriticalSection( &writer
->cs
);
1950 if (writer
->magic
!= WRITER_MAGIC
)
1952 LeaveCriticalSection( &writer
->cs
);
1953 return E_INVALIDARG
;
1956 if (writer
->state
!= WRITER_STATE_STARTELEMENT
)
1958 LeaveCriticalSection( &writer
->cs
);
1959 return WS_E_INVALID_OPERATION
;
1962 if ((hr
= set_namespaces( writer
)) != S_OK
) goto done
;
1963 if ((hr
= write_startelement( writer
)) != S_OK
) goto done
;
1964 if ((hr
= write_endstartelement( writer
)) != S_OK
) goto done
;
1965 writer
->state
= WRITER_STATE_ENDSTARTELEMENT
;
1968 LeaveCriticalSection( &writer
->cs
);
1972 static HRESULT
write_add_attribute( struct writer
*writer
, const WS_XML_STRING
*prefix
,
1973 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
1976 WS_XML_ATTRIBUTE
*attr
;
1977 WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
1980 if (!(attr
= heap_alloc_zero( sizeof(*attr
) ))) return E_OUTOFMEMORY
;
1982 if (!prefix
&& ns
->length
) prefix
= elem
->prefix
;
1984 attr
->singleQuote
= !!single
;
1985 if (prefix
&& !(attr
->prefix
= dup_xml_string( prefix
, writer
->dict_do_lookup
)))
1987 free_attribute( attr
);
1988 return E_OUTOFMEMORY
;
1990 if (!(attr
->localName
= dup_xml_string( localname
, writer
->dict_do_lookup
)))
1992 free_attribute( attr
);
1993 return E_OUTOFMEMORY
;
1995 if (!(attr
->ns
= dup_xml_string( ns
, writer
->dict_do_lookup
)))
1997 free_attribute( attr
);
1998 return E_OUTOFMEMORY
;
2000 if ((hr
= append_attribute( elem
, attr
)) != S_OK
)
2002 free_attribute( attr
);
2008 /**************************************************************************
2009 * WsWriteStartAttribute [webservices.@]
2011 HRESULT WINAPI
WsWriteStartAttribute( WS_XML_WRITER
*handle
, const WS_XML_STRING
*prefix
,
2012 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
2013 BOOL single
, WS_ERROR
*error
)
2015 struct writer
*writer
= (struct writer
*)handle
;
2018 TRACE( "%p %s %s %s %d %p\n", handle
, debugstr_xmlstr(prefix
), debugstr_xmlstr(localname
),
2019 debugstr_xmlstr(ns
), single
, error
);
2020 if (error
) FIXME( "ignoring error parameter\n" );
2022 if (!writer
|| !localname
|| !ns
) return E_INVALIDARG
;
2024 EnterCriticalSection( &writer
->cs
);
2026 if (writer
->magic
!= WRITER_MAGIC
)
2028 LeaveCriticalSection( &writer
->cs
);
2029 return E_INVALIDARG
;
2032 if (writer
->state
!= WRITER_STATE_STARTELEMENT
)
2034 LeaveCriticalSection( &writer
->cs
);
2035 return WS_E_INVALID_OPERATION
;
2038 if ((hr
= write_add_attribute( writer
, prefix
, localname
, ns
, single
)) == S_OK
)
2039 writer
->state
= WRITER_STATE_STARTATTRIBUTE
;
2041 LeaveCriticalSection( &writer
->cs
);
2045 /* flush current start element if necessary */
2046 static HRESULT
write_flush( struct writer
*writer
)
2048 if (writer
->state
== WRITER_STATE_STARTELEMENT
)
2051 if ((hr
= set_namespaces( writer
)) != S_OK
) return hr
;
2052 if ((hr
= write_startelement( writer
)) != S_OK
) return hr
;
2053 if ((hr
= write_endstartelement( writer
)) != S_OK
) return hr
;
2054 writer
->state
= WRITER_STATE_ENDSTARTELEMENT
;
2059 static HRESULT
write_add_cdata_node( struct writer
*writer
)
2061 struct node
*node
, *parent
;
2062 if (!(parent
= find_parent( writer
))) return WS_E_INVALID_FORMAT
;
2063 if (!(node
= alloc_node( WS_XML_NODE_TYPE_CDATA
))) return E_OUTOFMEMORY
;
2064 write_insert_node( writer
, parent
, node
);
2068 static HRESULT
write_add_endcdata_node( struct writer
*writer
)
2071 if (!(node
= alloc_node( WS_XML_NODE_TYPE_END_CDATA
))) return E_OUTOFMEMORY
;
2072 node
->parent
= writer
->current
;
2073 list_add_tail( &node
->parent
->children
, &node
->entry
);
2077 static HRESULT
write_cdata( struct writer
*writer
)
2080 if ((hr
= write_grow_buffer( writer
, 9 )) != S_OK
) return hr
;
2081 write_bytes( writer
, (const BYTE
*)"<![CDATA[", 9 );
2085 static HRESULT
write_cdata_node( struct writer
*writer
)
2088 if ((hr
= write_flush( writer
)) != S_OK
) return hr
;
2089 if ((hr
= write_add_cdata_node( writer
)) != S_OK
) return hr
;
2090 if ((hr
= write_add_endcdata_node( writer
)) != S_OK
) return hr
;
2091 if ((hr
= write_cdata( writer
)) != S_OK
) return hr
;
2092 writer
->state
= WRITER_STATE_STARTCDATA
;
2096 /**************************************************************************
2097 * WsWriteStartCData [webservices.@]
2099 HRESULT WINAPI
WsWriteStartCData( WS_XML_WRITER
*handle
, WS_ERROR
*error
)
2101 struct writer
*writer
= (struct writer
*)handle
;
2104 TRACE( "%p %p\n", handle
, error
);
2105 if (error
) FIXME( "ignoring error parameter\n" );
2107 if (!writer
) return E_INVALIDARG
;
2109 EnterCriticalSection( &writer
->cs
);
2111 if (writer
->magic
!= WRITER_MAGIC
)
2113 LeaveCriticalSection( &writer
->cs
);
2114 return E_INVALIDARG
;
2117 hr
= write_cdata_node( writer
);
2119 LeaveCriticalSection( &writer
->cs
);
2123 static HRESULT
write_endcdata( struct writer
*writer
)
2126 if ((hr
= write_grow_buffer( writer
, 3 )) != S_OK
) return hr
;
2127 write_bytes( writer
, (const BYTE
*)"]]>", 3 );
2131 static HRESULT
write_endcdata_node( struct writer
*writer
)
2134 if ((hr
= write_endcdata( writer
)) != S_OK
) return hr
;
2135 writer
->current
= writer
->current
->parent
;
2136 writer
->state
= WRITER_STATE_ENDCDATA
;
2140 /**************************************************************************
2141 * WsWriteEndCData [webservices.@]
2143 HRESULT WINAPI
WsWriteEndCData( WS_XML_WRITER
*handle
, WS_ERROR
*error
)
2145 struct writer
*writer
= (struct writer
*)handle
;
2148 TRACE( "%p %p\n", handle
, error
);
2149 if (error
) FIXME( "ignoring error parameter\n" );
2151 if (!writer
) return E_INVALIDARG
;
2153 EnterCriticalSection( &writer
->cs
);
2155 if (writer
->magic
!= WRITER_MAGIC
)
2157 LeaveCriticalSection( &writer
->cs
);
2158 return E_INVALIDARG
;
2161 if (writer
->state
!= WRITER_STATE_TEXT
)
2163 LeaveCriticalSection( &writer
->cs
);
2164 return WS_E_INVALID_OPERATION
;
2167 hr
= write_endcdata_node( writer
);
2169 LeaveCriticalSection( &writer
->cs
);
2173 static HRESULT
write_add_element_node( struct writer
*writer
, const WS_XML_STRING
*prefix
,
2174 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
)
2176 struct node
*node
, *parent
;
2177 WS_XML_ELEMENT_NODE
*elem
;
2179 if (!(parent
= find_parent( writer
))) return WS_E_INVALID_FORMAT
;
2181 if (!prefix
&& node_type( parent
) == WS_XML_NODE_TYPE_ELEMENT
)
2183 elem
= &parent
->hdr
;
2184 if (WsXmlStringEquals( ns
, elem
->ns
, NULL
) == S_OK
) prefix
= elem
->prefix
;
2187 if (!(node
= alloc_node( WS_XML_NODE_TYPE_ELEMENT
))) return E_OUTOFMEMORY
;
2190 if (prefix
&& !(elem
->prefix
= dup_xml_string( prefix
, writer
->dict_do_lookup
)))
2193 return E_OUTOFMEMORY
;
2195 if (!(elem
->localName
= dup_xml_string( localname
, writer
->dict_do_lookup
)))
2198 return E_OUTOFMEMORY
;
2200 if (!(elem
->ns
= dup_xml_string( ns
, writer
->dict_do_lookup
)))
2203 return E_OUTOFMEMORY
;
2205 write_insert_node( writer
, parent
, node
);
2209 static HRESULT
write_add_endelement_node( struct writer
*writer
, struct node
*parent
)
2212 if (!(node
= alloc_node( WS_XML_NODE_TYPE_END_ELEMENT
))) return E_OUTOFMEMORY
;
2213 node
->parent
= parent
;
2214 list_add_tail( &parent
->children
, &node
->entry
);
2218 static HRESULT
write_element_node( struct writer
*writer
, const WS_XML_STRING
*prefix
,
2219 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
)
2222 if ((hr
= write_flush( writer
)) != S_OK
) return hr
;
2223 if ((hr
= write_add_element_node( writer
, prefix
, localname
, ns
)) != S_OK
) return hr
;
2224 if ((hr
= write_add_endelement_node( writer
, writer
->current
)) != S_OK
) return hr
;
2225 writer
->state
= WRITER_STATE_STARTELEMENT
;
2229 /**************************************************************************
2230 * WsWriteStartElement [webservices.@]
2232 HRESULT WINAPI
WsWriteStartElement( WS_XML_WRITER
*handle
, const WS_XML_STRING
*prefix
,
2233 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
2236 struct writer
*writer
= (struct writer
*)handle
;
2239 TRACE( "%p %s %s %s %p\n", handle
, debugstr_xmlstr(prefix
), debugstr_xmlstr(localname
),
2240 debugstr_xmlstr(ns
), error
);
2241 if (error
) FIXME( "ignoring error parameter\n" );
2243 if (!writer
|| !localname
|| !ns
) return E_INVALIDARG
;
2245 EnterCriticalSection( &writer
->cs
);
2247 if (writer
->magic
!= WRITER_MAGIC
)
2249 LeaveCriticalSection( &writer
->cs
);
2250 return E_INVALIDARG
;
2253 hr
= write_element_node( writer
, prefix
, localname
, ns
);
2255 LeaveCriticalSection( &writer
->cs
);
2259 HRESULT
text_to_text( const WS_XML_TEXT
*text
, const WS_XML_TEXT
*old
, ULONG
*offset
, WS_XML_TEXT
**ret
)
2261 if (offset
) *offset
= 0;
2262 switch (text
->textType
)
2264 case WS_XML_TEXT_TYPE_UTF8
:
2266 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2267 const WS_XML_UTF8_TEXT
*utf8_old
= (const WS_XML_UTF8_TEXT
*)old
;
2268 WS_XML_UTF8_TEXT
*new;
2269 ULONG len
= utf8
->value
.length
, len_old
= utf8_old
? utf8_old
->value
.length
: 0;
2271 if (!(new = alloc_utf8_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
2272 if (utf8_old
) memcpy( new->value
.bytes
, utf8_old
->value
.bytes
, len_old
);
2273 memcpy( new->value
.bytes
+ len_old
, utf8
->value
.bytes
, len
);
2274 if (offset
) *offset
= len_old
;
2278 case WS_XML_TEXT_TYPE_UTF16
:
2280 const WS_XML_UTF16_TEXT
*utf16
= (const WS_XML_UTF16_TEXT
*)text
;
2281 const WS_XML_UTF16_TEXT
*utf16_old
= (const WS_XML_UTF16_TEXT
*)old
;
2282 WS_XML_UTF16_TEXT
*new;
2283 ULONG len
= utf16
->byteCount
, len_old
= utf16_old
? utf16_old
->byteCount
: 0;
2285 if (utf16
->byteCount
% sizeof(WCHAR
)) return E_INVALIDARG
;
2286 if (!(new = alloc_utf16_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
2287 if (utf16_old
) memcpy( new->bytes
, utf16_old
->bytes
, len_old
);
2288 memcpy( new->bytes
+ len_old
, utf16
->bytes
, len
);
2289 if (offset
) *offset
= len_old
;
2293 case WS_XML_TEXT_TYPE_BASE64
:
2295 const WS_XML_BASE64_TEXT
*base64
= (const WS_XML_BASE64_TEXT
*)text
;
2296 const WS_XML_BASE64_TEXT
*base64_old
= (const WS_XML_BASE64_TEXT
*)old
;
2297 WS_XML_BASE64_TEXT
*new;
2298 ULONG len
= base64
->length
, len_old
= base64_old
? base64_old
->length
: 0;
2300 if (!(new = alloc_base64_text( NULL
, len_old
+ len
))) return E_OUTOFMEMORY
;
2301 if (base64_old
) memcpy( new->bytes
, base64_old
->bytes
, len_old
);
2302 memcpy( new->bytes
+ len_old
, base64
->bytes
, len
);
2303 if (offset
) *offset
= len_old
;
2307 case WS_XML_TEXT_TYPE_BOOL
:
2309 const WS_XML_BOOL_TEXT
*bool_text
= (const WS_XML_BOOL_TEXT
*)text
;
2310 WS_XML_BOOL_TEXT
*new;
2312 if (!(new = alloc_bool_text( bool_text
->value
))) return E_OUTOFMEMORY
;
2316 case WS_XML_TEXT_TYPE_INT32
:
2318 const WS_XML_INT32_TEXT
*int32_text
= (const WS_XML_INT32_TEXT
*)text
;
2319 WS_XML_INT32_TEXT
*new;
2321 if (!(new = alloc_int32_text( int32_text
->value
))) return E_OUTOFMEMORY
;
2325 case WS_XML_TEXT_TYPE_INT64
:
2327 const WS_XML_INT64_TEXT
*int64_text
= (const WS_XML_INT64_TEXT
*)text
;
2328 WS_XML_INT64_TEXT
*new;
2330 if (!(new = alloc_int64_text( int64_text
->value
))) return E_OUTOFMEMORY
;
2334 case WS_XML_TEXT_TYPE_UINT64
:
2336 const WS_XML_UINT64_TEXT
*uint64_text
= (const WS_XML_UINT64_TEXT
*)text
;
2337 WS_XML_UINT64_TEXT
*new;
2339 if (!(new = alloc_uint64_text( uint64_text
->value
))) return E_OUTOFMEMORY
;
2343 case WS_XML_TEXT_TYPE_DOUBLE
:
2345 const WS_XML_DOUBLE_TEXT
*double_text
= (const WS_XML_DOUBLE_TEXT
*)text
;
2346 WS_XML_DOUBLE_TEXT
*new;
2348 if (!(new = alloc_double_text( double_text
->value
))) return E_OUTOFMEMORY
;
2352 case WS_XML_TEXT_TYPE_GUID
:
2354 const WS_XML_GUID_TEXT
*id
= (const WS_XML_GUID_TEXT
*)text
;
2355 WS_XML_GUID_TEXT
*new;
2357 if (!(new = alloc_guid_text( &id
->value
))) return E_OUTOFMEMORY
;
2361 case WS_XML_TEXT_TYPE_UNIQUE_ID
:
2363 const WS_XML_UNIQUE_ID_TEXT
*id
= (const WS_XML_UNIQUE_ID_TEXT
*)text
;
2364 WS_XML_UNIQUE_ID_TEXT
*new;
2366 if (!(new = alloc_unique_id_text( &id
->value
))) return E_OUTOFMEMORY
;
2370 case WS_XML_TEXT_TYPE_DATETIME
:
2372 const WS_XML_DATETIME_TEXT
*dt
= (const WS_XML_DATETIME_TEXT
*)text
;
2373 WS_XML_DATETIME_TEXT
*new;
2375 if (!(new = alloc_datetime_text( &dt
->value
))) return E_OUTOFMEMORY
;
2380 FIXME( "unhandled text type %u\n", text
->textType
);
2385 static HRESULT
write_set_attribute_value( struct writer
*writer
, const WS_XML_TEXT
*value
)
2387 WS_XML_ELEMENT_NODE
*elem
= &writer
->current
->hdr
;
2390 switch (value
->textType
)
2392 case WS_XML_TEXT_TYPE_UTF8
:
2393 case WS_XML_TEXT_TYPE_UTF16
:
2394 case WS_XML_TEXT_TYPE_BASE64
:
2397 case WS_XML_TEXT_TYPE_BOOL
:
2398 case WS_XML_TEXT_TYPE_INT32
:
2399 case WS_XML_TEXT_TYPE_INT64
:
2400 case WS_XML_TEXT_TYPE_UINT64
:
2401 case WS_XML_TEXT_TYPE_DOUBLE
:
2402 case WS_XML_TEXT_TYPE_GUID
:
2403 case WS_XML_TEXT_TYPE_UNIQUE_ID
:
2404 case WS_XML_TEXT_TYPE_DATETIME
:
2405 if (elem
->attributes
[elem
->attributeCount
- 1]->value
) return WS_E_INVALID_OPERATION
;
2409 FIXME( "unhandled text type %u\n", value
->textType
);
2413 switch (writer
->output_enc
)
2415 case WS_XML_WRITER_ENCODING_TYPE_TEXT
:
2417 WS_XML_UTF8_TEXT
*new, *old
= (WS_XML_UTF8_TEXT
*)elem
->attributes
[elem
->attributeCount
- 1]->value
;
2418 if ((hr
= text_to_utf8text( value
, old
, NULL
, &new )) != S_OK
) return hr
;
2420 elem
->attributes
[elem
->attributeCount
- 1]->value
= &new->text
;
2423 case WS_XML_WRITER_ENCODING_TYPE_BINARY
:
2425 WS_XML_TEXT
*new, *old
= elem
->attributes
[elem
->attributeCount
- 1]->value
;
2426 if ((hr
= text_to_text( value
, old
, NULL
, &new )) != S_OK
) return hr
;
2428 elem
->attributes
[elem
->attributeCount
- 1]->value
= new;
2432 FIXME( "unhandled output encoding %u\n", writer
->output_enc
);
2439 static HRESULT
write_add_text_node( struct writer
*writer
, const WS_XML_TEXT
*value
)
2442 WS_XML_TEXT_NODE
*text
;
2445 if (node_type( writer
->current
) != WS_XML_NODE_TYPE_ELEMENT
&&
2446 node_type( writer
->current
) != WS_XML_NODE_TYPE_BOF
&&
2447 node_type( writer
->current
) != WS_XML_NODE_TYPE_CDATA
) return WS_E_INVALID_FORMAT
;
2449 if (!(node
= alloc_node( WS_XML_NODE_TYPE_TEXT
))) return E_OUTOFMEMORY
;
2450 text
= (WS_XML_TEXT_NODE
*)node
;
2452 switch (writer
->output_enc
)
2454 case WS_XML_WRITER_ENCODING_TYPE_TEXT
:
2456 WS_XML_UTF8_TEXT
*new;
2457 if ((hr
= text_to_utf8text( value
, NULL
, NULL
, &new )) != S_OK
)
2462 text
->text
= &new->text
;
2465 case WS_XML_WRITER_ENCODING_TYPE_BINARY
:
2468 if ((hr
= text_to_text( value
, NULL
, NULL
, &new )) != S_OK
)
2477 FIXME( "unhandled output encoding %u\n", writer
->output_enc
);
2482 write_insert_node( writer
, writer
->current
, node
);
2486 static HRESULT
write_text_text( struct writer
*writer
, const WS_XML_TEXT
*text
, ULONG offset
)
2488 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2491 if (node_type( writer
->current
->parent
) == WS_XML_NODE_TYPE_ELEMENT
)
2493 const struct escape
*escapes
[3] = { &escape_lt
, &escape_gt
, &escape_amp
};
2494 return write_bytes_escape( writer
, utf8
->value
.bytes
+ offset
, utf8
->value
.length
- offset
, escapes
, 3 );
2496 else if (node_type( writer
->current
->parent
) == WS_XML_NODE_TYPE_CDATA
)
2498 if ((hr
= write_grow_buffer( writer
, utf8
->value
.length
- offset
)) != S_OK
) return hr
;
2499 write_bytes( writer
, utf8
->value
.bytes
+ offset
, utf8
->value
.length
- offset
);
2503 return WS_E_INVALID_FORMAT
;
2506 static enum record_type
get_text_record_type( const WS_XML_TEXT
*text
, BOOL use_dict
)
2508 switch (text
->textType
)
2510 case WS_XML_TEXT_TYPE_UTF8
:
2512 const WS_XML_UTF8_TEXT
*text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2513 if (use_dict
) return RECORD_DICTIONARY_TEXT_WITH_ENDELEMENT
;
2514 if (text_utf8
->value
.length
<= MAX_UINT8
) return RECORD_CHARS8_TEXT_WITH_ENDELEMENT
;
2515 if (text_utf8
->value
.length
<= MAX_UINT16
) return RECORD_CHARS16_TEXT_WITH_ENDELEMENT
;
2516 return RECORD_CHARS32_TEXT_WITH_ENDELEMENT
;
2518 case WS_XML_TEXT_TYPE_UTF16
:
2520 const WS_XML_UTF16_TEXT
*text_utf16
= (const WS_XML_UTF16_TEXT
*)text
;
2521 int len
= text_utf16
->byteCount
/ sizeof(WCHAR
);
2522 int len_utf8
= WideCharToMultiByte( CP_UTF8
, 0, (const WCHAR
*)text_utf16
->bytes
, len
, NULL
, 0, NULL
, NULL
);
2523 if (len_utf8
<= MAX_UINT8
) return RECORD_CHARS8_TEXT_WITH_ENDELEMENT
;
2524 if (len_utf8
<= MAX_UINT16
) return RECORD_CHARS16_TEXT_WITH_ENDELEMENT
;
2525 return RECORD_CHARS32_TEXT_WITH_ENDELEMENT
;
2527 case WS_XML_TEXT_TYPE_BASE64
:
2529 const WS_XML_BASE64_TEXT
*text_base64
= (const WS_XML_BASE64_TEXT
*)text
;
2530 ULONG rem
= text_base64
->length
% 3, len
= text_base64
->length
- rem
;
2531 if (len
<= MAX_UINT8
) return RECORD_BYTES8_TEXT
;
2532 if (len
<= MAX_UINT16
) return RECORD_BYTES16_TEXT
;
2533 return RECORD_BYTES32_TEXT
;
2535 case WS_XML_TEXT_TYPE_BOOL
:
2537 const WS_XML_BOOL_TEXT
*text_bool
= (const WS_XML_BOOL_TEXT
*)text
;
2538 return text_bool
->value
? RECORD_TRUE_TEXT_WITH_ENDELEMENT
: RECORD_FALSE_TEXT_WITH_ENDELEMENT
;
2540 case WS_XML_TEXT_TYPE_INT32
:
2542 const WS_XML_INT32_TEXT
*text_int32
= (const WS_XML_INT32_TEXT
*)text
;
2543 if (!text_int32
->value
) return RECORD_ZERO_TEXT_WITH_ENDELEMENT
;
2544 if (text_int32
->value
== 1) return RECORD_ONE_TEXT_WITH_ENDELEMENT
;
2545 if (text_int32
->value
>= MIN_INT8
&& text_int32
->value
<= MAX_INT8
) return RECORD_INT8_TEXT_WITH_ENDELEMENT
;
2546 if (text_int32
->value
>= MIN_INT16
&& text_int32
->value
<= MAX_INT16
) return RECORD_INT16_TEXT_WITH_ENDELEMENT
;
2547 return RECORD_INT32_TEXT_WITH_ENDELEMENT
;
2549 case WS_XML_TEXT_TYPE_INT64
:
2551 const WS_XML_INT64_TEXT
*text_int64
= (const WS_XML_INT64_TEXT
*)text
;
2552 if (!text_int64
->value
) return RECORD_ZERO_TEXT_WITH_ENDELEMENT
;
2553 if (text_int64
->value
== 1) return RECORD_ONE_TEXT_WITH_ENDELEMENT
;
2554 if (text_int64
->value
>= MIN_INT8
&& text_int64
->value
<= MAX_INT8
) return RECORD_INT8_TEXT_WITH_ENDELEMENT
;
2555 if (text_int64
->value
>= MIN_INT16
&& text_int64
->value
<= MAX_INT16
) return RECORD_INT16_TEXT_WITH_ENDELEMENT
;
2556 if (text_int64
->value
>= MIN_INT32
&& text_int64
->value
<= MAX_INT32
) return RECORD_INT32_TEXT_WITH_ENDELEMENT
;
2557 return RECORD_INT64_TEXT_WITH_ENDELEMENT
;
2559 case WS_XML_TEXT_TYPE_UINT64
:
2561 const WS_XML_UINT64_TEXT
*text_uint64
= (const WS_XML_UINT64_TEXT
*)text
;
2562 if (!text_uint64
->value
) return RECORD_ZERO_TEXT_WITH_ENDELEMENT
;
2563 if (text_uint64
->value
== 1) return RECORD_ONE_TEXT_WITH_ENDELEMENT
;
2564 if (text_uint64
->value
<= MAX_INT8
) return RECORD_INT8_TEXT_WITH_ENDELEMENT
;
2565 if (text_uint64
->value
<= MAX_INT16
) return RECORD_INT16_TEXT_WITH_ENDELEMENT
;
2566 if (text_uint64
->value
<= MAX_INT32
) return RECORD_INT32_TEXT_WITH_ENDELEMENT
;
2567 if (text_uint64
->value
<= MAX_INT64
) return RECORD_INT64_TEXT_WITH_ENDELEMENT
;
2568 return RECORD_UINT64_TEXT_WITH_ENDELEMENT
;
2570 case WS_XML_TEXT_TYPE_DOUBLE
:
2572 const WS_XML_DOUBLE_TEXT
*text_double
= (const WS_XML_DOUBLE_TEXT
*)text
;
2573 if (!text_double
->value
) return RECORD_ZERO_TEXT_WITH_ENDELEMENT
;
2574 if (text_double
->value
== 1) return RECORD_ONE_TEXT_WITH_ENDELEMENT
;
2575 if (isinf( text_double
->value
) || (INT64
)text_double
->value
!= text_double
->value
)
2576 return RECORD_DOUBLE_TEXT_WITH_ENDELEMENT
;
2577 if (text_double
->value
<= MAX_INT8
) return RECORD_INT8_TEXT_WITH_ENDELEMENT
;
2578 if (text_double
->value
<= MAX_INT16
) return RECORD_INT16_TEXT_WITH_ENDELEMENT
;
2579 if (text_double
->value
<= MAX_INT32
) return RECORD_INT32_TEXT_WITH_ENDELEMENT
;
2580 return RECORD_INT64_TEXT_WITH_ENDELEMENT
;
2582 case WS_XML_TEXT_TYPE_GUID
:
2583 return RECORD_GUID_TEXT_WITH_ENDELEMENT
;
2585 case WS_XML_TEXT_TYPE_UNIQUE_ID
:
2586 return RECORD_UNIQUE_ID_TEXT_WITH_ENDELEMENT
;
2588 case WS_XML_TEXT_TYPE_DATETIME
:
2589 return RECORD_DATETIME_TEXT_WITH_ENDELEMENT
;
2592 FIXME( "unhandled text type %u\n", text
->textType
);
2597 static HRESULT
write_text_bin( struct writer
*writer
, const WS_XML_TEXT
*text
, ULONG offset
)
2599 enum record_type type
;
2600 BOOL use_dict
= FALSE
;
2606 FIXME( "no support for appending text in binary mode\n" );
2610 if (text
->textType
== WS_XML_TEXT_TYPE_UTF8
)
2612 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2613 use_dict
= get_string_id( writer
, &utf8
->value
, &id
);
2616 switch ((type
= get_text_record_type( text
, use_dict
)))
2618 case RECORD_CHARS8_TEXT_WITH_ENDELEMENT
:
2620 const WS_XML_UTF8_TEXT
*text_utf8
;
2621 WS_XML_UTF8_TEXT
*new = NULL
;
2624 if (text
->textType
== WS_XML_TEXT_TYPE_UTF8
) text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2627 if ((hr
= text_to_utf8text( text
, NULL
, NULL
, &new )) != S_OK
) return hr
;
2630 len
= text_utf8
->value
.length
;
2631 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(len
) + len
)) != S_OK
)
2636 write_char( writer
, type
);
2637 write_char( writer
, len
);
2638 write_bytes( writer
, text_utf8
->value
.bytes
, len
);
2642 case RECORD_CHARS16_TEXT_WITH_ENDELEMENT
:
2644 const WS_XML_UTF8_TEXT
*text_utf8
;
2645 WS_XML_UTF8_TEXT
*new = NULL
;
2648 if (text
->textType
== WS_XML_TEXT_TYPE_UTF8
) text_utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2651 if ((hr
= text_to_utf8text( text
, NULL
, NULL
, &new )) != S_OK
) return hr
;
2654 len
= text_utf8
->value
.length
;
2655 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(len
) + len
)) != S_OK
)
2660 write_char( writer
, type
);
2661 write_bytes( writer
, (const BYTE
*)&len
, sizeof(len
) );
2662 write_bytes( writer
, text_utf8
->value
.bytes
, len
);
2666 case RECORD_BYTES8_TEXT
:
2668 const WS_XML_BASE64_TEXT
*text_base64
= (const WS_XML_BASE64_TEXT
*)text
;
2669 UINT8 rem
= text_base64
->length
% 3, len
= text_base64
->length
- rem
;
2673 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(len
) + len
)) != S_OK
) return hr
;
2674 write_char( writer
, rem
? RECORD_BYTES8_TEXT
: RECORD_BYTES8_TEXT_WITH_ENDELEMENT
);
2675 write_char( writer
, len
);
2676 write_bytes( writer
, text_base64
->bytes
, len
);
2680 if ((hr
= write_grow_buffer( writer
, 3 )) != S_OK
) return hr
;
2681 write_char( writer
, RECORD_BYTES8_TEXT_WITH_ENDELEMENT
);
2682 write_char( writer
, rem
);
2683 write_bytes( writer
, (const BYTE
*)text_base64
->bytes
+ len
, rem
);
2687 case RECORD_BYTES16_TEXT
:
2689 const WS_XML_BASE64_TEXT
*text_base64
= (const WS_XML_BASE64_TEXT
*)text
;
2690 UINT16 rem
= text_base64
->length
% 3, len
= text_base64
->length
- rem
;
2694 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(len
) + len
)) != S_OK
) return hr
;
2695 write_char( writer
, rem
? RECORD_BYTES16_TEXT
: RECORD_BYTES16_TEXT_WITH_ENDELEMENT
);
2696 write_bytes( writer
, (const BYTE
*)&len
, sizeof(len
) );
2697 write_bytes( writer
, text_base64
->bytes
, len
);
2701 if ((hr
= write_grow_buffer( writer
, 3 )) != S_OK
) return hr
;
2702 write_char( writer
, RECORD_BYTES8_TEXT_WITH_ENDELEMENT
);
2703 write_char( writer
, rem
);
2704 write_bytes( writer
, (const BYTE
*)text_base64
->bytes
+ len
, rem
);
2708 case RECORD_ZERO_TEXT_WITH_ENDELEMENT
:
2709 case RECORD_ONE_TEXT_WITH_ENDELEMENT
:
2710 case RECORD_FALSE_TEXT_WITH_ENDELEMENT
:
2711 case RECORD_TRUE_TEXT_WITH_ENDELEMENT
:
2713 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
2714 write_char( writer
, type
);
2717 case RECORD_INT8_TEXT_WITH_ENDELEMENT
:
2719 INT8 val
= get_text_value_int( text
);
2720 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(val
) )) != S_OK
) return hr
;
2721 write_char( writer
, type
);
2722 write_char( writer
, val
);
2725 case RECORD_INT16_TEXT_WITH_ENDELEMENT
:
2727 INT16 val
= get_text_value_int( text
);
2728 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(val
) )) != S_OK
) return hr
;
2729 write_char( writer
, type
);
2730 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
2733 case RECORD_INT32_TEXT_WITH_ENDELEMENT
:
2735 INT32 val
= get_text_value_int( text
);
2736 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(val
) )) != S_OK
) return hr
;
2737 write_char( writer
, type
);
2738 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
2741 case RECORD_INT64_TEXT_WITH_ENDELEMENT
:
2743 INT64 val
= get_text_value_int( text
);
2744 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(val
) )) != S_OK
) return hr
;
2745 write_char( writer
, type
);
2746 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
2749 case RECORD_UINT64_TEXT_WITH_ENDELEMENT
:
2751 WS_XML_UINT64_TEXT
*text_uint64
= (WS_XML_UINT64_TEXT
*)text
;
2752 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(text_uint64
->value
) )) != S_OK
) return hr
;
2753 write_char( writer
, type
);
2754 write_bytes( writer
, (const BYTE
*)&text_uint64
->value
, sizeof(text_uint64
->value
) );
2757 case RECORD_DOUBLE_TEXT_WITH_ENDELEMENT
:
2759 WS_XML_DOUBLE_TEXT
*text_double
= (WS_XML_DOUBLE_TEXT
*)text
;
2760 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(text_double
->value
) )) != S_OK
) return hr
;
2761 write_char( writer
, type
);
2762 write_bytes( writer
, (const BYTE
*)&text_double
->value
, sizeof(text_double
->value
) );
2765 case RECORD_GUID_TEXT_WITH_ENDELEMENT
:
2767 WS_XML_GUID_TEXT
*text_guid
= (WS_XML_GUID_TEXT
*)text
;
2768 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(text_guid
->value
) )) != S_OK
) return hr
;
2769 write_char( writer
, type
);
2770 write_bytes( writer
, (const BYTE
*)&text_guid
->value
, sizeof(text_guid
->value
) );
2773 case RECORD_UNIQUE_ID_TEXT_WITH_ENDELEMENT
:
2775 WS_XML_UNIQUE_ID_TEXT
*text_unique_id
= (WS_XML_UNIQUE_ID_TEXT
*)text
;
2776 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(text_unique_id
->value
) )) != S_OK
) return hr
;
2777 write_char( writer
, type
);
2778 write_bytes( writer
, (const BYTE
*)&text_unique_id
->value
, sizeof(text_unique_id
->value
) );
2781 case RECORD_DATETIME_TEXT_WITH_ENDELEMENT
:
2783 WS_XML_DATETIME_TEXT
*text_datetime
= (WS_XML_DATETIME_TEXT
*)text
;
2784 UINT64 val
= text_datetime
->value
.ticks
;
2786 assert( val
<= TICKS_MAX
);
2787 if (text_datetime
->value
.format
== WS_DATETIME_FORMAT_UTC
) val
|= (UINT64
)1 << 62;
2788 else if (text_datetime
->value
.format
== WS_DATETIME_FORMAT_LOCAL
) val
|= (UINT64
)1 << 63;
2790 if ((hr
= write_grow_buffer( writer
, 1 + sizeof(val
) )) != S_OK
) return hr
;
2791 write_char( writer
, type
);
2792 write_bytes( writer
, (const BYTE
*)&val
, sizeof(val
) );
2795 case RECORD_DICTIONARY_TEXT_WITH_ENDELEMENT
:
2797 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
2798 write_char( writer
, type
);
2799 return write_dict_string( writer
, id
);
2802 FIXME( "unhandled record type %02x\n", type
);
2807 static HRESULT
write_text( struct writer
*writer
, const WS_XML_TEXT
*text
, ULONG offset
)
2809 if (!writer
->current
->parent
) return WS_E_INVALID_FORMAT
;
2811 switch (writer
->output_enc
)
2813 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_text_text( writer
, text
, offset
);
2814 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_text_bin( writer
, text
, offset
);
2816 ERR( "unhandled encoding %u\n", writer
->output_enc
);
2817 return WS_E_NOT_SUPPORTED
;
2821 static HRESULT
write_text_node( struct writer
*writer
, const WS_XML_TEXT
*text
)
2823 WS_XML_TEXT_NODE
*node
= (WS_XML_TEXT_NODE
*)writer
->current
;
2827 if ((hr
= write_flush( writer
)) != S_OK
) return hr
;
2828 if (node_type( writer
->current
) != WS_XML_NODE_TYPE_TEXT
)
2830 if ((hr
= write_add_text_node( writer
, text
)) != S_OK
) return hr
;
2831 node
= (WS_XML_TEXT_NODE
*)writer
->current
;
2835 switch (writer
->output_enc
)
2837 case WS_XML_WRITER_ENCODING_TYPE_TEXT
:
2839 WS_XML_UTF8_TEXT
*new, *old
= (WS_XML_UTF8_TEXT
*)node
->text
;
2840 offset
= old
->value
.length
;
2841 if ((hr
= text_to_utf8text( text
, old
, &offset
, &new )) != S_OK
) return hr
;
2843 node
->text
= &new->text
;
2846 case WS_XML_WRITER_ENCODING_TYPE_BINARY
:
2848 WS_XML_TEXT
*new, *old
= node
->text
;
2849 if ((hr
= text_to_text( text
, old
, &offset
, &new )) != S_OK
) return hr
;
2855 FIXME( "unhandled output encoding %u\n", writer
->output_enc
);
2860 if ((hr
= write_text( writer
, node
->text
, offset
)) != S_OK
) return hr
;
2862 writer
->state
= WRITER_STATE_TEXT
;
2866 /**************************************************************************
2867 * WsWriteText [webservices.@]
2869 HRESULT WINAPI
WsWriteText( WS_XML_WRITER
*handle
, const WS_XML_TEXT
*text
, WS_ERROR
*error
)
2871 struct writer
*writer
= (struct writer
*)handle
;
2874 TRACE( "%p %p %p\n", handle
, text
, error
);
2875 if (error
) FIXME( "ignoring error parameter\n" );
2877 if (!writer
|| !text
) return E_INVALIDARG
;
2879 EnterCriticalSection( &writer
->cs
);
2881 if (writer
->magic
!= WRITER_MAGIC
)
2883 LeaveCriticalSection( &writer
->cs
);
2884 return E_INVALIDARG
;
2887 if (writer
->state
== WRITER_STATE_STARTATTRIBUTE
) hr
= write_set_attribute_value( writer
, text
);
2888 else hr
= write_text_node( writer
, text
);
2890 LeaveCriticalSection( &writer
->cs
);
2894 /**************************************************************************
2895 * WsWriteBytes [webservices.@]
2897 HRESULT WINAPI
WsWriteBytes( WS_XML_WRITER
*handle
, const void *bytes
, ULONG count
, WS_ERROR
*error
)
2899 struct writer
*writer
= (struct writer
*)handle
;
2900 WS_XML_BASE64_TEXT base64
;
2903 TRACE( "%p %p %u %p\n", handle
, bytes
, count
, error
);
2904 if (error
) FIXME( "ignoring error parameter\n" );
2906 if (!writer
) return E_INVALIDARG
;
2908 EnterCriticalSection( &writer
->cs
);
2910 if (writer
->magic
!= WRITER_MAGIC
)
2912 LeaveCriticalSection( &writer
->cs
);
2913 return E_INVALIDARG
;
2916 if (!writer
->output_type
)
2918 LeaveCriticalSection( &writer
->cs
);
2919 return WS_E_INVALID_OPERATION
;
2922 base64
.text
.textType
= WS_XML_TEXT_TYPE_BASE64
;
2923 base64
.bytes
= (BYTE
*)bytes
;
2924 base64
.length
= count
;
2926 if (writer
->state
== WRITER_STATE_STARTATTRIBUTE
) hr
= write_set_attribute_value( writer
, &base64
.text
);
2927 else hr
= write_text_node( writer
, &base64
.text
);
2929 LeaveCriticalSection( &writer
->cs
);
2933 /**************************************************************************
2934 * WsWriteChars [webservices.@]
2936 HRESULT WINAPI
WsWriteChars( WS_XML_WRITER
*handle
, const WCHAR
*chars
, ULONG count
, WS_ERROR
*error
)
2938 struct writer
*writer
= (struct writer
*)handle
;
2939 WS_XML_UTF16_TEXT utf16
;
2942 TRACE( "%p %s %u %p\n", handle
, debugstr_wn(chars
, count
), count
, error
);
2943 if (error
) FIXME( "ignoring error parameter\n" );
2945 if (!writer
) return E_INVALIDARG
;
2947 EnterCriticalSection( &writer
->cs
);
2949 if (writer
->magic
!= WRITER_MAGIC
)
2951 LeaveCriticalSection( &writer
->cs
);
2952 return E_INVALIDARG
;
2955 if (!writer
->output_type
)
2957 LeaveCriticalSection( &writer
->cs
);
2958 return WS_E_INVALID_OPERATION
;
2961 utf16
.text
.textType
= WS_XML_TEXT_TYPE_UTF16
;
2962 utf16
.bytes
= (BYTE
*)chars
;
2963 utf16
.byteCount
= count
* sizeof(WCHAR
);
2965 if (writer
->state
== WRITER_STATE_STARTATTRIBUTE
) hr
= write_set_attribute_value( writer
, &utf16
.text
);
2966 else hr
= write_text_node( writer
, &utf16
.text
);
2968 LeaveCriticalSection( &writer
->cs
);
2972 /**************************************************************************
2973 * WsWriteCharsUtf8 [webservices.@]
2975 HRESULT WINAPI
WsWriteCharsUtf8( WS_XML_WRITER
*handle
, const BYTE
*bytes
, ULONG count
, WS_ERROR
*error
)
2977 struct writer
*writer
= (struct writer
*)handle
;
2978 WS_XML_UTF8_TEXT utf8
;
2981 TRACE( "%p %s %u %p\n", handle
, debugstr_an((const char *)bytes
, count
), count
, error
);
2982 if (error
) FIXME( "ignoring error parameter\n" );
2984 if (!writer
) return E_INVALIDARG
;
2986 EnterCriticalSection( &writer
->cs
);
2988 if (writer
->magic
!= WRITER_MAGIC
)
2990 LeaveCriticalSection( &writer
->cs
);
2991 return E_INVALIDARG
;
2994 if (!writer
->output_type
)
2996 LeaveCriticalSection( &writer
->cs
);
2997 return WS_E_INVALID_OPERATION
;
3000 utf8
.text
.textType
= WS_XML_TEXT_TYPE_UTF8
;
3001 utf8
.value
.bytes
= (BYTE
*)bytes
;
3002 utf8
.value
.length
= count
;
3004 if (writer
->state
== WRITER_STATE_STARTATTRIBUTE
) hr
= write_set_attribute_value( writer
, &utf8
.text
);
3005 else hr
= write_text_node( writer
, &utf8
.text
);
3007 LeaveCriticalSection( &writer
->cs
);
3011 static HRESULT
write_type_text( struct writer
*writer
, WS_TYPE_MAPPING mapping
, const WS_XML_TEXT
*text
)
3015 case WS_ELEMENT_TYPE_MAPPING
:
3016 case WS_ELEMENT_CONTENT_TYPE_MAPPING
:
3017 return write_text_node( writer
, text
);
3019 case WS_ATTRIBUTE_TYPE_MAPPING
:
3020 return write_set_attribute_value( writer
, text
);
3022 case WS_ANY_ELEMENT_TYPE_MAPPING
:
3023 switch (writer
->state
)
3025 case WRITER_STATE_STARTATTRIBUTE
:
3026 return write_set_attribute_value( writer
, text
);
3028 case WRITER_STATE_STARTELEMENT
:
3029 return write_text_node( writer
, text
);
3032 FIXME( "writer state %u not handled\n", writer
->state
);
3037 FIXME( "mapping %u not implemented\n", mapping
);
3042 static HRESULT
write_add_nil_attribute( struct writer
*writer
)
3044 static const WS_XML_STRING prefix
= {1, (BYTE
*)"a"};
3045 static const WS_XML_STRING localname
= {3, (BYTE
*)"nil"};
3046 static const WS_XML_STRING ns
= {41, (BYTE
*)"http://www.w3.org/2001/XMLSchema-instance"};
3047 static const WS_XML_UTF8_TEXT value
= {{WS_XML_TEXT_TYPE_UTF8
}, {4, (BYTE
*)"true"}};
3050 if ((hr
= write_add_attribute( writer
, &prefix
, &localname
, &ns
, FALSE
)) != S_OK
) return hr
;
3051 if ((hr
= write_set_attribute_value( writer
, &value
.text
)) != S_OK
) return hr
;
3052 return add_namespace_attribute( writer
, &prefix
, &ns
, FALSE
);
3055 static HRESULT
get_value_ptr( WS_WRITE_OPTION option
, const void *value
, ULONG size
, ULONG expected_size
,
3060 case WS_WRITE_REQUIRED_VALUE
:
3061 case WS_WRITE_NILLABLE_VALUE
:
3062 if (!value
|| size
!= expected_size
) return E_INVALIDARG
;
3066 case WS_WRITE_REQUIRED_POINTER
:
3067 if (size
!= sizeof(const void *) || !(*ptr
= *(const void **)value
)) return E_INVALIDARG
;
3070 case WS_WRITE_NILLABLE_POINTER
:
3071 if (size
!= sizeof(const void *)) return E_INVALIDARG
;
3072 *ptr
= *(const void **)value
;
3076 return E_INVALIDARG
;
3080 static HRESULT
write_type_bool( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3081 const WS_BOOL_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3082 const BOOL
*value
, ULONG size
)
3084 WS_XML_BOOL_TEXT text_bool
;
3090 FIXME( "description not supported\n" );
3094 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3095 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(BOOL
), (const void **)&ptr
)) != S_OK
) return hr
;
3096 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3098 text_bool
.text
.textType
= WS_XML_TEXT_TYPE_BOOL
;
3099 text_bool
.value
= *ptr
;
3100 return write_type_text( writer
, mapping
, &text_bool
.text
);
3103 static HRESULT
write_type_int8( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3104 const WS_INT8_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3105 const BOOL
*value
, ULONG size
)
3107 WS_XML_INT32_TEXT text_int32
;
3113 FIXME( "description not supported\n" );
3117 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3118 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(INT8
), (const void **)&ptr
)) != S_OK
) return hr
;
3119 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3121 text_int32
.text
.textType
= WS_XML_TEXT_TYPE_INT32
;
3122 text_int32
.value
= *ptr
;
3123 return write_type_text( writer
, mapping
, &text_int32
.text
);
3126 static HRESULT
write_type_int16( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3127 const WS_INT16_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3128 const BOOL
*value
, ULONG size
)
3130 WS_XML_INT32_TEXT text_int32
;
3136 FIXME( "description not supported\n" );
3140 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3141 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(INT16
), (const void **)&ptr
)) != S_OK
) return hr
;
3142 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3144 text_int32
.text
.textType
= WS_XML_TEXT_TYPE_INT32
;
3145 text_int32
.value
= *ptr
;
3146 return write_type_text( writer
, mapping
, &text_int32
.text
);
3149 static HRESULT
write_type_int32( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3150 const WS_INT32_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3151 const void *value
, ULONG size
)
3153 WS_XML_INT32_TEXT text_int32
;
3159 FIXME( "description not supported\n" );
3163 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3164 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(INT32
), (const void **)&ptr
)) != S_OK
) return hr
;
3165 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3167 text_int32
.text
.textType
= WS_XML_TEXT_TYPE_INT32
;
3168 text_int32
.value
= *ptr
;
3169 return write_type_text( writer
, mapping
, &text_int32
.text
);
3172 static HRESULT
write_type_int64( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3173 const WS_INT64_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3174 const void *value
, ULONG size
)
3176 WS_XML_INT64_TEXT text_int64
;
3182 FIXME( "description not supported\n" );
3186 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3187 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(INT64
), (const void **)&ptr
)) != S_OK
) return hr
;
3188 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3190 text_int64
.text
.textType
= WS_XML_TEXT_TYPE_INT64
;
3191 text_int64
.value
= *ptr
;
3192 return write_type_text( writer
, mapping
, &text_int64
.text
);
3195 static HRESULT
write_type_uint8( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3196 const WS_UINT8_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3197 const void *value
, ULONG size
)
3199 WS_XML_UINT64_TEXT text_uint64
;
3205 FIXME( "description not supported\n" );
3209 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3210 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(UINT8
), (const void **)&ptr
)) != S_OK
) return hr
;
3211 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3213 text_uint64
.text
.textType
= WS_XML_TEXT_TYPE_UINT64
;
3214 text_uint64
.value
= *ptr
;
3215 return write_type_text( writer
, mapping
, &text_uint64
.text
);
3218 static HRESULT
write_type_uint16( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3219 const WS_UINT16_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3220 const void *value
, ULONG size
)
3222 WS_XML_UINT64_TEXT text_uint64
;
3228 FIXME( "description not supported\n" );
3232 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3233 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(UINT16
), (const void **)&ptr
)) != S_OK
) return hr
;
3234 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3236 text_uint64
.text
.textType
= WS_XML_TEXT_TYPE_UINT64
;
3237 text_uint64
.value
= *ptr
;
3238 return write_type_text( writer
, mapping
, &text_uint64
.text
);
3241 static HRESULT
write_type_uint32( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3242 const WS_UINT32_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3243 const void *value
, ULONG size
)
3245 WS_XML_UINT64_TEXT text_uint64
;
3251 FIXME( "description not supported\n" );
3255 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3256 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(UINT32
), (const void **)&ptr
)) != S_OK
) return hr
;
3257 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3259 text_uint64
.text
.textType
= WS_XML_TEXT_TYPE_UINT64
;
3260 text_uint64
.value
= *ptr
;
3261 return write_type_text( writer
, mapping
, &text_uint64
.text
);
3264 static HRESULT
write_type_uint64( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3265 const WS_UINT64_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3266 const void *value
, ULONG size
)
3268 WS_XML_UINT64_TEXT text_uint64
;
3274 FIXME( "description not supported\n" );
3278 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3279 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(UINT64
), (const void **)&ptr
)) != S_OK
) return hr
;
3280 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3282 text_uint64
.text
.textType
= WS_XML_TEXT_TYPE_UINT64
;
3283 text_uint64
.value
= *ptr
;
3284 return write_type_text( writer
, mapping
, &text_uint64
.text
);
3287 static HRESULT
write_type_double( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3288 const WS_DOUBLE_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3289 const void *value
, ULONG size
)
3291 WS_XML_DOUBLE_TEXT text_double
;
3297 FIXME( "description not supported\n" );
3301 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3302 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(double), (const void **)&ptr
)) != S_OK
) return hr
;
3303 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3305 text_double
.text
.textType
= WS_XML_TEXT_TYPE_DOUBLE
;
3306 text_double
.value
= *ptr
;
3307 return write_type_text( writer
, mapping
, &text_double
.text
);
3310 static HRESULT
write_type_datetime( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3311 const WS_DATETIME_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3312 const void *value
, ULONG size
)
3314 WS_XML_DATETIME_TEXT text_datetime
;
3315 const WS_DATETIME
*ptr
;
3320 FIXME( "description not supported\n" );
3324 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3325 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(WS_DATETIME
), (const void **)&ptr
)) != S_OK
) return hr
;
3326 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3327 if (ptr
->ticks
> TICKS_MAX
|| ptr
->format
> WS_DATETIME_FORMAT_NONE
) return WS_E_INVALID_FORMAT
;
3329 text_datetime
.text
.textType
= WS_XML_TEXT_TYPE_DATETIME
;
3330 text_datetime
.value
= *ptr
;
3331 return write_type_text( writer
, mapping
, &text_datetime
.text
);
3334 static HRESULT
write_type_guid( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3335 const WS_GUID_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3336 const void *value
, ULONG size
)
3338 WS_XML_GUID_TEXT text_guid
;
3344 FIXME( "description not supported\n" );
3348 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3349 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(GUID
), (const void **)&ptr
)) != S_OK
) return hr
;
3350 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3352 text_guid
.text
.textType
= WS_XML_TEXT_TYPE_GUID
;
3353 text_guid
.value
= *ptr
;
3354 return write_type_text( writer
, mapping
, &text_guid
.text
);
3357 static HRESULT
write_type_unique_id( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3358 const WS_UNIQUE_ID_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3359 const void *value
, ULONG size
)
3361 WS_XML_UNIQUE_ID_TEXT text_unique_id
;
3362 WS_XML_UTF16_TEXT text_utf16
;
3363 const WS_UNIQUE_ID
*ptr
;
3368 FIXME( "description not supported\n" );
3372 if (!option
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3373 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(*ptr
), (const void **)&ptr
)) != S_OK
) return hr
;
3374 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3376 if (ptr
->uri
.length
)
3378 text_utf16
.text
.textType
= WS_XML_TEXT_TYPE_UTF16
;
3379 text_utf16
.bytes
= (BYTE
*)ptr
->uri
.chars
;
3380 text_utf16
.byteCount
= ptr
->uri
.length
* sizeof(WCHAR
);
3381 return write_type_text( writer
, mapping
, &text_utf16
.text
);
3384 text_unique_id
.text
.textType
= WS_XML_TEXT_TYPE_UNIQUE_ID
;
3385 text_unique_id
.value
= ptr
->guid
;
3386 return write_type_text( writer
, mapping
, &text_unique_id
.text
);
3389 static HRESULT
write_type_string( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3390 const WS_STRING_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3391 const void *value
, ULONG size
)
3393 WS_XML_UTF16_TEXT utf16
;
3394 const WS_STRING
*ptr
;
3399 FIXME( "description not supported\n" );
3403 if (!option
) return E_INVALIDARG
;
3404 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(WS_STRING
), (const void **)&ptr
)) != S_OK
) return hr
;
3405 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3406 if (!ptr
->length
) return S_OK
;
3408 utf16
.text
.textType
= WS_XML_TEXT_TYPE_UTF16
;
3409 utf16
.bytes
= (BYTE
*)ptr
->chars
;
3410 utf16
.byteCount
= ptr
->length
* sizeof(WCHAR
);
3411 return write_type_text( writer
, mapping
, &utf16
.text
);
3414 static HRESULT
write_type_wsz( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3415 const WS_WSZ_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3416 const void *value
, ULONG size
)
3418 WS_XML_UTF16_TEXT utf16
;
3425 FIXME( "description not supported\n" );
3429 if (!option
|| option
== WS_WRITE_REQUIRED_VALUE
|| option
== WS_WRITE_NILLABLE_VALUE
) return E_INVALIDARG
;
3430 if ((hr
= get_value_ptr( option
, value
, size
, 0, (const void **)&ptr
)) != S_OK
) return hr
;
3431 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3432 if (!(len
= strlenW( ptr
))) return S_OK
;
3434 utf16
.text
.textType
= WS_XML_TEXT_TYPE_UTF16
;
3435 utf16
.bytes
= (BYTE
*)ptr
;
3436 utf16
.byteCount
= len
* sizeof(WCHAR
);
3437 return write_type_text( writer
, mapping
, &utf16
.text
);
3440 static HRESULT
write_type_bytes( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3441 const WS_BYTES_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3442 const void *value
, ULONG size
)
3444 WS_XML_BASE64_TEXT base64
;
3445 const WS_BYTES
*ptr
;
3450 FIXME( "description not supported\n" );
3454 if (!option
) return E_INVALIDARG
;
3455 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(WS_BYTES
), (const void **)&ptr
)) != S_OK
) return hr
;
3456 if ((option
== WS_WRITE_NILLABLE_VALUE
&& is_nil_value( value
, size
)) ||
3457 (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
)) return write_add_nil_attribute( writer
);
3458 if (!ptr
->length
) return S_OK
;
3460 base64
.text
.textType
= WS_XML_TEXT_TYPE_BASE64
;
3461 base64
.bytes
= ptr
->bytes
;
3462 base64
.length
= ptr
->length
;
3463 return write_type_text( writer
, mapping
, &base64
.text
);
3466 static HRESULT
write_type_xml_string( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3467 const WS_XML_STRING_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3468 const void *value
, ULONG size
)
3470 WS_XML_UTF8_TEXT utf8
;
3471 const WS_XML_STRING
*ptr
;
3476 FIXME( "description not supported\n" );
3480 if (!option
) return E_INVALIDARG
;
3481 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(WS_XML_STRING
), (const void **)&ptr
)) != S_OK
) return hr
;
3482 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3483 if (option
== WS_WRITE_NILLABLE_VALUE
&& is_nil_value( value
, size
)) return write_add_nil_attribute( writer
);
3484 if (!ptr
->length
) return S_OK
;
3486 utf8
.text
.textType
= WS_XML_TEXT_TYPE_UTF8
;
3487 utf8
.value
.bytes
= ptr
->bytes
;
3488 utf8
.value
.length
= ptr
->length
;
3489 return write_type_text( writer
, mapping
, &utf8
.text
);
3492 static HRESULT
find_prefix( struct writer
*writer
, const WS_XML_STRING
*ns
, const WS_XML_STRING
**prefix
)
3494 const struct node
*node
;
3495 for (node
= writer
->current
; node_type( node
) == WS_XML_NODE_TYPE_ELEMENT
; node
= node
->parent
)
3497 const WS_XML_ELEMENT_NODE
*elem
= &node
->hdr
;
3499 for (i
= 0; i
< elem
->attributeCount
; i
++)
3501 if (!elem
->attributes
[i
]->isXmlNs
) continue;
3502 if (WsXmlStringEquals( elem
->attributes
[i
]->ns
, ns
, NULL
) != S_OK
) continue;
3503 *prefix
= elem
->attributes
[i
]->prefix
;
3507 return WS_E_INVALID_FORMAT
;
3510 static HRESULT
write_type_qname( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3511 const WS_XML_QNAME_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3512 const void *value
, ULONG size
)
3514 WS_XML_QNAME_TEXT qname
;
3515 const WS_XML_QNAME
*ptr
;
3516 const WS_XML_STRING
*prefix
;
3521 FIXME( "description not supported\n" );
3525 if (!option
) return E_INVALIDARG
;
3526 if ((hr
= get_value_ptr( option
, value
, size
, sizeof(*ptr
), (const void **)&ptr
)) != S_OK
) return hr
;
3527 if (option
== WS_WRITE_NILLABLE_POINTER
&& !ptr
) return write_add_nil_attribute( writer
);
3528 if (option
== WS_WRITE_NILLABLE_VALUE
&& is_nil_value( value
, size
)) return write_add_nil_attribute( writer
);
3530 if (((hr
= find_prefix( writer
, &ptr
->ns
, &prefix
)) != S_OK
)) return hr
;
3532 qname
.text
.textType
= WS_XML_TEXT_TYPE_QNAME
;
3533 qname
.prefix
= (WS_XML_STRING
*)prefix
;
3534 qname
.localName
= (WS_XML_STRING
*)&ptr
->localName
;
3535 qname
.ns
= (WS_XML_STRING
*)&ptr
->ns
;
3536 return write_type_text( writer
, mapping
, &qname
.text
);
3539 static WS_WRITE_OPTION
get_field_write_option( WS_TYPE type
, ULONG options
)
3541 if (options
& WS_FIELD_POINTER
)
3543 if (options
& (WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
)) return WS_WRITE_NILLABLE_POINTER
;
3544 return WS_WRITE_REQUIRED_POINTER
;
3555 case WS_UINT16_TYPE
:
3556 case WS_UINT32_TYPE
:
3557 case WS_UINT64_TYPE
:
3558 case WS_DOUBLE_TYPE
:
3559 case WS_DATETIME_TYPE
:
3561 case WS_UNIQUE_ID_TYPE
:
3562 case WS_STRING_TYPE
:
3564 case WS_XML_STRING_TYPE
:
3565 case WS_XML_QNAME_TYPE
:
3566 case WS_STRUCT_TYPE
:
3569 if (options
& (WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
)) return WS_WRITE_NILLABLE_VALUE
;
3570 return WS_WRITE_REQUIRED_VALUE
;
3573 case WS_DESCRIPTION_TYPE
:
3574 if (options
& (WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
)) return WS_WRITE_NILLABLE_POINTER
;
3575 return WS_WRITE_REQUIRED_POINTER
;
3578 FIXME( "unhandled type %u\n", type
);
3583 static HRESULT
write_type_field( struct writer
*, const WS_FIELD_DESCRIPTION
*, const char *, ULONG
);
3585 static HRESULT
write_type( struct writer
*, WS_TYPE_MAPPING
, WS_TYPE
, const void *, WS_WRITE_OPTION
,
3586 const void *, ULONG
);
3588 static HRESULT
write_type_union( struct writer
*writer
, const WS_UNION_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3589 const void *value
, ULONG size
)
3596 if ((hr
= get_value_ptr( option
, value
, size
, desc
->size
, &ptr
)) != S_OK
) return hr
;
3598 if (size
< sizeof(enum_value
)) return E_INVALIDARG
;
3599 if ((enum_value
= *(int *)(char *)ptr
+ desc
->enumOffset
) == desc
->noneEnumValue
)
3603 case WS_WRITE_REQUIRED_VALUE
:
3604 return WS_E_INVALID_FORMAT
;
3606 case WS_WRITE_NILLABLE_VALUE
:
3610 ERR( "unhandled write option %u\n", option
);
3611 return E_INVALIDARG
;
3615 for (i
= 0; i
< desc
->fieldCount
; i
++)
3617 if (desc
->fields
[i
]->value
== enum_value
)
3619 offset
= desc
->fields
[i
]->field
.offset
;
3620 return write_type_field( writer
, &desc
->fields
[i
]->field
, ptr
, offset
);
3624 return E_INVALIDARG
;
3627 static HRESULT
write_type_array( struct writer
*writer
, const WS_FIELD_DESCRIPTION
*desc
, const char *buf
,
3631 ULONG i
, size
, offset
= 0;
3632 WS_WRITE_OPTION option
;
3634 if (!(option
= get_field_write_option( desc
->type
, desc
->options
))) return E_INVALIDARG
;
3636 /* wrapper element */
3637 if (desc
->localName
&& ((hr
= write_element_node( writer
, NULL
, desc
->localName
, desc
->ns
)) != S_OK
))
3640 if (option
== WS_WRITE_REQUIRED_VALUE
|| option
== WS_WRITE_NILLABLE_VALUE
)
3641 size
= get_type_size( desc
->type
, desc
->typeDescription
);
3643 size
= sizeof(const void *);
3645 for (i
= 0; i
< count
; i
++)
3647 if (desc
->type
== WS_UNION_TYPE
)
3649 if ((hr
= write_type_union( writer
, desc
->typeDescription
, option
, buf
+ offset
, size
)) != S_OK
)
3654 if ((hr
= write_element_node( writer
, NULL
, desc
->itemLocalName
, desc
->itemNs
)) != S_OK
) return hr
;
3655 if ((hr
= write_type( writer
, WS_ELEMENT_TYPE_MAPPING
, desc
->type
, desc
->typeDescription
, option
,
3656 buf
+ offset
, size
)) != S_OK
) return hr
;
3657 if ((hr
= write_endelement_node( writer
)) != S_OK
) return hr
;
3662 if (desc
->localName
) hr
= write_endelement_node( writer
);
3666 static HRESULT
write_type_field( struct writer
*writer
, const WS_FIELD_DESCRIPTION
*desc
, const char *buf
,
3670 WS_TYPE_MAPPING mapping
;
3671 WS_WRITE_OPTION option
;
3672 ULONG count
, size
, field_options
= desc
->options
;
3673 const char *ptr
= buf
+ offset
;
3675 if (field_options
& ~(WS_FIELD_POINTER
|WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
))
3677 FIXME( "options 0x%x not supported\n", desc
->options
);
3681 /* zero-terminated strings are always pointers */
3682 if (desc
->type
== WS_WSZ_TYPE
) field_options
|= WS_FIELD_POINTER
;
3684 if (field_options
& WS_FIELD_POINTER
)
3685 size
= sizeof(const void *);
3687 size
= get_type_size( desc
->type
, desc
->typeDescription
);
3689 if (is_nil_value( ptr
, size
))
3691 if (field_options
& WS_FIELD_OPTIONAL
) return S_OK
;
3692 if (field_options
& WS_FIELD_NILLABLE
)
3694 if (field_options
& WS_FIELD_POINTER
) option
= WS_WRITE_NILLABLE_POINTER
;
3695 else option
= WS_WRITE_NILLABLE_VALUE
;
3699 if (field_options
& WS_FIELD_POINTER
) option
= WS_WRITE_REQUIRED_POINTER
;
3700 else option
= WS_WRITE_REQUIRED_VALUE
;
3705 if (field_options
& WS_FIELD_POINTER
) option
= WS_WRITE_REQUIRED_POINTER
;
3706 else option
= WS_WRITE_REQUIRED_VALUE
;
3709 switch (desc
->mapping
)
3711 case WS_ATTRIBUTE_FIELD_MAPPING
:
3712 if (!desc
->localName
|| !desc
->ns
) return E_INVALIDARG
;
3713 if ((hr
= write_add_attribute( writer
, NULL
, desc
->localName
, desc
->ns
, FALSE
)) != S_OK
)
3715 writer
->state
= WRITER_STATE_STARTATTRIBUTE
;
3717 mapping
= WS_ATTRIBUTE_TYPE_MAPPING
;
3720 case WS_ELEMENT_FIELD_MAPPING
:
3721 if ((hr
= write_element_node( writer
, NULL
, desc
->localName
, desc
->ns
)) != S_OK
) return hr
;
3722 mapping
= WS_ELEMENT_TYPE_MAPPING
;
3725 case WS_ELEMENT_CHOICE_FIELD_MAPPING
:
3726 if (desc
->type
!= WS_UNION_TYPE
|| !desc
->typeDescription
) return E_INVALIDARG
;
3727 option
= (field_options
& WS_FIELD_OPTIONAL
) ? WS_WRITE_NILLABLE_VALUE
: WS_WRITE_REQUIRED_VALUE
;
3728 return write_type_union( writer
, desc
->typeDescription
, option
, ptr
, size
);
3730 case WS_REPEATING_ELEMENT_FIELD_MAPPING
:
3731 case WS_REPEATING_ELEMENT_CHOICE_FIELD_MAPPING
:
3732 count
= *(const ULONG
*)(buf
+ desc
->countOffset
);
3733 return write_type_array( writer
, desc
, *(const char **)ptr
, count
);
3735 case WS_TEXT_FIELD_MAPPING
:
3736 switch (writer
->state
)
3738 case WRITER_STATE_STARTELEMENT
:
3739 mapping
= WS_ELEMENT_CONTENT_TYPE_MAPPING
;
3742 case WRITER_STATE_STARTATTRIBUTE
:
3743 mapping
= WS_ATTRIBUTE_TYPE_MAPPING
;
3747 FIXME( "unhandled writer state %u\n", writer
->state
);
3753 FIXME( "field mapping %u not supported\n", desc
->mapping
);
3757 if ((hr
= write_type( writer
, mapping
, desc
->type
, desc
->typeDescription
, option
, ptr
, size
)) != S_OK
)
3762 case WS_ATTRIBUTE_TYPE_MAPPING
:
3763 writer
->state
= WRITER_STATE_STARTELEMENT
;
3766 case WS_ELEMENT_TYPE_MAPPING
:
3767 if ((hr
= write_endelement_node( writer
)) != S_OK
) return hr
;
3776 static HRESULT
write_type_struct( struct writer
*writer
, WS_TYPE_MAPPING mapping
,
3777 const WS_STRUCT_DESCRIPTION
*desc
, WS_WRITE_OPTION option
,
3778 const void *value
, ULONG size
)
3784 if (!desc
) return E_INVALIDARG
;
3785 if (desc
->structOptions
) FIXME( "struct options 0x%x not supported\n", desc
->structOptions
);
3787 if ((hr
= get_value_ptr( option
, value
, size
, desc
->size
, &ptr
)) != S_OK
) return hr
;
3789 for (i
= 0; i
< desc
->fieldCount
; i
++)
3791 offset
= desc
->fields
[i
]->offset
;
3792 if ((hr
= write_type_field( writer
, desc
->fields
[i
], ptr
, offset
)) != S_OK
) return hr
;
3798 static HRESULT
write_type( struct writer
*writer
, WS_TYPE_MAPPING mapping
, WS_TYPE type
,
3799 const void *desc
, WS_WRITE_OPTION option
, const void *value
,
3805 return write_type_bool( writer
, mapping
, desc
, option
, value
, size
);
3808 return write_type_int8( writer
, mapping
, desc
, option
, value
, size
);
3811 return write_type_int16( writer
, mapping
, desc
, option
, value
, size
);
3814 return write_type_int32( writer
, mapping
, desc
, option
, value
, size
);
3817 return write_type_int64( writer
, mapping
, desc
, option
, value
, size
);
3820 return write_type_uint8( writer
, mapping
, desc
, option
, value
, size
);
3822 case WS_UINT16_TYPE
:
3823 return write_type_uint16( writer
, mapping
, desc
, option
, value
, size
);
3825 case WS_UINT32_TYPE
:
3826 return write_type_uint32( writer
, mapping
, desc
, option
, value
, size
);
3828 case WS_UINT64_TYPE
:
3829 return write_type_uint64( writer
, mapping
, desc
, option
, value
, size
);
3831 case WS_DOUBLE_TYPE
:
3832 return write_type_double( writer
, mapping
, desc
, option
, value
, size
);
3834 case WS_DATETIME_TYPE
:
3835 return write_type_datetime( writer
, mapping
, desc
, option
, value
, size
);
3838 return write_type_guid( writer
, mapping
, desc
, option
, value
, size
);
3840 case WS_UNIQUE_ID_TYPE
:
3841 return write_type_unique_id( writer
, mapping
, desc
, option
, value
, size
);
3843 case WS_STRING_TYPE
:
3844 return write_type_string( writer
, mapping
, desc
, option
, value
, size
);
3847 return write_type_wsz( writer
, mapping
, desc
, option
, value
, size
);
3850 return write_type_bytes( writer
, mapping
, desc
, option
, value
, size
);
3852 case WS_XML_STRING_TYPE
:
3853 return write_type_xml_string( writer
, mapping
, desc
, option
, value
, size
);
3855 case WS_XML_QNAME_TYPE
:
3856 return write_type_qname( writer
, mapping
, desc
, option
, value
, size
);
3858 case WS_STRUCT_TYPE
:
3859 return write_type_struct( writer
, mapping
, desc
, option
, value
, size
);
3862 FIXME( "type %u not supported\n", type
);
3867 /**************************************************************************
3868 * WsWriteAttribute [webservices.@]
3870 HRESULT WINAPI
WsWriteAttribute( WS_XML_WRITER
*handle
, const WS_ATTRIBUTE_DESCRIPTION
*desc
,
3871 WS_WRITE_OPTION option
, const void *value
, ULONG size
,
3874 struct writer
*writer
= (struct writer
*)handle
;
3877 TRACE( "%p %p %u %p %u %p\n", handle
, desc
, option
, value
, size
, error
);
3878 if (error
) FIXME( "ignoring error parameter\n" );
3880 if (!writer
|| !desc
|| !desc
->attributeLocalName
|| !desc
->attributeNs
|| !value
)
3881 return E_INVALIDARG
;
3883 EnterCriticalSection( &writer
->cs
);
3885 if (writer
->magic
!= WRITER_MAGIC
)
3887 LeaveCriticalSection( &writer
->cs
);
3888 return E_INVALIDARG
;
3891 if (writer
->state
!= WRITER_STATE_STARTELEMENT
)
3893 LeaveCriticalSection( &writer
->cs
);
3894 return WS_E_INVALID_OPERATION
;
3897 if ((hr
= write_add_attribute( writer
, NULL
, desc
->attributeLocalName
, desc
->attributeNs
, FALSE
)) != S_OK
)
3899 LeaveCriticalSection( &writer
->cs
);
3902 writer
->state
= WRITER_STATE_STARTATTRIBUTE
;
3904 hr
= write_type( writer
, WS_ATTRIBUTE_TYPE_MAPPING
, desc
->type
, desc
->typeDescription
, option
, value
, size
);
3906 LeaveCriticalSection( &writer
->cs
);
3910 /**************************************************************************
3911 * WsWriteElement [webservices.@]
3913 HRESULT WINAPI
WsWriteElement( WS_XML_WRITER
*handle
, const WS_ELEMENT_DESCRIPTION
*desc
,
3914 WS_WRITE_OPTION option
, const void *value
, ULONG size
,
3917 struct writer
*writer
= (struct writer
*)handle
;
3920 TRACE( "%p %p %u %p %u %p\n", handle
, desc
, option
, value
, size
, error
);
3921 if (error
) FIXME( "ignoring error parameter\n" );
3923 if (!writer
|| !desc
|| !desc
->elementLocalName
|| !desc
->elementNs
|| !value
)
3924 return E_INVALIDARG
;
3926 EnterCriticalSection( &writer
->cs
);
3928 if (writer
->magic
!= WRITER_MAGIC
)
3930 LeaveCriticalSection( &writer
->cs
);
3931 return E_INVALIDARG
;
3934 if ((hr
= write_element_node( writer
, NULL
, desc
->elementLocalName
, desc
->elementNs
)) != S_OK
) goto done
;
3936 if ((hr
= write_type( writer
, WS_ANY_ELEMENT_TYPE_MAPPING
, desc
->type
, desc
->typeDescription
,
3937 option
, value
, size
)) != S_OK
) goto done
;
3939 hr
= write_endelement_node( writer
);
3942 LeaveCriticalSection( &writer
->cs
);
3946 /**************************************************************************
3947 * WsWriteType [webservices.@]
3949 HRESULT WINAPI
WsWriteType( WS_XML_WRITER
*handle
, WS_TYPE_MAPPING mapping
, WS_TYPE type
,
3950 const void *desc
, WS_WRITE_OPTION option
, const void *value
,
3951 ULONG size
, WS_ERROR
*error
)
3953 struct writer
*writer
= (struct writer
*)handle
;
3956 TRACE( "%p %u %u %p %u %p %u %p\n", handle
, mapping
, type
, desc
, option
, value
,
3958 if (error
) FIXME( "ignoring error parameter\n" );
3960 if (!writer
|| !value
) return E_INVALIDARG
;
3962 EnterCriticalSection( &writer
->cs
);
3964 if (writer
->magic
!= WRITER_MAGIC
)
3966 LeaveCriticalSection( &writer
->cs
);
3967 return E_INVALIDARG
;
3972 case WS_ATTRIBUTE_TYPE_MAPPING
:
3973 if (writer
->state
!= WRITER_STATE_STARTATTRIBUTE
) hr
= WS_E_INVALID_FORMAT
;
3974 else hr
= write_type( writer
, mapping
, type
, desc
, option
, value
, size
);
3977 case WS_ELEMENT_TYPE_MAPPING
:
3978 case WS_ELEMENT_CONTENT_TYPE_MAPPING
:
3979 case WS_ANY_ELEMENT_TYPE_MAPPING
:
3980 hr
= write_type( writer
, mapping
, type
, desc
, option
, value
, size
);
3984 FIXME( "mapping %u not implemented\n", mapping
);
3988 LeaveCriticalSection( &writer
->cs
);
3992 WS_TYPE
map_value_type( WS_VALUE_TYPE type
)
3996 case WS_BOOL_VALUE_TYPE
: return WS_BOOL_TYPE
;
3997 case WS_INT8_VALUE_TYPE
: return WS_INT8_TYPE
;
3998 case WS_INT16_VALUE_TYPE
: return WS_INT16_TYPE
;
3999 case WS_INT32_VALUE_TYPE
: return WS_INT32_TYPE
;
4000 case WS_INT64_VALUE_TYPE
: return WS_INT64_TYPE
;
4001 case WS_UINT8_VALUE_TYPE
: return WS_UINT8_TYPE
;
4002 case WS_UINT16_VALUE_TYPE
: return WS_UINT16_TYPE
;
4003 case WS_UINT32_VALUE_TYPE
: return WS_UINT32_TYPE
;
4004 case WS_UINT64_VALUE_TYPE
: return WS_UINT64_TYPE
;
4005 case WS_FLOAT_VALUE_TYPE
: return WS_FLOAT_TYPE
;
4006 case WS_DOUBLE_VALUE_TYPE
: return WS_DOUBLE_TYPE
;
4007 case WS_DECIMAL_VALUE_TYPE
: return WS_DECIMAL_TYPE
;
4008 case WS_DATETIME_VALUE_TYPE
: return WS_DATETIME_TYPE
;
4009 case WS_TIMESPAN_VALUE_TYPE
: return WS_TIMESPAN_TYPE
;
4010 case WS_GUID_VALUE_TYPE
: return WS_GUID_TYPE
;
4012 FIXME( "unhandled type %u\n", type
);
4017 /**************************************************************************
4018 * WsWriteValue [webservices.@]
4020 HRESULT WINAPI
WsWriteValue( WS_XML_WRITER
*handle
, WS_VALUE_TYPE value_type
, const void *value
,
4021 ULONG size
, WS_ERROR
*error
)
4023 struct writer
*writer
= (struct writer
*)handle
;
4024 WS_TYPE_MAPPING mapping
;
4028 TRACE( "%p %u %p %u %p\n", handle
, value_type
, value
, size
, error
);
4029 if (error
) FIXME( "ignoring error parameter\n" );
4031 if (!writer
|| !value
|| (type
= map_value_type( value_type
)) == ~0u) return E_INVALIDARG
;
4033 EnterCriticalSection( &writer
->cs
);
4035 if (writer
->magic
!= WRITER_MAGIC
)
4037 LeaveCriticalSection( &writer
->cs
);
4038 return E_INVALIDARG
;
4041 switch (writer
->state
)
4043 case WRITER_STATE_STARTATTRIBUTE
:
4044 mapping
= WS_ATTRIBUTE_TYPE_MAPPING
;
4047 case WRITER_STATE_STARTELEMENT
:
4048 mapping
= WS_ELEMENT_TYPE_MAPPING
;
4052 hr
= WS_E_INVALID_FORMAT
;
4055 if (hr
== S_OK
) hr
= write_type( writer
, mapping
, type
, NULL
, WS_WRITE_REQUIRED_VALUE
, value
, size
);
4057 LeaveCriticalSection( &writer
->cs
);
4061 /**************************************************************************
4062 * WsWriteArray [webservices.@]
4064 HRESULT WINAPI
WsWriteArray( WS_XML_WRITER
*handle
, const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4065 WS_VALUE_TYPE value_type
, const void *array
, ULONG size
, ULONG offset
,
4066 ULONG count
, WS_ERROR
*error
)
4068 struct writer
*writer
= (struct writer
*)handle
;
4073 TRACE( "%p %s %s %u %p %u %u %u %p\n", handle
, debugstr_xmlstr(localname
), debugstr_xmlstr(ns
),
4074 value_type
, array
, size
, offset
, count
, error
);
4075 if (error
) FIXME( "ignoring error parameter\n" );
4077 if (!writer
) return E_INVALIDARG
;
4079 EnterCriticalSection( &writer
->cs
);
4081 if (writer
->magic
!= WRITER_MAGIC
)
4083 LeaveCriticalSection( &writer
->cs
);
4084 return E_INVALIDARG
;
4087 if (!writer
->output_type
)
4089 LeaveCriticalSection( &writer
->cs
);
4090 return WS_E_INVALID_OPERATION
;
4093 if (!localname
|| !ns
|| (type
= map_value_type( value_type
)) == ~0u)
4095 LeaveCriticalSection( &writer
->cs
);
4096 return E_INVALIDARG
;
4099 type_size
= get_type_size( type
, NULL
);
4100 if (size
% type_size
|| (offset
+ count
) * type_size
> size
|| (count
&& !array
))
4102 LeaveCriticalSection( &writer
->cs
);
4103 return E_INVALIDARG
;
4106 for (i
= offset
; i
< count
; i
++)
4108 const char *ptr
= (const char *)array
+ (offset
+ i
) * type_size
;
4109 if ((hr
= write_element_node( writer
, NULL
, localname
, ns
)) != S_OK
) goto done
;
4110 if ((hr
= write_type( writer
, WS_ELEMENT_TYPE_MAPPING
, type
, NULL
, WS_WRITE_REQUIRED_POINTER
,
4111 &ptr
, sizeof(ptr
) )) != S_OK
) goto done
;
4112 if ((hr
= write_endelement_node( writer
)) != S_OK
) goto done
;
4116 LeaveCriticalSection( &writer
->cs
);
4120 /**************************************************************************
4121 * WsWriteXmlBuffer [webservices.@]
4123 HRESULT WINAPI
WsWriteXmlBuffer( WS_XML_WRITER
*handle
, WS_XML_BUFFER
*buffer
, WS_ERROR
*error
)
4125 struct writer
*writer
= (struct writer
*)handle
;
4126 struct xmlbuf
*xmlbuf
= (struct xmlbuf
*)buffer
;
4129 TRACE( "%p %p %p\n", handle
, buffer
, error
);
4130 if (error
) FIXME( "ignoring error parameter\n" );
4132 if (!writer
|| !xmlbuf
) return E_INVALIDARG
;
4134 EnterCriticalSection( &writer
->cs
);
4136 if (writer
->magic
!= WRITER_MAGIC
)
4138 LeaveCriticalSection( &writer
->cs
);
4139 return E_INVALIDARG
;
4142 if (xmlbuf
->encoding
!= writer
->output_enc
|| xmlbuf
->charset
!= writer
->output_charset
)
4144 FIXME( "no support for different encoding and/or charset\n" );
4149 if ((hr
= write_flush( writer
)) != S_OK
) goto done
;
4150 if ((hr
= write_grow_buffer( writer
, xmlbuf
->bytes
.length
)) != S_OK
) goto done
;
4151 write_bytes( writer
, xmlbuf
->bytes
.bytes
, xmlbuf
->bytes
.length
);
4154 LeaveCriticalSection( &writer
->cs
);
4158 /**************************************************************************
4159 * WsWriteXmlBufferToBytes [webservices.@]
4161 HRESULT WINAPI
WsWriteXmlBufferToBytes( WS_XML_WRITER
*handle
, WS_XML_BUFFER
*buffer
,
4162 const WS_XML_WRITER_ENCODING
*encoding
,
4163 const WS_XML_WRITER_PROPERTY
*properties
, ULONG count
,
4164 WS_HEAP
*heap
, void **bytes
, ULONG
*size
, WS_ERROR
*error
)
4166 struct writer
*writer
= (struct writer
*)handle
;
4167 struct xmlbuf
*xmlbuf
= (struct xmlbuf
*)buffer
;
4172 TRACE( "%p %p %p %p %u %p %p %p %p\n", handle
, buffer
, encoding
, properties
, count
, heap
,
4173 bytes
, size
, error
);
4174 if (error
) FIXME( "ignoring error parameter\n" );
4176 if (!writer
|| !xmlbuf
|| !heap
|| !bytes
) return E_INVALIDARG
;
4178 if (encoding
&& encoding
->encodingType
!= WS_XML_WRITER_ENCODING_TYPE_TEXT
)
4180 FIXME( "encoding type %u not supported\n", encoding
->encodingType
);
4184 EnterCriticalSection( &writer
->cs
);
4186 if (writer
->magic
!= WRITER_MAGIC
)
4188 LeaveCriticalSection( &writer
->cs
);
4189 return E_INVALIDARG
;
4192 for (i
= 0; i
< count
; i
++)
4194 hr
= prop_set( writer
->prop
, writer
->prop_count
, properties
[i
].id
, properties
[i
].value
,
4195 properties
[i
].valueSize
);
4196 if (hr
!= S_OK
) goto done
;
4199 if (!(buf
= ws_alloc( heap
, xmlbuf
->bytes
.length
))) hr
= WS_E_QUOTA_EXCEEDED
;
4202 memcpy( buf
, xmlbuf
->bytes
.bytes
, xmlbuf
->bytes
.length
);
4204 *size
= xmlbuf
->bytes
.length
;
4208 LeaveCriticalSection( &writer
->cs
);
4212 /**************************************************************************
4213 * WsWriteXmlnsAttribute [webservices.@]
4215 HRESULT WINAPI
WsWriteXmlnsAttribute( WS_XML_WRITER
*handle
, const WS_XML_STRING
*prefix
,
4216 const WS_XML_STRING
*ns
, BOOL single
, WS_ERROR
*error
)
4218 struct writer
*writer
= (struct writer
*)handle
;
4221 TRACE( "%p %s %s %d %p\n", handle
, debugstr_xmlstr(prefix
), debugstr_xmlstr(ns
),
4223 if (error
) FIXME( "ignoring error parameter\n" );
4225 if (!writer
|| !ns
) return E_INVALIDARG
;
4227 EnterCriticalSection( &writer
->cs
);
4229 if (writer
->magic
!= WRITER_MAGIC
)
4231 LeaveCriticalSection( &writer
->cs
);
4232 return E_INVALIDARG
;
4235 if (writer
->state
!= WRITER_STATE_STARTELEMENT
)
4237 LeaveCriticalSection( &writer
->cs
);
4238 return WS_E_INVALID_OPERATION
;
4241 if (!namespace_in_scope( &writer
->current
->hdr
, prefix
, ns
))
4242 hr
= add_namespace_attribute( writer
, prefix
, ns
, single
);
4244 LeaveCriticalSection( &writer
->cs
);
4248 static HRESULT
write_qualified_name( struct writer
*writer
, const WS_XML_STRING
*prefix
,
4249 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
)
4251 WS_XML_QNAME_TEXT qname
= {{WS_XML_TEXT_TYPE_QNAME
}};
4254 if ((hr
= write_flush( writer
)) != S_OK
) return hr
;
4255 if (!prefix
&& ((hr
= find_prefix( writer
, ns
, &prefix
)) != S_OK
)) return hr
;
4257 qname
.prefix
= (WS_XML_STRING
*)prefix
;
4258 qname
.localName
= (WS_XML_STRING
*)localname
;
4259 qname
.ns
= (WS_XML_STRING
*)ns
;
4261 if ((hr
= write_add_text_node( writer
, &qname
.text
)) != S_OK
) return hr
;
4262 return write_text( writer
, ((const WS_XML_TEXT_NODE
*)writer
->current
)->text
, 0 );
4265 /**************************************************************************
4266 * WsWriteQualifiedName [webservices.@]
4268 HRESULT WINAPI
WsWriteQualifiedName( WS_XML_WRITER
*handle
, const WS_XML_STRING
*prefix
,
4269 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4272 struct writer
*writer
= (struct writer
*)handle
;
4275 TRACE( "%p %s %s %s %p\n", handle
, debugstr_xmlstr(prefix
), debugstr_xmlstr(localname
),
4276 debugstr_xmlstr(ns
), error
);
4277 if (error
) FIXME( "ignoring error parameter\n" );
4279 if (!writer
) return E_INVALIDARG
;
4281 EnterCriticalSection( &writer
->cs
);
4283 if (writer
->magic
!= WRITER_MAGIC
)
4285 LeaveCriticalSection( &writer
->cs
);
4286 return E_INVALIDARG
;
4289 if (!writer
->output_type
)
4291 LeaveCriticalSection( &writer
->cs
);
4292 return WS_E_INVALID_OPERATION
;
4295 if (writer
->state
!= WRITER_STATE_STARTELEMENT
)
4297 LeaveCriticalSection( &writer
->cs
);
4298 return WS_E_INVALID_FORMAT
;
4301 if (!localname
|| (!prefix
&& !ns
))
4303 LeaveCriticalSection( &writer
->cs
);
4304 return E_INVALIDARG
;
4307 hr
= write_qualified_name( writer
, prefix
, localname
, ns
);
4309 LeaveCriticalSection( &writer
->cs
);
4313 static HRESULT
write_move_to( struct writer
*writer
, WS_MOVE_TO move
, BOOL
*found
)
4315 BOOL success
= FALSE
;
4316 struct node
*node
= writer
->current
;
4320 case WS_MOVE_TO_ROOT_ELEMENT
:
4321 success
= move_to_root_element( writer
->root
, &node
);
4324 case WS_MOVE_TO_NEXT_ELEMENT
:
4325 success
= move_to_next_element( &node
);
4328 case WS_MOVE_TO_PREVIOUS_ELEMENT
:
4329 success
= move_to_prev_element( &node
);
4332 case WS_MOVE_TO_CHILD_ELEMENT
:
4333 success
= move_to_child_element( &node
);
4336 case WS_MOVE_TO_END_ELEMENT
:
4337 success
= move_to_end_element( &node
);
4340 case WS_MOVE_TO_PARENT_ELEMENT
:
4341 success
= move_to_parent_element( &node
);
4344 case WS_MOVE_TO_FIRST_NODE
:
4345 success
= move_to_first_node( &node
);
4348 case WS_MOVE_TO_NEXT_NODE
:
4349 success
= move_to_next_node( &node
);
4352 case WS_MOVE_TO_PREVIOUS_NODE
:
4353 success
= move_to_prev_node( &node
);
4356 case WS_MOVE_TO_CHILD_NODE
:
4357 success
= move_to_child_node( &node
);
4360 case WS_MOVE_TO_BOF
:
4361 success
= move_to_bof( writer
->root
, &node
);
4364 case WS_MOVE_TO_EOF
:
4365 success
= move_to_eof( writer
->root
, &node
);
4369 FIXME( "unhandled move %u\n", move
);
4373 if (success
&& node
== writer
->root
) return E_INVALIDARG
;
4374 writer
->current
= node
;
4381 return success
? S_OK
: WS_E_INVALID_FORMAT
;
4384 /**************************************************************************
4385 * WsMoveWriter [webservices.@]
4387 HRESULT WINAPI
WsMoveWriter( WS_XML_WRITER
*handle
, WS_MOVE_TO move
, BOOL
*found
, WS_ERROR
*error
)
4389 struct writer
*writer
= (struct writer
*)handle
;
4392 TRACE( "%p %u %p %p\n", handle
, move
, found
, error
);
4393 if (error
) FIXME( "ignoring error parameter\n" );
4395 if (!writer
) return E_INVALIDARG
;
4397 EnterCriticalSection( &writer
->cs
);
4399 if (writer
->magic
!= WRITER_MAGIC
)
4401 LeaveCriticalSection( &writer
->cs
);
4402 return E_INVALIDARG
;
4405 if (!writer
->output_type
)
4407 LeaveCriticalSection( &writer
->cs
);
4408 return WS_E_INVALID_OPERATION
;
4411 hr
= write_move_to( writer
, move
, found
);
4413 LeaveCriticalSection( &writer
->cs
);
4417 /**************************************************************************
4418 * WsGetWriterPosition [webservices.@]
4420 HRESULT WINAPI
WsGetWriterPosition( WS_XML_WRITER
*handle
, WS_XML_NODE_POSITION
*pos
, WS_ERROR
*error
)
4422 struct writer
*writer
= (struct writer
*)handle
;
4424 TRACE( "%p %p %p\n", handle
, pos
, error
);
4425 if (error
) FIXME( "ignoring error parameter\n" );
4427 if (!writer
|| !pos
) return E_INVALIDARG
;
4429 EnterCriticalSection( &writer
->cs
);
4431 if (writer
->magic
!= WRITER_MAGIC
)
4433 LeaveCriticalSection( &writer
->cs
);
4434 return E_INVALIDARG
;
4437 if (!writer
->output_type
)
4439 LeaveCriticalSection( &writer
->cs
);
4440 return WS_E_INVALID_OPERATION
;
4443 pos
->buffer
= (WS_XML_BUFFER
*)writer
->output_buf
;
4444 pos
->node
= writer
->current
;
4446 LeaveCriticalSection( &writer
->cs
);
4450 /**************************************************************************
4451 * WsSetWriterPosition [webservices.@]
4453 HRESULT WINAPI
WsSetWriterPosition( WS_XML_WRITER
*handle
, const WS_XML_NODE_POSITION
*pos
, WS_ERROR
*error
)
4455 struct writer
*writer
= (struct writer
*)handle
;
4457 TRACE( "%p %p %p\n", handle
, pos
, error
);
4458 if (error
) FIXME( "ignoring error parameter\n" );
4460 if (!writer
|| !pos
) return E_INVALIDARG
;
4462 EnterCriticalSection( &writer
->cs
);
4464 if (writer
->magic
!= WRITER_MAGIC
|| (struct xmlbuf
*)pos
->buffer
!= writer
->output_buf
)
4466 LeaveCriticalSection( &writer
->cs
);
4467 return E_INVALIDARG
;
4470 if (!writer
->output_type
)
4472 LeaveCriticalSection( &writer
->cs
);
4473 return WS_E_INVALID_OPERATION
;
4476 writer
->current
= pos
->node
;
4478 LeaveCriticalSection( &writer
->cs
);
4482 static HRESULT
write_add_comment_node( struct writer
*writer
, const WS_XML_STRING
*value
)
4484 struct node
*node
, *parent
;
4485 WS_XML_COMMENT_NODE
*comment
;
4487 if (!(parent
= find_parent( writer
))) return WS_E_INVALID_FORMAT
;
4488 if (!(node
= alloc_node( WS_XML_NODE_TYPE_COMMENT
))) return E_OUTOFMEMORY
;
4489 comment
= (WS_XML_COMMENT_NODE
*)node
;
4491 if (value
->length
&& !(comment
->value
.bytes
= heap_alloc( value
->length
)))
4494 return E_OUTOFMEMORY
;
4496 memcpy( comment
->value
.bytes
, value
->bytes
, value
->length
);
4497 comment
->value
.length
= value
->length
;
4499 write_insert_node( writer
, parent
, node
);
4503 static HRESULT
write_comment_text( struct writer
*writer
)
4505 const WS_XML_COMMENT_NODE
*comment
= (const WS_XML_COMMENT_NODE
*)writer
->current
;
4508 if ((hr
= write_grow_buffer( writer
, comment
->value
.length
+ 7 )) != S_OK
) return hr
;
4509 write_bytes( writer
, (const BYTE
*)"<!--", 4 );
4510 write_bytes( writer
, comment
->value
.bytes
, comment
->value
.length
);
4511 write_bytes( writer
, (const BYTE
*)"-->", 3 );
4515 static HRESULT
write_comment_bin( struct writer
*writer
)
4517 const WS_XML_COMMENT_NODE
*comment
= (const WS_XML_COMMENT_NODE
*)writer
->current
;
4520 if ((hr
= write_grow_buffer( writer
, 1 )) != S_OK
) return hr
;
4521 write_char( writer
, RECORD_COMMENT
);
4522 return write_string( writer
, comment
->value
.bytes
, comment
->value
.length
);
4525 static HRESULT
write_comment( struct writer
*writer
)
4527 switch (writer
->output_enc
)
4529 case WS_XML_WRITER_ENCODING_TYPE_TEXT
: return write_comment_text( writer
);
4530 case WS_XML_WRITER_ENCODING_TYPE_BINARY
: return write_comment_bin( writer
);
4532 ERR( "unhandled encoding %u\n", writer
->output_enc
);
4533 return WS_E_NOT_SUPPORTED
;
4537 static HRESULT
write_comment_node( struct writer
*writer
, const WS_XML_STRING
*value
)
4540 if ((hr
= write_flush( writer
)) != S_OK
) return hr
;
4541 if ((hr
= write_add_comment_node( writer
, value
)) != S_OK
) return hr
;
4542 if ((hr
= write_comment( writer
)) != S_OK
) return hr
;
4543 writer
->state
= WRITER_STATE_COMMENT
;
4547 static HRESULT
write_set_attributes( struct writer
*writer
, WS_XML_ATTRIBUTE
**attrs
, ULONG count
)
4552 for (i
= 0; i
< count
; i
++)
4554 const WS_XML_STRING
*prefix
= attrs
[i
]->prefix
;
4555 const WS_XML_STRING
*localname
= attrs
[i
]->localName
;
4556 const WS_XML_STRING
*ns
= attrs
[i
]->ns
;
4557 BOOL single
= attrs
[i
]->singleQuote
;
4559 if (attrs
[i
]->isXmlNs
)
4561 if ((hr
= add_namespace_attribute( writer
, prefix
, ns
, single
)) != S_OK
) return hr
;
4565 if ((hr
= write_add_attribute( writer
, prefix
, localname
, ns
, single
)) != S_OK
) return hr
;
4566 if ((hr
= write_set_attribute_value( writer
, attrs
[i
]->value
)) != S_OK
) return hr
;
4572 static HRESULT
write_node( struct writer
*writer
, const WS_XML_NODE
*node
)
4576 switch (node
->nodeType
)
4578 case WS_XML_NODE_TYPE_ELEMENT
:
4580 const WS_XML_ELEMENT_NODE
*elem
= (const WS_XML_ELEMENT_NODE
*)node
;
4581 if ((hr
= write_element_node( writer
, elem
->prefix
, elem
->localName
, elem
->ns
)) != S_OK
) return hr
;
4582 return write_set_attributes( writer
, elem
->attributes
, elem
->attributeCount
);
4584 case WS_XML_NODE_TYPE_TEXT
:
4586 const WS_XML_TEXT_NODE
*text
= (const WS_XML_TEXT_NODE
*)node
;
4587 return write_text_node( writer
, text
->text
);
4589 case WS_XML_NODE_TYPE_END_ELEMENT
:
4590 return write_endelement_node( writer
);
4592 case WS_XML_NODE_TYPE_COMMENT
:
4594 const WS_XML_COMMENT_NODE
*comment
= (const WS_XML_COMMENT_NODE
*)node
;
4595 return write_comment_node( writer
, &comment
->value
);
4597 case WS_XML_NODE_TYPE_CDATA
:
4598 return write_cdata_node( writer
);
4600 case WS_XML_NODE_TYPE_END_CDATA
:
4601 return write_endcdata_node( writer
);
4603 case WS_XML_NODE_TYPE_EOF
:
4604 case WS_XML_NODE_TYPE_BOF
:
4608 WARN( "unknown node type %u\n", node
->nodeType
);
4609 return E_INVALIDARG
;
4613 /**************************************************************************
4614 * WsWriteNode [webservices.@]
4616 HRESULT WINAPI
WsWriteNode( WS_XML_WRITER
*handle
, const WS_XML_NODE
*node
, WS_ERROR
*error
)
4618 struct writer
*writer
= (struct writer
*)handle
;
4621 TRACE( "%p %p %p\n", handle
, node
, error
);
4622 if (error
) FIXME( "ignoring error parameter\n" );
4624 if (!writer
|| !node
) return E_INVALIDARG
;
4626 EnterCriticalSection( &writer
->cs
);
4628 if (writer
->magic
!= WRITER_MAGIC
)
4630 LeaveCriticalSection( &writer
->cs
);
4631 return E_INVALIDARG
;
4634 if (!writer
->output_type
)
4636 LeaveCriticalSection( &writer
->cs
);
4637 return WS_E_INVALID_OPERATION
;
4640 hr
= write_node( writer
, node
);
4642 LeaveCriticalSection( &writer
->cs
);
4646 static HRESULT
write_tree_node( struct writer
*writer
)
4650 switch (node_type( writer
->current
))
4652 case WS_XML_NODE_TYPE_ELEMENT
:
4653 if (writer
->state
== WRITER_STATE_STARTELEMENT
&& (hr
= write_endstartelement( writer
)) != S_OK
)
4655 if ((hr
= write_startelement( writer
)) != S_OK
) return hr
;
4656 writer
->state
= WRITER_STATE_STARTELEMENT
;
4659 case WS_XML_NODE_TYPE_TEXT
:
4660 if (writer
->state
== WRITER_STATE_STARTELEMENT
&& (hr
= write_endstartelement( writer
)) != S_OK
)
4662 if ((hr
= write_text( writer
, ((const WS_XML_TEXT_NODE
*)writer
->current
)->text
, 0 )) != S_OK
) return hr
;
4663 writer
->state
= WRITER_STATE_TEXT
;
4666 case WS_XML_NODE_TYPE_END_ELEMENT
:
4667 if ((hr
= write_close_element( writer
, writer
->current
->parent
)) != S_OK
) return hr
;
4668 writer
->state
= WRITER_STATE_ENDELEMENT
;
4671 case WS_XML_NODE_TYPE_COMMENT
:
4672 if (writer
->state
== WRITER_STATE_STARTELEMENT
&& (hr
= write_endstartelement( writer
)) != S_OK
)
4674 if ((hr
= write_comment( writer
)) != S_OK
) return hr
;
4675 writer
->state
= WRITER_STATE_COMMENT
;
4678 case WS_XML_NODE_TYPE_CDATA
:
4679 if (writer
->state
== WRITER_STATE_STARTELEMENT
&& (hr
= write_endstartelement( writer
)) != S_OK
)
4681 if ((hr
= write_cdata( writer
)) != S_OK
) return hr
;
4682 writer
->state
= WRITER_STATE_STARTCDATA
;
4685 case WS_XML_NODE_TYPE_END_CDATA
:
4686 if ((hr
= write_endcdata( writer
)) != S_OK
) return hr
;
4687 writer
->state
= WRITER_STATE_ENDCDATA
;
4690 case WS_XML_NODE_TYPE_EOF
:
4691 case WS_XML_NODE_TYPE_BOF
:
4695 ERR( "unknown node type %u\n", node_type(writer
->current
) );
4696 return E_INVALIDARG
;
4700 static HRESULT
write_tree( struct writer
*writer
)
4704 if ((hr
= write_tree_node( writer
)) != S_OK
) return hr
;
4707 if (node_type( writer
->current
) == WS_XML_NODE_TYPE_EOF
) break;
4708 if (move_to_child_node( &writer
->current
))
4710 if ((hr
= write_tree_node( writer
)) != S_OK
) return hr
;
4713 if (move_to_next_node( &writer
->current
))
4715 if ((hr
= write_tree_node( writer
)) != S_OK
) return hr
;
4718 if (!move_to_parent_node( &writer
->current
) || !move_to_next_node( &writer
->current
))
4720 ERR( "invalid tree\n" );
4721 return WS_E_INVALID_FORMAT
;
4723 if ((hr
= write_tree_node( writer
)) != S_OK
) return hr
;
4728 static void write_rewind( struct writer
*writer
)
4730 writer
->write_pos
= 0;
4731 writer
->current
= writer
->root
;
4732 writer
->state
= WRITER_STATE_INITIAL
;
4735 /**************************************************************************
4736 * WsCopyNode [webservices.@]
4738 HRESULT WINAPI
WsCopyNode( WS_XML_WRITER
*handle
, WS_XML_READER
*reader
, WS_ERROR
*error
)
4740 struct writer
*writer
= (struct writer
*)handle
;
4741 struct node
*parent
, *current
, *node
= NULL
;
4744 TRACE( "%p %p %p\n", handle
, reader
, error
);
4745 if (error
) FIXME( "ignoring error parameter\n" );
4747 if (!writer
) return E_INVALIDARG
;
4749 EnterCriticalSection( &writer
->cs
);
4751 if (writer
->magic
!= WRITER_MAGIC
)
4753 LeaveCriticalSection( &writer
->cs
);
4754 return E_INVALIDARG
;
4757 if (!(parent
= find_parent( writer
)))
4759 LeaveCriticalSection( &writer
->cs
);
4760 return WS_E_INVALID_FORMAT
;
4763 if ((hr
= copy_node( reader
, writer
->output_enc
, &node
)) != S_OK
) goto done
;
4764 current
= writer
->current
;
4765 write_insert_node( writer
, parent
, node
);
4767 write_rewind( writer
);
4768 if ((hr
= write_tree( writer
)) != S_OK
) goto done
;
4769 writer
->current
= current
;
4771 WsMoveReader( reader
, WS_MOVE_TO_NEXT_NODE
, NULL
, NULL
);
4774 LeaveCriticalSection( &writer
->cs
);
4778 static HRESULT
write_param( struct writer
*writer
, const WS_FIELD_DESCRIPTION
*desc
, const void *value
)
4780 return write_type_field( writer
, desc
, value
, 0 );
4783 static ULONG
get_array_len( const WS_PARAMETER_DESCRIPTION
*params
, ULONG count
, ULONG index
, const void **args
)
4786 for (i
= 0; i
< count
; i
++)
4788 if (params
[i
].inputMessageIndex
!= index
|| params
[i
].parameterType
!= WS_PARAMETER_TYPE_ARRAY_COUNT
)
4790 if (args
[i
]) ret
= *(const ULONG
*)args
[i
];
4796 static HRESULT
write_param_array( struct writer
*writer
, const WS_FIELD_DESCRIPTION
*desc
, const void *value
,
4799 return write_type_array( writer
, desc
, value
, len
);
4802 HRESULT
write_input_params( WS_XML_WRITER
*handle
, const WS_ELEMENT_DESCRIPTION
*desc
,
4803 const WS_PARAMETER_DESCRIPTION
*params
, ULONG count
, const void **args
)
4805 struct writer
*writer
= (struct writer
*)handle
;
4806 const WS_STRUCT_DESCRIPTION
*desc_struct
;
4807 const WS_FIELD_DESCRIPTION
*desc_field
;
4811 if (desc
->type
!= WS_STRUCT_TYPE
|| !(desc_struct
= desc
->typeDescription
)) return E_INVALIDARG
;
4813 EnterCriticalSection( &writer
->cs
);
4815 if (writer
->magic
!= WRITER_MAGIC
)
4817 LeaveCriticalSection( &writer
->cs
);
4818 return E_INVALIDARG
;
4821 if ((hr
= write_element_node( writer
, NULL
, desc
->elementLocalName
, desc
->elementNs
)) != S_OK
) goto done
;
4823 for (i
= 0; i
< count
; i
++)
4825 if (params
[i
].inputMessageIndex
== INVALID_PARAMETER_INDEX
) continue;
4826 if (params
[i
].parameterType
== WS_PARAMETER_TYPE_MESSAGES
)
4828 FIXME( "messages type not supported\n" );
4832 if ((hr
= get_param_desc( desc_struct
, params
[i
].inputMessageIndex
, &desc_field
)) != S_OK
) goto done
;
4833 if (params
[i
].parameterType
== WS_PARAMETER_TYPE_NORMAL
)
4835 if ((hr
= write_param( writer
, desc_field
, args
[i
] )) != S_OK
) goto done
;
4837 else if (params
[i
].parameterType
== WS_PARAMETER_TYPE_ARRAY
)
4839 const void *ptr
= *(const void **)args
[i
];
4840 ULONG len
= get_array_len( params
, count
, params
[i
].inputMessageIndex
, args
);
4841 if ((hr
= write_param_array( writer
, desc_field
, ptr
, len
)) != S_OK
) goto done
;
4845 hr
= write_endelement_node( writer
);
4848 LeaveCriticalSection( &writer
->cs
);
4852 HRESULT
writer_set_lookup( WS_XML_WRITER
*handle
, BOOL enable
)
4854 struct writer
*writer
= (struct writer
*)handle
;
4856 EnterCriticalSection( &writer
->cs
);
4858 if (writer
->magic
!= WRITER_MAGIC
)
4860 LeaveCriticalSection( &writer
->cs
);
4861 return E_INVALIDARG
;
4864 writer
->dict_do_lookup
= enable
;
4866 LeaveCriticalSection( &writer
->cs
);
4870 HRESULT
writer_set_dict_callback( WS_XML_WRITER
*handle
, WS_DYNAMIC_STRING_CALLBACK cb
, void *state
)
4872 struct writer
*writer
= (struct writer
*)handle
;
4874 EnterCriticalSection( &writer
->cs
);
4876 if (writer
->magic
!= WRITER_MAGIC
)
4878 LeaveCriticalSection( &writer
->cs
);
4879 return E_INVALIDARG
;
4882 writer
->dict_cb
= cb
;
4883 writer
->dict_cb_state
= state
;
4885 LeaveCriticalSection( &writer
->cs
);