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
24 #include "webservices.h"
26 #include "wine/debug.h"
27 #include "wine/list.h"
28 #include "webservices_private.h"
30 WINE_DEFAULT_DEBUG_CHANNEL(webservices
);
32 const char *debugstr_xmlstr( const WS_XML_STRING
*str
)
34 if (!str
) return "(null)";
35 return debugstr_an( (const char *)str
->bytes
, str
->length
);
38 ULONG
prop_size( const struct prop_desc
*desc
, ULONG count
)
40 ULONG i
, ret
= count
* sizeof(struct prop
);
41 for (i
= 0; i
< count
; i
++) ret
+= desc
[i
].size
;
45 void prop_init( const struct prop_desc
*desc
, ULONG count
, struct prop
*prop
, void *data
)
49 for (i
= 0; i
< count
; i
++)
52 prop
[i
].size
= desc
[i
].size
;
53 prop
[i
].readonly
= desc
[i
].readonly
;
54 prop
[i
].writeonly
= desc
[i
].writeonly
;
59 HRESULT
prop_set( const struct prop
*prop
, ULONG count
, ULONG id
, const void *value
, ULONG size
)
61 if (id
>= count
|| size
!= prop
[id
].size
|| prop
[id
].readonly
) return E_INVALIDARG
;
62 memcpy( prop
[id
].value
, value
, size
);
66 HRESULT
prop_get( const struct prop
*prop
, ULONG count
, ULONG id
, void *buf
, ULONG size
)
68 if (id
>= count
|| size
!= prop
[id
].size
|| prop
[id
].writeonly
) return E_INVALIDARG
;
69 memcpy( buf
, prop
[id
].value
, prop
[id
].size
);
73 struct node
*alloc_node( WS_XML_NODE_TYPE type
)
77 if (!(ret
= heap_alloc_zero( sizeof(*ret
) ))) return NULL
;
78 ret
->hdr
.node
.nodeType
= type
;
79 list_init( &ret
->entry
);
80 list_init( &ret
->children
);
84 void free_attribute( WS_XML_ATTRIBUTE
*attr
)
87 heap_free( attr
->prefix
);
88 heap_free( attr
->localName
);
89 heap_free( attr
->ns
);
90 heap_free( attr
->value
);
94 void free_node( struct node
*node
)
97 switch (node_type( node
))
99 case WS_XML_NODE_TYPE_ELEMENT
:
101 WS_XML_ELEMENT_NODE
*elem
= &node
->hdr
;
104 for (i
= 0; i
< elem
->attributeCount
; i
++) free_attribute( elem
->attributes
[i
] );
105 heap_free( elem
->attributes
);
106 heap_free( elem
->prefix
);
107 heap_free( elem
->localName
);
108 heap_free( elem
->ns
);
111 case WS_XML_NODE_TYPE_TEXT
:
113 WS_XML_TEXT_NODE
*text
= (WS_XML_TEXT_NODE
*)node
;
114 heap_free( text
->text
);
117 case WS_XML_NODE_TYPE_COMMENT
:
119 WS_XML_COMMENT_NODE
*comment
= (WS_XML_COMMENT_NODE
*)node
;
120 heap_free( comment
->value
.bytes
);
123 case WS_XML_NODE_TYPE_CDATA
:
124 case WS_XML_NODE_TYPE_END_CDATA
:
125 case WS_XML_NODE_TYPE_END_ELEMENT
:
126 case WS_XML_NODE_TYPE_EOF
:
127 case WS_XML_NODE_TYPE_BOF
:
131 ERR( "unhandled type %u\n", node_type( node
) );
137 void destroy_nodes( struct node
*node
)
142 while ((ptr
= list_head( &node
->children
)))
144 struct node
*child
= LIST_ENTRY( ptr
, struct node
, entry
);
145 list_remove( &child
->entry
);
146 destroy_nodes( child
);
151 static WS_XML_ATTRIBUTE
*dup_attribute( const WS_XML_ATTRIBUTE
*src
)
153 WS_XML_ATTRIBUTE
*dst
;
154 const WS_XML_STRING
*prefix
= src
->prefix
;
155 const WS_XML_STRING
*localname
= src
->localName
;
156 const WS_XML_STRING
*ns
= src
->localName
;
157 const WS_XML_TEXT
*text
= src
->value
;
159 if (!(dst
= heap_alloc( sizeof(*dst
) ))) return NULL
;
160 dst
->singleQuote
= src
->singleQuote
;
161 dst
->isXmlNs
= src
->isXmlNs
;
163 if (!prefix
) dst
->prefix
= NULL
;
164 else if (!(dst
->prefix
= alloc_xml_string( prefix
->bytes
, prefix
->length
))) goto error
;
165 if (!(dst
->localName
= alloc_xml_string( localname
->bytes
, localname
->length
))) goto error
;
166 if (!(dst
->ns
= alloc_xml_string( ns
->bytes
, ns
->length
))) goto error
;
170 WS_XML_UTF8_TEXT
*utf8
;
171 const WS_XML_UTF8_TEXT
*utf8_src
= (const WS_XML_UTF8_TEXT
*)text
;
172 if (!(utf8
= alloc_utf8_text( utf8_src
->value
.bytes
, utf8_src
->value
.length
))) goto error
;
173 dst
->value
= &utf8
->text
;
179 free_attribute( dst
);
183 static WS_XML_ATTRIBUTE
**dup_attributes( WS_XML_ATTRIBUTE
* const *src
, ULONG count
)
185 WS_XML_ATTRIBUTE
**dst
;
188 if (!(dst
= heap_alloc( sizeof(*dst
) * count
))) return NULL
;
189 for (i
= 0; i
< count
; i
++)
191 if (!(dst
[i
] = dup_attribute( src
[i
] )))
193 for (; i
> 0; i
--) free_attribute( dst
[i
- 1] );
201 static struct node
*dup_element_node( const WS_XML_ELEMENT_NODE
*src
)
204 WS_XML_ELEMENT_NODE
*dst
;
205 ULONG count
= src
->attributeCount
;
206 WS_XML_ATTRIBUTE
**attrs
= src
->attributes
;
207 const WS_XML_STRING
*prefix
= (src
->prefix
&& src
->prefix
->length
) ? src
->prefix
: NULL
;
208 const WS_XML_STRING
*localname
= src
->localName
;
209 const WS_XML_STRING
*ns
= src
->ns
;
211 if (!(node
= alloc_node( WS_XML_NODE_TYPE_ELEMENT
))) return NULL
;
214 if (count
&& !(dst
->attributes
= dup_attributes( attrs
, count
))) goto error
;
215 dst
->attributeCount
= count
;
217 if (prefix
&& !(dst
->prefix
= alloc_xml_string( prefix
->bytes
, prefix
->length
))) goto error
;
218 if (localname
&& !(dst
->localName
= alloc_xml_string( localname
->bytes
, localname
->length
))) goto error
;
219 if (ns
&& !(dst
->ns
= alloc_xml_string( ns
->bytes
, ns
->length
))) goto error
;
227 static struct node
*dup_text_node( const WS_XML_TEXT_NODE
*src
)
230 WS_XML_TEXT_NODE
*dst
;
232 if (!(node
= alloc_node( WS_XML_NODE_TYPE_TEXT
))) return NULL
;
233 dst
= (WS_XML_TEXT_NODE
*)node
;
237 WS_XML_UTF8_TEXT
*utf8
;
238 const WS_XML_UTF8_TEXT
*utf8_src
= (const WS_XML_UTF8_TEXT
*)src
->text
;
239 if (!(utf8
= alloc_utf8_text( utf8_src
->value
.bytes
, utf8_src
->value
.length
)))
244 dst
->text
= &utf8
->text
;
249 static struct node
*dup_comment_node( const WS_XML_COMMENT_NODE
*src
)
252 WS_XML_COMMENT_NODE
*dst
;
254 if (!(node
= alloc_node( WS_XML_NODE_TYPE_COMMENT
))) return NULL
;
255 dst
= (WS_XML_COMMENT_NODE
*)node
;
257 if (src
->value
.length
&& !(dst
->value
.bytes
= heap_alloc( src
->value
.length
)))
262 memcpy( dst
->value
.bytes
, src
->value
.bytes
, src
->value
.length
);
263 dst
->value
.length
= src
->value
.length
;
267 static struct node
*dup_node( const struct node
*src
)
269 switch (node_type( src
))
271 case WS_XML_NODE_TYPE_ELEMENT
:
272 return dup_element_node( &src
->hdr
);
274 case WS_XML_NODE_TYPE_TEXT
:
275 return dup_text_node( (const WS_XML_TEXT_NODE
*)src
);
277 case WS_XML_NODE_TYPE_COMMENT
:
278 return dup_comment_node( (const WS_XML_COMMENT_NODE
*)src
);
280 case WS_XML_NODE_TYPE_CDATA
:
281 case WS_XML_NODE_TYPE_END_CDATA
:
282 case WS_XML_NODE_TYPE_END_ELEMENT
:
283 case WS_XML_NODE_TYPE_EOF
:
284 case WS_XML_NODE_TYPE_BOF
:
285 return alloc_node( node_type( src
) );
288 ERR( "unhandled type %u\n", node_type( src
) );
294 static HRESULT
dup_tree( struct node
**dst
, const struct node
*src
)
297 const struct node
*child
;
299 if (!*dst
&& !(*dst
= dup_node( src
))) return E_OUTOFMEMORY
;
302 LIST_FOR_EACH_ENTRY( child
, &src
->children
, struct node
, entry
)
304 HRESULT hr
= E_OUTOFMEMORY
;
305 struct node
*new_child
;
307 if (!(new_child
= dup_node( child
)) || (hr
= dup_tree( &new_child
, child
)) != S_OK
)
309 destroy_nodes( *dst
);
312 new_child
->parent
= parent
;
313 list_add_tail( &parent
->children
, &new_child
->entry
);
318 static const struct prop_desc reader_props
[] =
320 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_MAX_DEPTH */
321 { sizeof(BOOL
), FALSE
}, /* WS_XML_READER_PROPERTY_ALLOW_FRAGMENT */
322 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_MAX_ATTRIBUTES */
323 { sizeof(BOOL
), FALSE
}, /* WS_XML_READER_PROPERTY_READ_DECLARATION */
324 { sizeof(WS_CHARSET
), FALSE
}, /* WS_XML_READER_PROPERTY_CHARSET */
325 { sizeof(ULONGLONG
), TRUE
}, /* WS_XML_READER_PROPERTY_ROW */
326 { sizeof(ULONGLONG
), TRUE
}, /* WS_XML_READER_PROPERTY_COLUMN */
327 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_UTF8_TRIM_SIZE */
328 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_STREAM_BUFFER_SIZE */
329 { sizeof(BOOL
), TRUE
}, /* WS_XML_READER_PROPERTY_IN_ATTRIBUTE */
330 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_STREAM_MAX_ROOT_MIME_PART_SIZE */
331 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_STREAM_MAX_MIME_HEADERS_SIZE */
332 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_MAX_MIME_PARTS */
333 { sizeof(BOOL
), FALSE
}, /* WS_XML_READER_PROPERTY_ALLOW_INVALID_CHARACTER_REFERENCES */
334 { sizeof(ULONG
), FALSE
}, /* WS_XML_READER_PROPERTY_MAX_NAMESPACES */
339 READER_STATE_INITIAL
,
341 READER_STATE_STARTELEMENT
,
342 READER_STATE_STARTATTRIBUTE
,
343 READER_STATE_STARTCDATA
,
346 READER_STATE_ENDELEMENT
,
347 READER_STATE_ENDCDATA
,
348 READER_STATE_COMMENT
,
364 const unsigned char *read_bufptr
;
365 enum reader_state state
;
367 struct node
*current
;
370 struct prefix
*prefixes
;
372 ULONG nb_prefixes_allocated
;
373 WS_XML_READER_ENCODING_TYPE input_enc
;
374 WS_XML_READER_INPUT_TYPE input_type
;
375 struct xmlbuf
*input_buf
;
376 const unsigned char *input_data
;
378 ULONG text_conv_offset
;
380 struct prop prop
[sizeof(reader_props
)/sizeof(reader_props
[0])];
383 #define READER_MAGIC (('R' << 24) | ('E' << 16) | ('A' << 8) | 'D')
385 static struct reader
*alloc_reader(void)
387 static const ULONG count
= sizeof(reader_props
)/sizeof(reader_props
[0]);
389 ULONG size
= sizeof(*ret
) + prop_size( reader_props
, count
);
391 if (!(ret
= heap_alloc_zero( size
))) return NULL
;
392 if (!(ret
->prefixes
= heap_alloc_zero( sizeof(*ret
->prefixes
) )))
397 ret
->nb_prefixes
= ret
->nb_prefixes_allocated
= 1;
399 ret
->magic
= READER_MAGIC
;
400 InitializeCriticalSection( &ret
->cs
);
401 ret
->cs
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": reader.cs");
403 prop_init( reader_props
, count
, ret
->prop
, &ret
[1] );
404 ret
->prop_count
= count
;
408 static void clear_prefixes( struct prefix
*prefixes
, ULONG count
)
411 for (i
= 0; i
< count
; i
++)
413 heap_free( prefixes
[i
].str
.bytes
);
414 prefixes
[i
].str
.bytes
= NULL
;
415 prefixes
[i
].str
.length
= 0;
417 heap_free( prefixes
[i
].ns
.bytes
);
418 prefixes
[i
].ns
.bytes
= NULL
;
419 prefixes
[i
].ns
.length
= 0;
423 static HRESULT
set_prefix( struct prefix
*prefix
, const WS_XML_STRING
*str
, const WS_XML_STRING
*ns
)
427 heap_free( prefix
->str
.bytes
);
428 if (!(prefix
->str
.bytes
= heap_alloc( str
->length
))) return E_OUTOFMEMORY
;
429 memcpy( prefix
->str
.bytes
, str
->bytes
, str
->length
);
430 prefix
->str
.length
= str
->length
;
433 heap_free( prefix
->ns
.bytes
);
434 if (!(prefix
->ns
.bytes
= heap_alloc( ns
->length
))) return E_OUTOFMEMORY
;
435 memcpy( prefix
->ns
.bytes
, ns
->bytes
, ns
->length
);
436 prefix
->ns
.length
= ns
->length
;
441 static HRESULT
bind_prefix( struct reader
*reader
, const WS_XML_STRING
*prefix
, const WS_XML_STRING
*ns
)
446 for (i
= 0; i
< reader
->nb_prefixes
; i
++)
448 if (WsXmlStringEquals( prefix
, &reader
->prefixes
[i
].str
, NULL
) == S_OK
)
449 return set_prefix( &reader
->prefixes
[i
], NULL
, ns
);
451 if (i
>= reader
->nb_prefixes_allocated
)
453 ULONG new_size
= reader
->nb_prefixes_allocated
* sizeof(*reader
->prefixes
) * 2;
454 struct prefix
*tmp
= heap_realloc_zero( reader
->prefixes
, new_size
);
455 if (!tmp
) return E_OUTOFMEMORY
;
456 reader
->prefixes
= tmp
;
457 reader
->nb_prefixes_allocated
*= 2;
460 if ((hr
= set_prefix( &reader
->prefixes
[i
], prefix
, ns
)) != S_OK
) return hr
;
461 reader
->nb_prefixes
++;
465 static const WS_XML_STRING
*get_namespace( struct reader
*reader
, const WS_XML_STRING
*prefix
)
468 for (i
= 0; i
< reader
->nb_prefixes
; i
++)
470 if (WsXmlStringEquals( prefix
, &reader
->prefixes
[i
].str
, NULL
) == S_OK
)
471 return &reader
->prefixes
[i
].ns
;
476 static void read_insert_eof( struct reader
*reader
, struct node
*eof
)
478 if (!reader
->root
) reader
->root
= eof
;
481 eof
->parent
= reader
->root
;
482 list_add_tail( &reader
->root
->children
, &eof
->entry
);
484 reader
->current
= reader
->last
= eof
;
487 static void read_insert_bof( struct reader
*reader
, struct node
*bof
)
489 reader
->root
->parent
= bof
;
490 list_add_tail( &bof
->children
, &reader
->root
->entry
);
491 reader
->current
= reader
->last
= reader
->root
= bof
;
494 static void read_insert_node( struct reader
*reader
, struct node
*parent
, struct node
*node
)
496 node
->parent
= parent
;
497 list_add_before( list_tail( &parent
->children
), &node
->entry
);
498 reader
->current
= reader
->last
= node
;
501 static void free_reader( struct reader
*reader
)
503 destroy_nodes( reader
->root
);
504 clear_prefixes( reader
->prefixes
, reader
->nb_prefixes
);
505 heap_free( reader
->prefixes
);
506 reader
->cs
.DebugInfo
->Spare
[0] = 0;
507 DeleteCriticalSection( &reader
->cs
);
511 static HRESULT
init_reader( struct reader
*reader
)
515 reader
->state
= READER_STATE_INITIAL
;
516 destroy_nodes( reader
->root
);
517 reader
->root
= reader
->current
= NULL
;
518 reader
->current_attr
= 0;
519 clear_prefixes( reader
->prefixes
, reader
->nb_prefixes
);
520 reader
->nb_prefixes
= 1;
521 if (!(node
= alloc_node( WS_XML_NODE_TYPE_EOF
))) return E_OUTOFMEMORY
;
522 read_insert_eof( reader
, node
);
523 reader
->input_enc
= WS_XML_READER_ENCODING_TYPE_TEXT
;
527 /**************************************************************************
528 * WsCreateReader [webservices.@]
530 HRESULT WINAPI
WsCreateReader( const WS_XML_READER_PROPERTY
*properties
, ULONG count
,
531 WS_XML_READER
**handle
, WS_ERROR
*error
)
533 struct reader
*reader
;
534 ULONG i
, max_depth
= 32, max_attrs
= 128, max_ns
= 32;
535 WS_CHARSET charset
= WS_CHARSET_UTF8
;
536 BOOL read_decl
= TRUE
;
539 TRACE( "%p %u %p %p\n", properties
, count
, handle
, error
);
540 if (error
) FIXME( "ignoring error parameter\n" );
542 if (!handle
) return E_INVALIDARG
;
543 if (!(reader
= alloc_reader())) return E_OUTOFMEMORY
;
545 prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_MAX_DEPTH
, &max_depth
, sizeof(max_depth
) );
546 prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_MAX_ATTRIBUTES
, &max_attrs
, sizeof(max_attrs
) );
547 prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_READ_DECLARATION
, &read_decl
, sizeof(read_decl
) );
548 prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_CHARSET
, &charset
, sizeof(charset
) );
549 prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_MAX_NAMESPACES
, &max_ns
, sizeof(max_ns
) );
551 for (i
= 0; i
< count
; i
++)
553 hr
= prop_set( reader
->prop
, reader
->prop_count
, properties
[i
].id
, properties
[i
].value
,
554 properties
[i
].valueSize
);
557 free_reader( reader
);
562 if ((hr
= init_reader( reader
)) != S_OK
)
564 free_reader( reader
);
568 *handle
= (WS_XML_READER
*)reader
;
572 /**************************************************************************
573 * WsFreeReader [webservices.@]
575 void WINAPI
WsFreeReader( WS_XML_READER
*handle
)
577 struct reader
*reader
= (struct reader
*)handle
;
579 TRACE( "%p\n", handle
);
583 EnterCriticalSection( &reader
->cs
);
585 if (reader
->magic
!= READER_MAGIC
)
587 LeaveCriticalSection( &reader
->cs
);
593 LeaveCriticalSection( &reader
->cs
);
594 free_reader( reader
);
597 /**************************************************************************
598 * WsFillReader [webservices.@]
600 HRESULT WINAPI
WsFillReader( WS_XML_READER
*handle
, ULONG min_size
, const WS_ASYNC_CONTEXT
*ctx
,
603 struct reader
*reader
= (struct reader
*)handle
;
605 TRACE( "%p %u %p %p\n", handle
, min_size
, ctx
, error
);
606 if (error
) FIXME( "ignoring error parameter\n" );
608 if (!reader
) return E_INVALIDARG
;
610 EnterCriticalSection( &reader
->cs
);
612 if (reader
->magic
!= READER_MAGIC
)
614 LeaveCriticalSection( &reader
->cs
);
618 /* FIXME: add support for stream input */
619 reader
->read_size
= min( min_size
, reader
->input_size
);
620 reader
->read_pos
= 0;
622 LeaveCriticalSection( &reader
->cs
);
626 /**************************************************************************
627 * WsGetNamespaceFromPrefix [webservices.@]
629 HRESULT WINAPI
WsGetNamespaceFromPrefix( WS_XML_READER
*handle
, const WS_XML_STRING
*prefix
,
630 BOOL required
, const WS_XML_STRING
**ns
, WS_ERROR
*error
)
632 static const WS_XML_STRING xml
= {3, (BYTE
*)"xml"};
633 static const WS_XML_STRING xmlns
= {5, (BYTE
*)"xmlns"};
634 static const WS_XML_STRING empty_ns
= {0, NULL
};
635 static const WS_XML_STRING xml_ns
= {36, (BYTE
*)"http://www.w3.org/XML/1998/namespace"};
636 static const WS_XML_STRING xmlns_ns
= {29, (BYTE
*)"http://www.w3.org/2000/xmlns/"};
637 struct reader
*reader
= (struct reader
*)handle
;
640 TRACE( "%p %s %d %p %p\n", handle
, debugstr_xmlstr(prefix
), required
, ns
, error
);
641 if (error
) FIXME( "ignoring error parameter\n" );
643 if (!reader
|| !prefix
|| !ns
) return E_INVALIDARG
;
645 EnterCriticalSection( &reader
->cs
);
647 if (reader
->magic
!= READER_MAGIC
)
649 LeaveCriticalSection( &reader
->cs
);
653 if (reader
->state
!= READER_STATE_STARTELEMENT
)
655 LeaveCriticalSection( &reader
->cs
);
656 return WS_E_INVALID_OPERATION
;
664 else if (WsXmlStringEquals( prefix
, &xml
, NULL
) == S_OK
)
669 else if (WsXmlStringEquals( prefix
, &xmlns
, NULL
) == S_OK
)
676 WS_XML_ELEMENT_NODE
*elem
= &reader
->current
->hdr
;
679 for (i
= 0; i
< elem
->attributeCount
; i
++)
681 if (!elem
->attributes
[i
]->isXmlNs
) continue;
682 if (WsXmlStringEquals( prefix
, elem
->attributes
[i
]->prefix
, NULL
) == S_OK
)
684 *ns
= elem
->attributes
[i
]->ns
;
691 LeaveCriticalSection( &reader
->cs
);
695 if (required
) return WS_E_INVALID_FORMAT
;
703 /**************************************************************************
704 * WsGetReaderNode [webservices.@]
706 HRESULT WINAPI
WsGetReaderNode( WS_XML_READER
*handle
, const WS_XML_NODE
**node
,
709 struct reader
*reader
= (struct reader
*)handle
;
711 TRACE( "%p %p %p\n", handle
, node
, error
);
712 if (error
) FIXME( "ignoring error parameter\n" );
714 if (!reader
|| !node
) return E_INVALIDARG
;
716 EnterCriticalSection( &reader
->cs
);
718 if (reader
->magic
!= READER_MAGIC
)
720 LeaveCriticalSection( &reader
->cs
);
724 *node
= &reader
->current
->hdr
.node
;
726 LeaveCriticalSection( &reader
->cs
);
730 /**************************************************************************
731 * WsGetReaderProperty [webservices.@]
733 HRESULT WINAPI
WsGetReaderProperty( WS_XML_READER
*handle
, WS_XML_READER_PROPERTY_ID id
,
734 void *buf
, ULONG size
, WS_ERROR
*error
)
736 struct reader
*reader
= (struct reader
*)handle
;
739 TRACE( "%p %u %p %u %p\n", handle
, id
, buf
, size
, error
);
740 if (error
) FIXME( "ignoring error parameter\n" );
742 if (!reader
) return E_INVALIDARG
;
744 EnterCriticalSection( &reader
->cs
);
746 if (reader
->magic
!= READER_MAGIC
)
748 LeaveCriticalSection( &reader
->cs
);
752 if (!reader
->input_type
)
754 LeaveCriticalSection( &reader
->cs
);
755 return WS_E_INVALID_OPERATION
;
758 if (id
== WS_XML_READER_PROPERTY_CHARSET
)
761 if ((hr
= prop_get( reader
->prop
, reader
->prop_count
, id
, &charset
, size
)) != S_OK
) goto done
;
764 hr
= WS_E_INVALID_FORMAT
;
767 *(WS_CHARSET
*)buf
= charset
;
770 else hr
= prop_get( reader
->prop
, reader
->prop_count
, id
, buf
, size
);
773 LeaveCriticalSection( &reader
->cs
);
777 /**************************************************************************
778 * WsGetXmlAttribute [webservices.@]
780 HRESULT WINAPI
WsGetXmlAttribute( WS_XML_READER
*handle
, const WS_XML_STRING
*attr
,
781 WS_HEAP
*heap
, WCHAR
**str
, ULONG
*len
, WS_ERROR
*error
)
783 FIXME( "%p %s %p %p %p %p: stub\n", handle
, debugstr_xmlstr(attr
), heap
, str
, len
, error
);
787 WS_XML_STRING
*alloc_xml_string( const unsigned char *data
, ULONG len
)
791 if (!(ret
= heap_alloc( sizeof(*ret
) + len
))) return NULL
;
793 ret
->bytes
= len
? (BYTE
*)(ret
+ 1) : NULL
;
794 ret
->dictionary
= NULL
;
796 if (data
) memcpy( ret
->bytes
, data
, len
);
800 WS_XML_UTF8_TEXT
*alloc_utf8_text( const unsigned char *data
, ULONG len
)
802 WS_XML_UTF8_TEXT
*ret
;
804 if (!(ret
= heap_alloc( sizeof(*ret
) + len
))) return NULL
;
805 ret
->text
.textType
= WS_XML_TEXT_TYPE_UTF8
;
806 ret
->value
.length
= len
;
807 ret
->value
.bytes
= len
? (BYTE
*)(ret
+ 1) : NULL
;
808 ret
->value
.dictionary
= NULL
;
810 if (data
) memcpy( ret
->value
.bytes
, data
, len
);
814 static inline BOOL
read_end_of_data( struct reader
*reader
)
816 return reader
->read_pos
>= reader
->read_size
;
819 static inline const unsigned char *read_current_ptr( struct reader
*reader
)
821 return &reader
->read_bufptr
[reader
->read_pos
];
824 static inline HRESULT
read_peek( struct reader
*reader
, unsigned char *byte
)
826 if (reader
->read_pos
>= reader
->read_size
) return WS_E_INVALID_FORMAT
;
827 *byte
= reader
->read_bufptr
[reader
->read_pos
];
831 static inline HRESULT
read_byte( struct reader
*reader
, unsigned char *byte
)
833 if (reader
->read_pos
>= reader
->read_size
) return WS_E_INVALID_FORMAT
;
834 *byte
= reader
->read_bufptr
[reader
->read_pos
++];
838 static inline HRESULT
read_bytes( struct reader
*reader
, unsigned char *bytes
, unsigned int len
)
840 if (reader
->read_pos
+ len
> reader
->read_size
) return WS_E_INVALID_FORMAT
;
841 memcpy( bytes
, reader
->read_bufptr
+ reader
->read_pos
, len
);
842 reader
->read_pos
+= len
;
846 /* UTF-8 support based on libs/wine/utf8.c */
848 /* number of following bytes in sequence based on first byte value (for bytes above 0x7f) */
849 static const char utf8_length
[128] =
851 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x80-0x8f */
852 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x90-0x9f */
853 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0xa0-0xaf */
854 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0xb0-0xbf */
855 0,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* 0xc0-0xcf */
856 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* 0xd0-0xdf */
857 2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2, /* 0xe0-0xef */
858 3,3,3,3,3,0,0,0,0,0,0,0,0,0,0,0 /* 0xf0-0xff */
861 /* first byte mask depending on UTF-8 sequence length */
862 static const unsigned char utf8_mask
[4] = { 0x7f, 0x1f, 0x0f, 0x07 };
864 /* minimum Unicode value depending on UTF-8 sequence length */
865 static const unsigned int utf8_minval
[4] = { 0x0, 0x80, 0x800, 0x10000 };
867 static inline unsigned int read_utf8_char( struct reader
*reader
, unsigned int *skip
)
869 unsigned int len
, res
;
870 unsigned char ch
= reader
->read_bufptr
[reader
->read_pos
];
871 const unsigned char *end
;
873 if (reader
->read_pos
>= reader
->read_size
) return 0;
880 len
= utf8_length
[ch
- 0x80];
881 if (reader
->read_pos
+ len
>= reader
->read_size
) return 0;
882 end
= reader
->read_bufptr
+ reader
->read_pos
+ len
+ 1;
883 res
= ch
& utf8_mask
[len
];
888 if ((ch
= end
[-3] ^ 0x80) >= 0x40) break;
889 res
= (res
<< 6) | ch
;
891 if ((ch
= end
[-2] ^ 0x80) >= 0x40) break;
892 res
= (res
<< 6) | ch
;
894 if ((ch
= end
[-1] ^ 0x80) >= 0x40) break;
895 res
= (res
<< 6) | ch
;
896 if (res
< utf8_minval
[len
]) break;
904 static inline void read_skip( struct reader
*reader
, unsigned int count
)
906 if (reader
->read_pos
+ count
> reader
->read_size
) return;
907 reader
->read_pos
+= count
;
910 static inline void read_rewind( struct reader
*reader
, unsigned int count
)
912 reader
->read_pos
-= count
;
915 static inline BOOL
read_isnamechar( unsigned int ch
)
917 /* FIXME: incomplete */
918 return (ch
>= 'A' && ch
<= 'Z') ||
919 (ch
>= 'a' && ch
<= 'z') ||
920 (ch
>= '0' && ch
<= '9') ||
921 ch
== '_' || ch
== '-' || ch
== '.' || ch
== ':';
924 static inline BOOL
read_isspace( unsigned int ch
)
926 return ch
== ' ' || ch
== '\t' || ch
== '\r' || ch
== '\n';
929 static inline void read_skip_whitespace( struct reader
*reader
)
931 while (reader
->read_pos
< reader
->read_size
&& read_isspace( reader
->read_bufptr
[reader
->read_pos
] ))
935 static inline int read_cmp( struct reader
*reader
, const char *str
, int len
)
937 const unsigned char *ptr
= read_current_ptr( reader
);
939 if (len
< 0) len
= strlen( str
);
940 if (reader
->read_pos
+ len
> reader
->read_size
) return -1;
943 if (*str
!= *ptr
) return *ptr
- *str
;
949 static HRESULT
read_xmldecl( struct reader
*reader
)
951 if (!reader
->read_size
) return WS_E_INVALID_FORMAT
;
953 if (read_cmp( reader
, "<", 1 ) || read_cmp( reader
, "<?", 2 ))
955 reader
->state
= READER_STATE_BOF
;
958 if (read_cmp( reader
, "<?xml ", 6 )) return WS_E_INVALID_FORMAT
;
959 read_skip( reader
, 6 );
961 /* FIXME: parse attributes */
962 while (reader
->read_pos
< reader
->read_size
&& reader
->read_bufptr
[reader
->read_pos
] != '?')
965 if (read_cmp( reader
, "?>", 2 )) return WS_E_INVALID_FORMAT
;
966 read_skip( reader
, 2 );
968 reader
->state
= READER_STATE_BOF
;
972 HRESULT
append_attribute( WS_XML_ELEMENT_NODE
*elem
, WS_XML_ATTRIBUTE
*attr
)
974 if (elem
->attributeCount
)
976 WS_XML_ATTRIBUTE
**tmp
;
977 if (!(tmp
= heap_realloc( elem
->attributes
, (elem
->attributeCount
+ 1) * sizeof(attr
) )))
978 return E_OUTOFMEMORY
;
979 elem
->attributes
= tmp
;
981 else if (!(elem
->attributes
= heap_alloc( sizeof(attr
) ))) return E_OUTOFMEMORY
;
982 elem
->attributes
[elem
->attributeCount
++] = attr
;
986 static HRESULT
split_name( const unsigned char *str
, ULONG len
, const unsigned char **prefix
,
987 ULONG
*prefix_len
, const unsigned char **localname
, ULONG
*localname_len
)
989 const unsigned char *ptr
= str
;
995 *localname_len
= len
;
1001 if (ptr
== str
) return WS_E_INVALID_FORMAT
;
1003 *prefix_len
= ptr
- str
;
1004 *localname
= ptr
+ 1;
1005 *localname_len
= len
;
1013 static HRESULT
parse_name( const unsigned char *str
, ULONG len
, WS_XML_STRING
**prefix
, WS_XML_STRING
**localname
)
1015 const unsigned char *localname_ptr
, *prefix_ptr
;
1016 ULONG localname_len
, prefix_len
;
1019 if ((hr
= split_name( str
, len
, &prefix_ptr
, &prefix_len
, &localname_ptr
, &localname_len
)) != S_OK
) return hr
;
1020 if (!(*prefix
= alloc_xml_string( prefix_ptr
, prefix_len
))) return E_OUTOFMEMORY
;
1021 if (!(*localname
= alloc_xml_string( localname_ptr
, localname_len
)))
1023 heap_free( *prefix
);
1025 return E_OUTOFMEMORY
;
1030 static int codepoint_to_utf8( int cp
, unsigned char *dst
)
1040 dst
[1] = 0x80 | (cp
& 0x3f);
1045 if ((cp
>= 0xd800 && cp
<= 0xdfff) || cp
== 0xfffe || cp
== 0xffff) return -1;
1048 dst
[2] = 0x80 | (cp
& 0x3f);
1050 dst
[1] = 0x80 | (cp
& 0x3f);
1055 if (cp
>= 0x110000) return -1;
1056 dst
[3] = 0x80 | (cp
& 0x3f);
1058 dst
[2] = 0x80 | (cp
& 0x3f);
1060 dst
[1] = 0x80 | (cp
& 0x3f);
1066 static HRESULT
decode_text( const unsigned char *str
, ULONG len
, unsigned char *ret
, ULONG
*ret_len
)
1068 const unsigned char *p
= str
;
1069 unsigned char *q
= ret
;
1077 if (!len
) return WS_E_INVALID_FORMAT
;
1079 if (len
>= 3 && !memcmp( p
, "lt;", 3 ))
1085 else if (len
>= 3 && !memcmp( p
, "gt;", 3 ))
1091 else if (len
>= 5 && !memcmp( p
, "quot;", 5 ))
1097 else if (len
>= 4 && !memcmp( p
, "amp;", 4 ))
1103 else if (len
>= 5 && !memcmp( p
, "apos;", 5 ))
1111 ULONG start
, nb_digits
, i
;
1112 int len_utf8
, cp
= 0;
1115 if (!len
) return WS_E_INVALID_FORMAT
;
1121 while (len
&& isxdigit( *p
)) { p
++; len
--; };
1122 if (!len
) return WS_E_INVALID_FORMAT
;
1124 p
-= nb_digits
= start
- len
;
1125 if (!nb_digits
|| nb_digits
> 6 || p
[nb_digits
] != ';') return WS_E_INVALID_FORMAT
;
1126 for (i
= 0; i
< nb_digits
; i
++)
1129 if (*p
>= '0' && *p
<= '9') cp
+= *p
- '0';
1130 else if (*p
>= 'a' && *p
<= 'f') cp
+= *p
- 'a' + 10;
1131 else cp
+= *p
- 'A' + 10;
1135 else if (isdigit( *p
))
1137 while (len
&& *p
== '0') { p
++; len
--; };
1138 if (!len
) return WS_E_INVALID_FORMAT
;
1141 while (len
&& isdigit( *p
)) { p
++; len
--; };
1142 if (!len
) return WS_E_INVALID_FORMAT
;
1144 p
-= nb_digits
= start
- len
;
1145 if (!nb_digits
|| nb_digits
> 7 || p
[nb_digits
] != ';') return WS_E_INVALID_FORMAT
;
1146 for (i
= 0; i
< nb_digits
; i
++)
1153 else return WS_E_INVALID_FORMAT
;
1155 if ((len_utf8
= codepoint_to_utf8( cp
, q
)) < 0) return WS_E_INVALID_FORMAT
;
1156 *ret_len
+= len_utf8
;
1160 else return WS_E_INVALID_FORMAT
;
1172 static HRESULT
read_attribute_value_text( struct reader
*reader
, WS_XML_ATTRIBUTE
*attr
)
1174 WS_XML_UTF8_TEXT
*utf8
= NULL
;
1175 unsigned int len
, ch
, skip
, quote
;
1176 const unsigned char *start
;
1177 HRESULT hr
= E_OUTOFMEMORY
;
1179 read_skip_whitespace( reader
);
1180 if (read_cmp( reader
, "=", 1 )) return WS_E_INVALID_FORMAT
;
1181 read_skip( reader
, 1 );
1183 read_skip_whitespace( reader
);
1184 if (read_cmp( reader
, "\"", 1 ) && read_cmp( reader
, "'", 1 )) return WS_E_INVALID_FORMAT
;
1185 quote
= read_utf8_char( reader
, &skip
);
1186 read_skip( reader
, 1 );
1189 start
= read_current_ptr( reader
);
1192 if (!(ch
= read_utf8_char( reader
, &skip
))) return WS_E_INVALID_FORMAT
;
1193 if (ch
== quote
) break;
1194 read_skip( reader
, skip
);
1197 read_skip( reader
, 1 );
1201 if (!(attr
->ns
= alloc_xml_string( start
, len
))) goto error
;
1202 if ((hr
= bind_prefix( reader
, attr
->prefix
, attr
->ns
)) != S_OK
) goto error
;
1203 if (!(utf8
= alloc_utf8_text( NULL
, 0 )))
1211 if (!(utf8
= alloc_utf8_text( NULL
, len
))) goto error
;
1212 if ((hr
= decode_text( start
, len
, utf8
->value
.bytes
, &utf8
->value
.length
)) != S_OK
) goto error
;
1215 attr
->value
= &utf8
->text
;
1216 attr
->singleQuote
= (quote
== '\'');
1224 static inline BOOL
is_text_type( unsigned char type
)
1226 return (type
>= RECORD_ZERO_TEXT
&& type
<= RECORD_QNAME_DICTIONARY_TEXT_WITH_ENDELEMENT
);
1229 static HRESULT
read_int31( struct reader
*reader
, ULONG
*len
)
1234 if ((hr
= read_byte( reader
, &byte
)) != S_OK
) return hr
;
1236 if (!(byte
& 0x80)) return S_OK
;
1238 if ((hr
= read_byte( reader
, &byte
)) != S_OK
) return hr
;
1239 *len
+= (byte
& 0x7f) << 7;
1240 if (!(byte
& 0x80)) return S_OK
;
1242 if ((hr
= read_byte( reader
, &byte
)) != S_OK
) return hr
;
1243 *len
+= (byte
& 0x7f) << 14;
1244 if (!(byte
& 0x80)) return S_OK
;
1246 if ((hr
= read_byte( reader
, &byte
)) != S_OK
) return hr
;
1247 *len
+= (byte
& 0x7f) << 21;
1248 if (!(byte
& 0x80)) return S_OK
;
1250 if ((hr
= read_byte( reader
, &byte
)) != S_OK
) return hr
;
1251 *len
+= (byte
& 0x07) << 28;
1255 static HRESULT
read_string( struct reader
*reader
, WS_XML_STRING
**str
)
1259 if ((hr
= read_int31( reader
, &len
)) != S_OK
) return hr
;
1260 if (!(*str
= alloc_xml_string( NULL
, len
))) return E_OUTOFMEMORY
;
1261 if ((hr
= read_bytes( reader
, (*str
)->bytes
, len
)) == S_OK
) return S_OK
;
1266 static HRESULT
read_attribute_value_bin( struct reader
*reader
, WS_XML_ATTRIBUTE
*attr
)
1268 WS_XML_UTF8_TEXT
*utf8
= NULL
;
1273 if ((hr
= read_byte( reader
, &type
)) != S_OK
) return hr
;
1274 if (!is_text_type( type
)) return WS_E_INVALID_FORMAT
;
1278 case RECORD_CHARS8_TEXT
:
1281 if ((hr
= read_byte( reader
, &len8
)) != S_OK
) return hr
;
1286 ERR( "unhandled record type %02x\n", type
);
1287 return WS_E_NOT_SUPPORTED
;
1290 if (!(utf8
= alloc_utf8_text( NULL
, len
))) return E_OUTOFMEMORY
;
1291 if (!len
) utf8
->value
.bytes
= (BYTE
*)(utf8
+ 1); /* quirk */
1292 if ((hr
= read_bytes( reader
, utf8
->value
.bytes
, len
)) != S_OK
)
1298 attr
->value
= &utf8
->text
;
1302 static HRESULT
read_attribute_text( struct reader
*reader
, WS_XML_ATTRIBUTE
**ret
)
1304 static const WS_XML_STRING xmlns
= {5, (BYTE
*)"xmlns"};
1305 WS_XML_ATTRIBUTE
*attr
;
1306 unsigned int len
= 0, ch
, skip
;
1307 const unsigned char *start
;
1308 WS_XML_STRING
*prefix
, *localname
;
1309 HRESULT hr
= WS_E_INVALID_FORMAT
;
1311 if (!(attr
= heap_alloc_zero( sizeof(*attr
) ))) return E_OUTOFMEMORY
;
1313 start
= read_current_ptr( reader
);
1316 if (!(ch
= read_utf8_char( reader
, &skip
))) goto error
;
1317 if (!read_isnamechar( ch
)) break;
1318 read_skip( reader
, skip
);
1321 if (!len
) goto error
;
1323 if ((hr
= parse_name( start
, len
, &prefix
, &localname
)) != S_OK
) goto error
;
1324 if (WsXmlStringEquals( prefix
, &xmlns
, NULL
) == S_OK
)
1326 heap_free( prefix
);
1328 if (!(attr
->prefix
= alloc_xml_string( localname
->bytes
, localname
->length
)))
1330 heap_free( localname
);
1334 attr
->localName
= localname
;
1336 else if (!prefix
->length
&& WsXmlStringEquals( localname
, &xmlns
, NULL
) == S_OK
)
1339 attr
->prefix
= prefix
;
1340 attr
->localName
= localname
;
1344 attr
->prefix
= prefix
;
1345 attr
->localName
= localname
;
1348 if ((hr
= read_attribute_value_text( reader
, attr
)) != S_OK
) goto error
;
1354 free_attribute( attr
);
1358 static inline BOOL
is_attribute_type( unsigned char type
)
1360 return (type
>= RECORD_SHORT_ATTRIBUTE
&& type
<= RECORD_PREFIX_ATTRIBUTE_Z
);
1363 static HRESULT
read_attribute_bin( struct reader
*reader
, WS_XML_ATTRIBUTE
**ret
)
1365 WS_XML_ATTRIBUTE
*attr
;
1366 unsigned char type
= 0;
1369 if ((hr
= read_byte( reader
, &type
)) != S_OK
) return hr
;
1370 if (!is_attribute_type( type
)) return WS_E_INVALID_FORMAT
;
1371 if (!(attr
= heap_alloc_zero( sizeof(*attr
) ))) return E_OUTOFMEMORY
;
1373 if (type
>= RECORD_PREFIX_ATTRIBUTE_A
&& type
<= RECORD_PREFIX_ATTRIBUTE_Z
)
1375 unsigned char ch
= type
- RECORD_PREFIX_ATTRIBUTE_A
+ 'a';
1376 if (!(attr
->prefix
= alloc_xml_string( &ch
, 1 )))
1381 if ((hr
= read_string( reader
, &attr
->localName
)) != S_OK
) goto error
;
1382 if ((hr
= read_attribute_value_bin( reader
, attr
)) != S_OK
) goto error
;
1388 case RECORD_SHORT_ATTRIBUTE
:
1389 if (!(attr
->prefix
= alloc_xml_string( NULL
, 0 )))
1394 if ((hr
= read_string( reader
, &attr
->localName
)) != S_OK
) goto error
;
1395 if ((hr
= read_attribute_value_bin( reader
, attr
)) != S_OK
) goto error
;
1398 case RECORD_ATTRIBUTE
:
1399 if ((hr
= read_string( reader
, &attr
->prefix
)) != S_OK
) goto error
;
1400 if ((hr
= read_string( reader
, &attr
->localName
)) != S_OK
) goto error
;
1401 if ((hr
= read_attribute_value_bin( reader
, attr
)) != S_OK
) goto error
;
1404 case RECORD_SHORT_XMLNS_ATTRIBUTE
:
1405 if (!(attr
->prefix
= alloc_xml_string( NULL
, 0 )))
1410 if ((hr
= read_string( reader
, &attr
->ns
)) != S_OK
) goto error
;
1414 case RECORD_XMLNS_ATTRIBUTE
:
1415 if ((hr
= read_string( reader
, &attr
->prefix
)) != S_OK
) goto error
;
1416 if ((hr
= read_string( reader
, &attr
->ns
)) != S_OK
) goto error
;
1417 if ((hr
= bind_prefix( reader
, attr
->prefix
, attr
->ns
)) != S_OK
) goto error
;
1422 ERR( "unhandled record type %02x\n", type
);
1423 return WS_E_NOT_SUPPORTED
;
1431 free_attribute( attr
);
1435 static inline struct node
*find_parent( struct reader
*reader
)
1437 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_END_ELEMENT
)
1439 if (is_valid_parent( reader
->current
->parent
->parent
)) return reader
->current
->parent
->parent
;
1442 if (is_valid_parent( reader
->current
)) return reader
->current
;
1443 if (is_valid_parent( reader
->current
->parent
)) return reader
->current
->parent
;
1447 static HRESULT
set_namespaces( struct reader
*reader
, WS_XML_ELEMENT_NODE
*elem
)
1449 static const WS_XML_STRING xml
= {3, (BYTE
*)"xml"};
1450 const WS_XML_STRING
*ns
;
1453 if (!(ns
= get_namespace( reader
, elem
->prefix
))) return WS_E_INVALID_FORMAT
;
1454 if (!(elem
->ns
= alloc_xml_string( ns
->bytes
, ns
->length
))) return E_OUTOFMEMORY
;
1455 if (!elem
->ns
->length
) elem
->ns
->bytes
= (BYTE
*)(elem
->ns
+ 1); /* quirk */
1457 for (i
= 0; i
< elem
->attributeCount
; i
++)
1459 WS_XML_ATTRIBUTE
*attr
= elem
->attributes
[i
];
1460 if (attr
->isXmlNs
|| WsXmlStringEquals( attr
->prefix
, &xml
, NULL
) == S_OK
) continue;
1461 if (!(ns
= get_namespace( reader
, attr
->prefix
))) return WS_E_INVALID_FORMAT
;
1462 if (!(attr
->ns
= alloc_xml_string( ns
->bytes
, ns
->length
))) return E_OUTOFMEMORY
;
1467 static WS_XML_ELEMENT_NODE
*alloc_element_pair(void)
1469 struct node
*node
, *end
;
1470 if (!(node
= alloc_node( WS_XML_NODE_TYPE_ELEMENT
))) return NULL
;
1471 if (!(end
= alloc_node( WS_XML_NODE_TYPE_END_ELEMENT
)))
1476 list_add_tail( &node
->children
, &end
->entry
);
1481 static HRESULT
read_attributes_text( struct reader
*reader
, WS_XML_ELEMENT_NODE
*elem
)
1483 WS_XML_ATTRIBUTE
*attr
;
1486 reader
->current_attr
= 0;
1489 read_skip_whitespace( reader
);
1490 if (!read_cmp( reader
, ">", 1 ) || !read_cmp( reader
, "/>", 2 )) break;
1491 if ((hr
= read_attribute_text( reader
, &attr
)) != S_OK
) return hr
;
1492 if ((hr
= append_attribute( elem
, attr
)) != S_OK
)
1494 free_attribute( attr
);
1497 reader
->current_attr
++;
1502 static HRESULT
read_element_text( struct reader
*reader
)
1504 unsigned int len
= 0, ch
, skip
;
1505 const unsigned char *start
;
1506 struct node
*node
= NULL
, *parent
;
1507 WS_XML_ELEMENT_NODE
*elem
;
1508 HRESULT hr
= WS_E_INVALID_FORMAT
;
1510 if (read_end_of_data( reader
))
1512 reader
->current
= LIST_ENTRY( list_tail( &reader
->root
->children
), struct node
, entry
);
1513 reader
->last
= reader
->current
;
1514 reader
->state
= READER_STATE_EOF
;
1518 if (read_cmp( reader
, "<", 1 )) return WS_E_INVALID_FORMAT
;
1519 read_skip( reader
, 1 );
1520 if (!read_isnamechar( read_utf8_char( reader
, &skip
)))
1522 read_rewind( reader
, 1 );
1523 return WS_E_INVALID_FORMAT
;
1526 if (!(elem
= alloc_element_pair())) return E_OUTOFMEMORY
;
1527 node
= (struct node
*)elem
;
1529 start
= read_current_ptr( reader
);
1532 if (!(ch
= read_utf8_char( reader
, &skip
))) goto error
;
1533 if (!read_isnamechar( ch
)) break;
1534 read_skip( reader
, skip
);
1537 if (!len
) goto error
;
1539 if (!(parent
= find_parent( reader
))) goto error
;
1540 if ((hr
= parse_name( start
, len
, &elem
->prefix
, &elem
->localName
)) != S_OK
) goto error
;
1542 if ((hr
= read_attributes_text( reader
, elem
)) != S_OK
) goto error
;
1543 if ((hr
= set_namespaces( reader
, elem
)) != S_OK
) goto error
;
1545 read_insert_node( reader
, parent
, node
);
1546 reader
->state
= READER_STATE_STARTELEMENT
;
1550 destroy_nodes( node
);
1554 static inline BOOL
is_element_type( unsigned char type
)
1556 return (type
>= RECORD_SHORT_ELEMENT
&& type
<= RECORD_PREFIX_ELEMENT_Z
);
1559 static HRESULT
read_attributes_bin( struct reader
*reader
, WS_XML_ELEMENT_NODE
*elem
)
1561 WS_XML_ATTRIBUTE
*attr
;
1565 reader
->current_attr
= 0;
1568 if ((hr
= read_peek( reader
, &type
)) != S_OK
) return hr
;
1569 if (!is_attribute_type( type
)) break;
1570 if ((hr
= read_attribute_bin( reader
, &attr
)) != S_OK
) return hr
;
1571 if ((hr
= append_attribute( elem
, attr
)) != S_OK
)
1573 free_attribute( attr
);
1576 reader
->current_attr
++;
1581 static HRESULT
read_element_bin( struct reader
*reader
)
1583 struct node
*node
= NULL
, *parent
;
1584 WS_XML_ELEMENT_NODE
*elem
;
1588 if ((hr
= read_byte( reader
, &type
)) != S_OK
) return hr
;
1589 if (!is_element_type( type
)) return WS_E_INVALID_FORMAT
;
1591 if (!(elem
= alloc_element_pair())) return E_OUTOFMEMORY
;
1592 node
= (struct node
*)elem
;
1594 if (type
>= RECORD_PREFIX_ELEMENT_A
&& type
<= RECORD_PREFIX_ELEMENT_Z
)
1596 unsigned char ch
= type
- RECORD_PREFIX_ELEMENT_A
+ 'a';
1597 if (!(elem
->prefix
= alloc_xml_string( &ch
, 1 )))
1602 if ((hr
= read_string( reader
, &elem
->localName
)) != S_OK
) goto error
;
1608 case RECORD_SHORT_ELEMENT
:
1609 if (!(elem
->prefix
= alloc_xml_string( NULL
, 0 )))
1614 if ((hr
= read_string( reader
, &elem
->localName
)) != S_OK
) goto error
;
1617 case RECORD_ELEMENT
:
1618 if ((hr
= read_string( reader
, &elem
->prefix
)) != S_OK
) goto error
;
1619 if ((hr
= read_string( reader
, &elem
->localName
)) != S_OK
) goto error
;
1623 ERR( "unhandled record type %02x\n", type
);
1624 return WS_E_NOT_SUPPORTED
;
1628 if (!(parent
= find_parent( reader
)))
1630 hr
= WS_E_INVALID_FORMAT
;
1634 if ((hr
= read_attributes_bin( reader
, elem
)) != S_OK
) goto error
;
1635 if ((hr
= set_namespaces( reader
, elem
)) != S_OK
) goto error
;
1637 read_insert_node( reader
, parent
, node
);
1638 reader
->state
= READER_STATE_STARTELEMENT
;
1642 destroy_nodes( node
);
1646 static HRESULT
read_text( struct reader
*reader
)
1648 unsigned int len
= 0, ch
, skip
;
1649 const unsigned char *start
;
1650 struct node
*node
, *parent
;
1651 WS_XML_TEXT_NODE
*text
;
1652 WS_XML_UTF8_TEXT
*utf8
;
1655 start
= read_current_ptr( reader
);
1658 if (read_end_of_data( reader
)) break;
1659 if (!(ch
= read_utf8_char( reader
, &skip
))) return WS_E_INVALID_FORMAT
;
1660 if (ch
== '<') break;
1661 read_skip( reader
, skip
);
1665 if (!(parent
= find_parent( reader
))) return WS_E_INVALID_FORMAT
;
1667 if (!(node
= alloc_node( WS_XML_NODE_TYPE_TEXT
))) return E_OUTOFMEMORY
;
1668 text
= (WS_XML_TEXT_NODE
*)node
;
1669 if (!(utf8
= alloc_utf8_text( NULL
, len
)))
1672 return E_OUTOFMEMORY
;
1674 if ((hr
= decode_text( start
, len
, utf8
->value
.bytes
, &utf8
->value
.length
)) != S_OK
)
1680 text
->text
= &utf8
->text
;
1682 read_insert_node( reader
, parent
, node
);
1683 reader
->state
= READER_STATE_TEXT
;
1684 reader
->text_conv_offset
= 0;
1688 static HRESULT
read_node_text( struct reader
* );
1690 static HRESULT
read_startelement( struct reader
*reader
)
1692 read_skip_whitespace( reader
);
1693 if (!read_cmp( reader
, "/>", 2 ))
1695 read_skip( reader
, 2 );
1696 reader
->current
= LIST_ENTRY( list_tail( &reader
->current
->children
), struct node
, entry
);
1697 reader
->last
= reader
->current
;
1698 reader
->state
= READER_STATE_ENDELEMENT
;
1701 else if (!read_cmp( reader
, ">", 1 ))
1703 read_skip( reader
, 1 );
1704 return read_node_text( reader
);
1706 return WS_E_INVALID_FORMAT
;
1709 static HRESULT
read_to_startelement( struct reader
*reader
, BOOL
*found
)
1713 switch (reader
->state
)
1715 case READER_STATE_INITIAL
:
1716 if ((hr
= read_xmldecl( reader
)) != S_OK
) return hr
;
1719 case READER_STATE_STARTELEMENT
:
1720 if (found
) *found
= TRUE
;
1727 read_skip_whitespace( reader
);
1728 if ((hr
= read_element_text( reader
)) == S_OK
&& found
)
1730 if (reader
->state
== READER_STATE_STARTELEMENT
)
1739 static int cmp_name( const unsigned char *name1
, ULONG len1
, const unsigned char *name2
, ULONG len2
)
1742 if (len1
!= len2
) return 1;
1743 for (i
= 0; i
< len1
; i
++) { if (toupper( name1
[i
] ) != toupper( name2
[i
] )) return 1; }
1747 static struct node
*find_startelement( struct reader
*reader
, const WS_XML_STRING
*prefix
,
1748 const WS_XML_STRING
*localname
)
1750 struct node
*parent
;
1751 const WS_XML_STRING
*str
;
1753 for (parent
= reader
->current
; parent
; parent
= parent
->parent
)
1755 if (node_type( parent
) == WS_XML_NODE_TYPE_ELEMENT
)
1757 str
= parent
->hdr
.prefix
;
1758 if (cmp_name( str
->bytes
, str
->length
, prefix
->bytes
, prefix
->length
)) continue;
1759 str
= parent
->hdr
.localName
;
1760 if (cmp_name( str
->bytes
, str
->length
, localname
->bytes
, localname
->length
)) continue;
1767 static HRESULT
read_endelement_text( struct reader
*reader
)
1769 struct node
*parent
;
1770 unsigned int len
= 0, ch
, skip
;
1771 const unsigned char *start
;
1772 WS_XML_STRING
*prefix
, *localname
;
1775 if (read_cmp( reader
, "</", 2 )) return WS_E_INVALID_FORMAT
;
1776 read_skip( reader
, 2 );
1778 start
= read_current_ptr( reader
);
1781 if (!(ch
= read_utf8_char( reader
, &skip
))) return WS_E_INVALID_FORMAT
;
1784 read_skip( reader
, 1 );
1787 if (!read_isnamechar( ch
)) return WS_E_INVALID_FORMAT
;
1788 read_skip( reader
, skip
);
1792 if ((hr
= parse_name( start
, len
, &prefix
, &localname
)) != S_OK
) return hr
;
1793 parent
= find_startelement( reader
, prefix
, localname
);
1794 heap_free( prefix
);
1795 heap_free( localname
);
1796 if (!parent
) return WS_E_INVALID_FORMAT
;
1798 reader
->current
= LIST_ENTRY( list_tail( &parent
->children
), struct node
, entry
);
1799 reader
->last
= reader
->current
;
1800 reader
->state
= READER_STATE_ENDELEMENT
;
1804 static HRESULT
read_endelement_bin( struct reader
*reader
)
1806 struct node
*parent
;
1810 if ((hr
= read_byte( reader
, &type
)) != S_OK
) return hr
;
1811 if (type
!= RECORD_ENDELEMENT
) return WS_E_INVALID_FORMAT
;
1813 if (!(parent
= find_parent( reader
))) return WS_E_INVALID_FORMAT
;
1815 reader
->current
= LIST_ENTRY( list_tail( &parent
->children
), struct node
, entry
);
1816 reader
->last
= reader
->current
;
1817 reader
->state
= READER_STATE_ENDELEMENT
;
1821 static HRESULT
read_endelement( struct reader
*reader
)
1823 if (reader
->state
== READER_STATE_EOF
) return WS_E_INVALID_FORMAT
;
1825 if (read_end_of_data( reader
))
1827 reader
->current
= LIST_ENTRY( list_tail( &reader
->root
->children
), struct node
, entry
);
1828 reader
->last
= reader
->current
;
1829 reader
->state
= READER_STATE_EOF
;
1833 switch (reader
->input_enc
)
1835 case WS_XML_READER_ENCODING_TYPE_TEXT
: return read_endelement_text( reader
);
1836 case WS_XML_READER_ENCODING_TYPE_BINARY
: return read_endelement_bin( reader
);
1838 ERR( "unhandled encoding %u\n", reader
->input_enc
);
1839 return WS_E_NOT_SUPPORTED
;
1843 static HRESULT
read_comment( struct reader
*reader
)
1845 unsigned int len
= 0, ch
, skip
;
1846 const unsigned char *start
;
1847 struct node
*node
, *parent
;
1848 WS_XML_COMMENT_NODE
*comment
;
1850 if (read_cmp( reader
, "<!--", 4 )) return WS_E_INVALID_FORMAT
;
1851 read_skip( reader
, 4 );
1853 start
= read_current_ptr( reader
);
1856 if (!read_cmp( reader
, "-->", 3 ))
1858 read_skip( reader
, 3 );
1861 if (!(ch
= read_utf8_char( reader
, &skip
))) return WS_E_INVALID_FORMAT
;
1862 read_skip( reader
, skip
);
1866 if (!(parent
= find_parent( reader
))) return WS_E_INVALID_FORMAT
;
1868 if (!(node
= alloc_node( WS_XML_NODE_TYPE_COMMENT
))) return E_OUTOFMEMORY
;
1869 comment
= (WS_XML_COMMENT_NODE
*)node
;
1870 if (!(comment
->value
.bytes
= heap_alloc( len
)))
1873 return E_OUTOFMEMORY
;
1875 memcpy( comment
->value
.bytes
, start
, len
);
1876 comment
->value
.length
= len
;
1878 read_insert_node( reader
, parent
, node
);
1879 reader
->state
= READER_STATE_COMMENT
;
1883 static HRESULT
read_startcdata( struct reader
*reader
)
1885 struct node
*node
, *endnode
, *parent
;
1887 if (read_cmp( reader
, "<![CDATA[", 9 )) return WS_E_INVALID_FORMAT
;
1888 read_skip( reader
, 9 );
1890 if (!(parent
= find_parent( reader
))) return WS_E_INVALID_FORMAT
;
1892 if (!(node
= alloc_node( WS_XML_NODE_TYPE_CDATA
))) return E_OUTOFMEMORY
;
1893 if (!(endnode
= alloc_node( WS_XML_NODE_TYPE_END_CDATA
)))
1896 return E_OUTOFMEMORY
;
1898 list_add_tail( &node
->children
, &endnode
->entry
);
1899 endnode
->parent
= node
;
1901 read_insert_node( reader
, parent
, node
);
1902 reader
->state
= READER_STATE_STARTCDATA
;
1906 static HRESULT
read_cdata( struct reader
*reader
)
1908 unsigned int len
= 0, ch
, skip
;
1909 const unsigned char *start
;
1911 WS_XML_TEXT_NODE
*text
;
1912 WS_XML_UTF8_TEXT
*utf8
;
1914 start
= read_current_ptr( reader
);
1917 if (!read_cmp( reader
, "]]>", 3 )) break;
1918 if (!(ch
= read_utf8_char( reader
, &skip
))) return WS_E_INVALID_FORMAT
;
1919 read_skip( reader
, skip
);
1923 if (!(node
= alloc_node( WS_XML_NODE_TYPE_TEXT
))) return E_OUTOFMEMORY
;
1924 text
= (WS_XML_TEXT_NODE
*)node
;
1925 if (!(utf8
= alloc_utf8_text( start
, len
)))
1928 return E_OUTOFMEMORY
;
1930 text
->text
= &utf8
->text
;
1932 read_insert_node( reader
, reader
->current
, node
);
1933 reader
->state
= READER_STATE_CDATA
;
1937 static HRESULT
read_endcdata( struct reader
*reader
)
1939 struct node
*parent
;
1941 if (read_cmp( reader
, "]]>", 3 )) return WS_E_INVALID_FORMAT
;
1942 read_skip( reader
, 3 );
1944 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_TEXT
) parent
= reader
->current
->parent
;
1945 else parent
= reader
->current
;
1947 reader
->current
= LIST_ENTRY( list_tail( &parent
->children
), struct node
, entry
);
1948 reader
->last
= reader
->current
;
1949 reader
->state
= READER_STATE_ENDCDATA
;
1953 static HRESULT
read_node_text( struct reader
*reader
)
1959 if (read_end_of_data( reader
))
1961 reader
->current
= LIST_ENTRY( list_tail( &reader
->root
->children
), struct node
, entry
);
1962 reader
->last
= reader
->current
;
1963 reader
->state
= READER_STATE_EOF
;
1966 if (reader
->state
== READER_STATE_STARTCDATA
) return read_cdata( reader
);
1967 else if (reader
->state
== READER_STATE_CDATA
) return read_endcdata( reader
);
1968 else if (!read_cmp( reader
, "<?", 2 ))
1970 hr
= read_xmldecl( reader
);
1971 if (FAILED( hr
)) return hr
;
1973 else if (!read_cmp( reader
, "</", 2 )) return read_endelement_text( reader
);
1974 else if (!read_cmp( reader
, "<![CDATA[", 9 )) return read_startcdata( reader
);
1975 else if (!read_cmp( reader
, "<!--", 4 )) return read_comment( reader
);
1976 else if (!read_cmp( reader
, "<", 1 )) return read_element_text( reader
);
1977 else if (!read_cmp( reader
, "/>", 2 ) || !read_cmp( reader
, ">", 1 )) return read_startelement( reader
);
1978 else return read_text( reader
);
1982 static HRESULT
read_node_bin( struct reader
*reader
)
1987 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_TEXT
)
1989 reader
->current
= LIST_ENTRY( list_tail( &reader
->current
->parent
->children
), struct node
, entry
);
1990 reader
->last
= reader
->current
;
1991 reader
->state
= READER_STATE_ENDELEMENT
;
1994 if (read_end_of_data( reader
))
1996 reader
->current
= LIST_ENTRY( list_tail( &reader
->root
->children
), struct node
, entry
);
1997 reader
->last
= reader
->current
;
1998 reader
->state
= READER_STATE_EOF
;
2002 if ((hr
= read_peek( reader
, &type
)) != S_OK
) return hr
;
2003 if (type
== RECORD_ENDELEMENT
)
2005 return read_endelement_bin( reader
);
2007 else if (type
>= RECORD_SHORT_ELEMENT
&& type
<= RECORD_PREFIX_ELEMENT_Z
)
2009 return read_element_bin( reader
);
2012 FIXME( "unhandled record type %02x\n", type
);
2013 return WS_E_NOT_SUPPORTED
;
2016 static HRESULT
read_node( struct reader
*reader
)
2018 switch (reader
->input_enc
)
2020 case WS_XML_READER_ENCODING_TYPE_TEXT
: return read_node_text( reader
);
2021 case WS_XML_READER_ENCODING_TYPE_BINARY
: return read_node_bin( reader
);
2023 ERR( "unhandled encoding %u\n", reader
->input_enc
);
2024 return WS_E_NOT_SUPPORTED
;
2028 HRESULT
copy_node( WS_XML_READER
*handle
, struct node
**node
)
2030 struct reader
*reader
= (struct reader
*)handle
;
2031 const struct list
*ptr
;
2032 const struct node
*start
;
2035 EnterCriticalSection( &reader
->cs
);
2037 if (reader
->magic
!= READER_MAGIC
)
2039 LeaveCriticalSection( &reader
->cs
);
2040 return E_INVALIDARG
;
2043 if (reader
->current
!= reader
->root
) ptr
= &reader
->current
->entry
;
2044 else /* copy whole tree */
2046 if (!read_end_of_data( reader
))
2050 if ((hr
= read_node( reader
)) != S_OK
) goto done
;
2051 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_EOF
) break;
2054 ptr
= list_head( &reader
->root
->children
);
2057 start
= LIST_ENTRY( ptr
, struct node
, entry
);
2058 if (node_type( start
) == WS_XML_NODE_TYPE_EOF
) hr
= WS_E_INVALID_OPERATION
;
2059 else hr
= dup_tree( node
, start
);
2062 LeaveCriticalSection( &reader
->cs
);
2066 /**************************************************************************
2067 * WsReadEndElement [webservices.@]
2069 HRESULT WINAPI
WsReadEndElement( WS_XML_READER
*handle
, WS_ERROR
*error
)
2071 struct reader
*reader
= (struct reader
*)handle
;
2074 TRACE( "%p %p\n", handle
, error
);
2075 if (error
) FIXME( "ignoring error parameter\n" );
2077 if (!reader
) return E_INVALIDARG
;
2079 EnterCriticalSection( &reader
->cs
);
2081 if (reader
->magic
!= READER_MAGIC
)
2083 LeaveCriticalSection( &reader
->cs
);
2084 return E_INVALIDARG
;
2087 hr
= read_endelement( reader
);
2089 LeaveCriticalSection( &reader
->cs
);
2093 /**************************************************************************
2094 * WsReadNode [webservices.@]
2096 HRESULT WINAPI
WsReadNode( WS_XML_READER
*handle
, WS_ERROR
*error
)
2098 struct reader
*reader
= (struct reader
*)handle
;
2101 TRACE( "%p %p\n", handle
, error
);
2102 if (error
) FIXME( "ignoring error parameter\n" );
2104 if (!reader
) return E_INVALIDARG
;
2106 EnterCriticalSection( &reader
->cs
);
2108 if (reader
->magic
!= READER_MAGIC
)
2110 LeaveCriticalSection( &reader
->cs
);
2111 return E_INVALIDARG
;
2114 hr
= read_node( reader
);
2116 LeaveCriticalSection( &reader
->cs
);
2120 static HRESULT
skip_node( struct reader
*reader
)
2122 const struct node
*parent
;
2125 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_EOF
) return WS_E_INVALID_OPERATION
;
2126 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_ELEMENT
) parent
= reader
->current
;
2131 if ((hr
= read_node( reader
) != S_OK
) || !parent
) break;
2132 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_END_ELEMENT
) continue;
2133 if (reader
->current
->parent
== parent
) return read_node( reader
);
2139 /**************************************************************************
2140 * WsSkipNode [webservices.@]
2142 HRESULT WINAPI
WsSkipNode( WS_XML_READER
*handle
, WS_ERROR
*error
)
2144 struct reader
*reader
= (struct reader
*)handle
;
2147 TRACE( "%p %p\n", handle
, error
);
2148 if (error
) FIXME( "ignoring error parameter\n" );
2150 if (!reader
) return E_INVALIDARG
;
2152 EnterCriticalSection( &reader
->cs
);
2154 if (reader
->magic
!= READER_MAGIC
)
2156 LeaveCriticalSection( &reader
->cs
);
2157 return E_INVALIDARG
;
2160 hr
= skip_node( reader
);
2162 LeaveCriticalSection( &reader
->cs
);
2166 /**************************************************************************
2167 * WsReadStartElement [webservices.@]
2169 HRESULT WINAPI
WsReadStartElement( WS_XML_READER
*handle
, WS_ERROR
*error
)
2171 struct reader
*reader
= (struct reader
*)handle
;
2174 TRACE( "%p %p\n", handle
, error
);
2175 if (error
) FIXME( "ignoring error parameter\n" );
2177 if (!reader
) return E_INVALIDARG
;
2179 EnterCriticalSection( &reader
->cs
);
2181 if (reader
->magic
!= READER_MAGIC
)
2183 LeaveCriticalSection( &reader
->cs
);
2184 return E_INVALIDARG
;
2187 hr
= read_startelement( reader
);
2189 LeaveCriticalSection( &reader
->cs
);
2193 /**************************************************************************
2194 * WsReadToStartElement [webservices.@]
2196 HRESULT WINAPI
WsReadToStartElement( WS_XML_READER
*handle
, const WS_XML_STRING
*localname
,
2197 const WS_XML_STRING
*ns
, BOOL
*found
, WS_ERROR
*error
)
2199 struct reader
*reader
= (struct reader
*)handle
;
2202 TRACE( "%p %s %s %p %p\n", handle
, debugstr_xmlstr(localname
), debugstr_xmlstr(ns
), found
, error
);
2203 if (error
) FIXME( "ignoring error parameter\n" );
2205 if (!reader
) return E_INVALIDARG
;
2206 if (localname
|| ns
) FIXME( "name and/or namespace not verified\n" );
2208 EnterCriticalSection( &reader
->cs
);
2210 if (reader
->magic
!= READER_MAGIC
)
2212 LeaveCriticalSection( &reader
->cs
);
2213 return E_INVALIDARG
;
2216 hr
= read_to_startelement( reader
, found
);
2218 LeaveCriticalSection( &reader
->cs
);
2222 BOOL
move_to_root_element( struct node
*root
, struct node
**current
)
2227 if (!(ptr
= list_head( &root
->children
))) return FALSE
;
2228 node
= LIST_ENTRY( ptr
, struct node
, entry
);
2229 if (node_type( node
) == WS_XML_NODE_TYPE_ELEMENT
)
2234 while ((ptr
= list_next( &root
->children
, &node
->entry
)))
2236 struct node
*next
= LIST_ENTRY( ptr
, struct node
, entry
);
2237 if (node_type( next
) == WS_XML_NODE_TYPE_ELEMENT
)
2247 BOOL
move_to_next_element( struct node
**current
)
2250 struct node
*node
= *current
, *parent
= (*current
)->parent
;
2252 if (!parent
) return FALSE
;
2253 while ((ptr
= list_next( &parent
->children
, &node
->entry
)))
2255 struct node
*next
= LIST_ENTRY( ptr
, struct node
, entry
);
2256 if (node_type( next
) == WS_XML_NODE_TYPE_ELEMENT
)
2266 BOOL
move_to_prev_element( struct node
**current
)
2269 struct node
*node
= *current
, *parent
= (*current
)->parent
;
2271 if (!parent
) return FALSE
;
2272 while ((ptr
= list_prev( &parent
->children
, &node
->entry
)))
2274 struct node
*prev
= LIST_ENTRY( ptr
, struct node
, entry
);
2275 if (node_type( prev
) == WS_XML_NODE_TYPE_ELEMENT
)
2285 BOOL
move_to_child_element( struct node
**current
)
2288 struct node
*child
, *node
= *current
;
2290 if (!(ptr
= list_head( &node
->children
))) return FALSE
;
2291 child
= LIST_ENTRY( ptr
, struct node
, entry
);
2292 if (node_type( child
) == WS_XML_NODE_TYPE_ELEMENT
)
2297 while ((ptr
= list_next( &node
->children
, &child
->entry
)))
2299 struct node
*next
= LIST_ENTRY( ptr
, struct node
, entry
);
2300 if (node_type( next
) == WS_XML_NODE_TYPE_ELEMENT
)
2310 BOOL
move_to_end_element( struct node
**current
)
2313 struct node
*node
= *current
;
2315 if (node_type( node
) != WS_XML_NODE_TYPE_ELEMENT
) return FALSE
;
2317 if ((ptr
= list_tail( &node
->children
)))
2319 struct node
*tail
= LIST_ENTRY( ptr
, struct node
, entry
);
2320 if (node_type( tail
) == WS_XML_NODE_TYPE_END_ELEMENT
)
2329 BOOL
move_to_parent_element( struct node
**current
)
2331 struct node
*parent
= (*current
)->parent
;
2333 if (parent
&& (node_type( parent
) == WS_XML_NODE_TYPE_ELEMENT
||
2334 node_type( parent
) == WS_XML_NODE_TYPE_BOF
))
2342 BOOL
move_to_first_node( struct node
**current
)
2345 struct node
*node
= *current
;
2347 if ((ptr
= list_head( &node
->parent
->children
)))
2349 *current
= LIST_ENTRY( ptr
, struct node
, entry
);
2355 BOOL
move_to_next_node( struct node
**current
)
2358 struct node
*node
= *current
;
2360 if ((ptr
= list_next( &node
->parent
->children
, &node
->entry
)))
2362 *current
= LIST_ENTRY( ptr
, struct node
, entry
);
2368 BOOL
move_to_prev_node( struct node
**current
)
2371 struct node
*node
= *current
;
2373 if ((ptr
= list_prev( &node
->parent
->children
, &node
->entry
)))
2375 *current
= LIST_ENTRY( ptr
, struct node
, entry
);
2381 BOOL
move_to_bof( struct node
*root
, struct node
**current
)
2387 BOOL
move_to_eof( struct node
*root
, struct node
**current
)
2390 if ((ptr
= list_tail( &root
->children
)))
2392 *current
= LIST_ENTRY( ptr
, struct node
, entry
);
2398 BOOL
move_to_child_node( struct node
**current
)
2401 struct node
*node
= *current
;
2403 if ((ptr
= list_head( &node
->children
)))
2405 *current
= LIST_ENTRY( ptr
, struct node
, entry
);
2411 BOOL
move_to_parent_node( struct node
**current
)
2413 struct node
*parent
= (*current
)->parent
;
2414 if (!parent
) return FALSE
;
2419 static HRESULT
read_move_to( struct reader
*reader
, WS_MOVE_TO move
, BOOL
*found
)
2421 BOOL success
= FALSE
;
2424 if (!read_end_of_data( reader
))
2426 while (reader
->state
!= READER_STATE_EOF
&& (hr
= read_node( reader
)) == S_OK
) { /* nothing */ };
2427 if (hr
!= S_OK
) return hr
;
2431 case WS_MOVE_TO_ROOT_ELEMENT
:
2432 success
= move_to_root_element( reader
->root
, &reader
->current
);
2435 case WS_MOVE_TO_NEXT_ELEMENT
:
2436 success
= move_to_next_element( &reader
->current
);
2439 case WS_MOVE_TO_PREVIOUS_ELEMENT
:
2440 success
= move_to_prev_element( &reader
->current
);
2443 case WS_MOVE_TO_CHILD_ELEMENT
:
2444 success
= move_to_child_element( &reader
->current
);
2447 case WS_MOVE_TO_END_ELEMENT
:
2448 success
= move_to_end_element( &reader
->current
);
2451 case WS_MOVE_TO_PARENT_ELEMENT
:
2452 success
= move_to_parent_element( &reader
->current
);
2455 case WS_MOVE_TO_FIRST_NODE
:
2456 success
= move_to_first_node( &reader
->current
);
2459 case WS_MOVE_TO_NEXT_NODE
:
2460 success
= move_to_next_node( &reader
->current
);
2463 case WS_MOVE_TO_PREVIOUS_NODE
:
2464 success
= move_to_prev_node( &reader
->current
);
2467 case WS_MOVE_TO_CHILD_NODE
:
2468 success
= move_to_child_node( &reader
->current
);
2471 case WS_MOVE_TO_BOF
:
2472 success
= move_to_bof( reader
->root
, &reader
->current
);
2475 case WS_MOVE_TO_EOF
:
2476 success
= move_to_eof( reader
->root
, &reader
->current
);
2480 FIXME( "unhandled move %u\n", move
);
2489 return success
? S_OK
: WS_E_INVALID_FORMAT
;
2492 /**************************************************************************
2493 * WsMoveReader [webservices.@]
2495 HRESULT WINAPI
WsMoveReader( WS_XML_READER
*handle
, WS_MOVE_TO move
, BOOL
*found
, WS_ERROR
*error
)
2497 struct reader
*reader
= (struct reader
*)handle
;
2500 TRACE( "%p %u %p %p\n", handle
, move
, found
, error
);
2501 if (error
) FIXME( "ignoring error parameter\n" );
2503 if (!reader
) return E_INVALIDARG
;
2505 EnterCriticalSection( &reader
->cs
);
2507 if (reader
->magic
!= READER_MAGIC
)
2509 LeaveCriticalSection( &reader
->cs
);
2510 return E_INVALIDARG
;
2513 if (!reader
->input_type
)
2515 LeaveCriticalSection( &reader
->cs
);
2516 return WS_E_INVALID_OPERATION
;
2519 hr
= read_move_to( reader
, move
, found
);
2521 LeaveCriticalSection( &reader
->cs
);
2525 /**************************************************************************
2526 * WsReadStartAttribute [webservices.@]
2528 HRESULT WINAPI
WsReadStartAttribute( WS_XML_READER
*handle
, ULONG index
, WS_ERROR
*error
)
2530 struct reader
*reader
= (struct reader
*)handle
;
2531 const WS_XML_ELEMENT_NODE
*elem
;
2533 TRACE( "%p %u %p\n", handle
, index
, error
);
2534 if (error
) FIXME( "ignoring error parameter\n" );
2536 if (!reader
) return E_INVALIDARG
;
2538 EnterCriticalSection( &reader
->cs
);
2540 if (reader
->magic
!= READER_MAGIC
)
2542 LeaveCriticalSection( &reader
->cs
);
2543 return E_INVALIDARG
;
2546 elem
= &reader
->current
->hdr
;
2547 if (reader
->state
!= READER_STATE_STARTELEMENT
|| index
>= elem
->attributeCount
)
2549 LeaveCriticalSection( &reader
->cs
);
2550 return WS_E_INVALID_FORMAT
;
2553 reader
->current_attr
= index
;
2554 reader
->state
= READER_STATE_STARTATTRIBUTE
;
2556 LeaveCriticalSection( &reader
->cs
);
2560 /**************************************************************************
2561 * WsReadEndAttribute [webservices.@]
2563 HRESULT WINAPI
WsReadEndAttribute( WS_XML_READER
*handle
, WS_ERROR
*error
)
2565 struct reader
*reader
= (struct reader
*)handle
;
2567 TRACE( "%p %p\n", handle
, error
);
2568 if (error
) FIXME( "ignoring error parameter\n" );
2570 if (!reader
) return E_INVALIDARG
;
2572 EnterCriticalSection( &reader
->cs
);
2574 if (reader
->magic
!= READER_MAGIC
)
2576 LeaveCriticalSection( &reader
->cs
);
2577 return E_INVALIDARG
;
2580 if (reader
->state
!= READER_STATE_STARTATTRIBUTE
)
2582 LeaveCriticalSection( &reader
->cs
);
2583 return WS_E_INVALID_FORMAT
;
2586 reader
->state
= READER_STATE_STARTELEMENT
;
2588 LeaveCriticalSection( &reader
->cs
);
2592 static HRESULT
find_namespace( struct reader
*reader
, const WS_XML_STRING
*prefix
, const WS_XML_STRING
**ns
)
2594 const struct node
*node
;
2595 for (node
= reader
->current
->parent
; node_type( node
) == WS_XML_NODE_TYPE_ELEMENT
; node
= node
->parent
)
2597 const WS_XML_ELEMENT_NODE
*elem
= &node
->hdr
;
2599 for (i
= 0; i
< elem
->attributeCount
; i
++)
2601 if (!elem
->attributes
[i
]->isXmlNs
) continue;
2602 if (WsXmlStringEquals( elem
->attributes
[i
]->prefix
, prefix
, NULL
) != S_OK
) continue;
2603 *ns
= elem
->attributes
[i
]->ns
;
2607 return WS_E_INVALID_FORMAT
;
2610 static HRESULT
read_qualified_name( struct reader
*reader
, WS_HEAP
*heap
, WS_XML_STRING
*prefix_ret
,
2611 WS_XML_STRING
*localname_ret
, WS_XML_STRING
*ns_ret
)
2613 const WS_XML_TEXT_NODE
*node
= (const WS_XML_TEXT_NODE
*)reader
->current
;
2614 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)node
->text
;
2615 unsigned char *prefix_bytes
, *localname_bytes
, *ns_bytes
;
2616 const unsigned char *ptr
= utf8
->value
.bytes
;
2617 WS_XML_STRING prefix
, localname
, empty
= {0, NULL
};
2618 const WS_XML_STRING
*ns
= &empty
;
2619 ULONG len
= utf8
->value
.length
;
2622 while (len
&& read_isspace( *ptr
)) { ptr
++; len
--; }
2623 while (len
&& read_isspace( ptr
[len
- 1] )) { len
--; }
2624 if (!len
) return WS_E_INVALID_FORMAT
;
2626 if ((hr
= split_name( ptr
, len
, (const unsigned char **)&prefix
.bytes
, &prefix
.length
,
2627 (const unsigned char **)&localname
.bytes
, &localname
.length
)) != S_OK
) return hr
;
2629 if (!localname
.length
) return WS_E_INVALID_FORMAT
;
2630 if (prefix
.length
&& (hr
= find_namespace( reader
, &prefix
, &ns
)) != S_OK
) return hr
;
2632 if (!(prefix_bytes
= ws_alloc( heap
, prefix
.length
))) return WS_E_QUOTA_EXCEEDED
;
2633 memcpy( prefix_bytes
, prefix
.bytes
, prefix
.length
);
2635 if (!(localname_bytes
= ws_alloc( heap
, localname
.length
)))
2637 ws_free( heap
, prefix_bytes
, prefix
.length
);
2638 return WS_E_QUOTA_EXCEEDED
;
2640 memcpy( localname_bytes
, localname
.bytes
, localname
.length
);
2642 if (!(ns_bytes
= ws_alloc( heap
, ns
->length
)))
2644 ws_free( heap
, prefix_bytes
, prefix
.length
);
2645 ws_free( heap
, localname_bytes
, localname
.length
);
2646 return WS_E_QUOTA_EXCEEDED
;
2648 memcpy( ns_bytes
, ns
->bytes
, ns
->length
);
2650 prefix_ret
->bytes
= prefix_bytes
;
2651 prefix_ret
->length
= prefix
.length
;
2653 localname_ret
->bytes
= localname_bytes
;
2654 localname_ret
->length
= localname
.length
;
2656 ns_ret
->bytes
= ns_bytes
;
2657 ns_ret
->length
= ns
->length
;
2662 /**************************************************************************
2663 * WsReadQualifiedName [webservices.@]
2665 HRESULT WINAPI
WsReadQualifiedName( WS_XML_READER
*handle
, WS_HEAP
*heap
, WS_XML_STRING
*prefix
,
2666 WS_XML_STRING
*localname
, WS_XML_STRING
*ns
,
2669 struct reader
*reader
= (struct reader
*)handle
;
2672 TRACE( "%p %p %p %p %p %p\n", handle
, heap
, prefix
, localname
, ns
, error
);
2673 if (error
) FIXME( "ignoring error parameter\n" );
2675 if (!reader
|| !heap
) return E_INVALIDARG
;
2677 EnterCriticalSection( &reader
->cs
);
2679 if (reader
->magic
!= READER_MAGIC
)
2681 LeaveCriticalSection( &reader
->cs
);
2682 return E_INVALIDARG
;
2685 if (!reader
->input_type
)
2687 LeaveCriticalSection( &reader
->cs
);
2688 return WS_E_INVALID_OPERATION
;
2693 LeaveCriticalSection( &reader
->cs
);
2694 return E_INVALIDARG
;
2697 if (reader
->state
!= READER_STATE_TEXT
)
2699 LeaveCriticalSection( &reader
->cs
);
2700 return WS_E_INVALID_FORMAT
;
2703 hr
= read_qualified_name( reader
, heap
, prefix
, localname
, ns
);
2705 LeaveCriticalSection( &reader
->cs
);
2709 static WCHAR
*xmltext_to_widechar( WS_HEAP
*heap
, const WS_XML_TEXT
*text
)
2713 switch (text
->textType
)
2715 case WS_XML_TEXT_TYPE_UTF8
:
2717 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
;
2718 int len
= MultiByteToWideChar( CP_UTF8
, 0, (char *)utf8
->value
.bytes
, utf8
->value
.length
, NULL
, 0 );
2719 if (!(ret
= ws_alloc( heap
, (len
+ 1) * sizeof(WCHAR
) ))) return NULL
;
2720 MultiByteToWideChar( CP_UTF8
, 0, (char *)utf8
->value
.bytes
, utf8
->value
.length
, ret
, len
);
2725 FIXME( "unhandled type %u\n", text
->textType
);
2732 #define MAX_INT8 0x7f
2733 #define MIN_INT8 (-MAX_INT8 - 1)
2734 #define MAX_INT16 0x7fff
2735 #define MIN_INT16 (-MAX_INT16 - 1)
2736 #define MAX_INT32 0x7fffffff
2737 #define MIN_INT32 (-MAX_INT32 - 1)
2738 #define MAX_INT64 (((INT64)0x7fffffff << 32) | 0xffffffff)
2739 #define MIN_INT64 (-MAX_INT64 - 1)
2740 #define MAX_UINT8 0xff
2741 #define MAX_UINT16 0xffff
2742 #define MAX_UINT32 0xffffffff
2743 #define MAX_UINT64 (((UINT64)0xffffffff << 32) | 0xffffffff)
2745 static HRESULT
str_to_int64( const unsigned char *str
, ULONG len
, INT64 min
, INT64 max
, INT64
*ret
)
2747 BOOL negative
= FALSE
;
2748 const unsigned char *ptr
= str
;
2751 while (len
&& read_isspace( *ptr
)) { ptr
++; len
--; }
2752 while (len
&& read_isspace( ptr
[len
- 1] )) { len
--; }
2753 if (!len
) return WS_E_INVALID_FORMAT
;
2761 if (!len
) return WS_E_INVALID_FORMAT
;
2767 if (!isdigit( *ptr
)) return WS_E_INVALID_FORMAT
;
2769 if (negative
) val
= -val
;
2771 if ((!negative
&& (*ret
> max
/ 10 || *ret
* 10 > max
- val
)) ||
2772 (negative
&& (*ret
< min
/ 10 || *ret
* 10 < min
- val
)))
2774 return WS_E_NUMERIC_OVERFLOW
;
2776 *ret
= *ret
* 10 + val
;
2783 static HRESULT
str_to_uint64( const unsigned char *str
, ULONG len
, UINT64 max
, UINT64
*ret
)
2785 const unsigned char *ptr
= str
;
2788 while (len
&& read_isspace( *ptr
)) { ptr
++; len
--; }
2789 while (len
&& read_isspace( ptr
[len
- 1] )) { len
--; }
2790 if (!len
) return WS_E_INVALID_FORMAT
;
2796 if (!isdigit( *ptr
)) return WS_E_INVALID_FORMAT
;
2799 if ((*ret
> max
/ 10 || *ret
* 10 > max
- val
)) return WS_E_NUMERIC_OVERFLOW
;
2800 *ret
= *ret
* 10 + val
;
2807 BOOL
set_fpword( unsigned short new, unsigned short *old
)
2809 #if defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__))
2810 unsigned short fpword
;
2812 __asm__
__volatile__( "fstcw %0" : "=m" (fpword
) );
2815 __asm__
__volatile__( "fldcw %0" : : "m" (fpword
) );
2818 FIXME( "not implemented\n" );
2823 void restore_fpword( unsigned short fpword
)
2825 #if defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__))
2826 __asm__
__volatile__( "fldcw %0" : : "m" (fpword
) );
2828 FIXME( "not implemented\n" );
2832 static HRESULT
str_to_double( const unsigned char *str
, ULONG len
, double *ret
)
2834 static const unsigned __int64 nan
= 0xfff8000000000000;
2835 static const unsigned __int64 inf
= 0x7ff0000000000000;
2836 static const unsigned __int64 inf_min
= 0xfff0000000000000;
2837 HRESULT hr
= WS_E_INVALID_FORMAT
;
2838 const unsigned char *p
= str
, *q
;
2839 int sign
= 1, exp_sign
= 1, exp
= 0, exp_tmp
= 0, neg_exp
, i
, nb_digits
, have_digits
;
2840 unsigned __int64 val
= 0, tmp
;
2841 long double exp_val
= 1.0, exp_mul
= 10.0;
2842 unsigned short fpword
;
2844 while (len
&& read_isspace( *p
)) { p
++; len
--; }
2845 while (len
&& read_isspace( p
[len
- 1] )) { len
--; }
2846 if (!len
) return WS_E_INVALID_FORMAT
;
2848 if (len
== 3 && !memcmp( p
, "NaN", 3 ))
2850 *(unsigned __int64
*)ret
= nan
;
2853 else if (len
== 3 && !memcmp( p
, "INF", 3 ))
2855 *(unsigned __int64
*)ret
= inf
;
2858 else if (len
== 4 && !memcmp( p
, "-INF", 4 ))
2860 *(unsigned __int64
*)ret
= inf_min
;
2870 else if (*p
== '+') { p
++; len
--; };
2871 if (!len
) return S_OK
;
2873 if (!set_fpword( 0x37f, &fpword
)) return E_NOTIMPL
;
2876 while (len
&& isdigit( *q
)) { q
++; len
--; }
2877 have_digits
= nb_digits
= q
- p
;
2878 for (i
= 0; i
< nb_digits
; i
++)
2880 tmp
= val
* 10 + p
[i
] - '0';
2881 if (val
> MAX_UINT64
/ 10 || tmp
< val
)
2883 for (; i
< nb_digits
; i
++) exp
++;
2894 while (len
&& isdigit( *q
)) { q
++; len
--; };
2895 have_digits
|= nb_digits
= q
- p
;
2896 for (i
= 0; i
< nb_digits
; i
++)
2898 tmp
= val
* 10 + p
[i
] - '0';
2899 if (val
> MAX_UINT64
/ 10 || tmp
< val
) break;
2904 if (len
> 1 && tolower(*q
) == 'e')
2906 if (!have_digits
) goto done
;
2913 else if (*p
== '+') { p
++; len
--; };
2916 while (len
&& isdigit( *q
)) { q
++; len
--; };
2918 if (!nb_digits
|| len
) goto done
;
2919 for (i
= 0; i
< nb_digits
; i
++)
2921 if (exp_tmp
> MAX_INT32
/ 10 || (exp_tmp
= exp_tmp
* 10 + p
[i
] - '0') < 0)
2922 exp_tmp
= MAX_INT32
;
2924 exp_tmp
*= exp_sign
;
2926 if (exp
< 0 && exp_tmp
< 0 && exp
+ exp_tmp
>= 0) exp
= MIN_INT32
;
2927 else if (exp
> 0 && exp_tmp
> 0 && exp
+ exp_tmp
< 0) exp
= MAX_INT32
;
2928 else exp
+= exp_tmp
;
2931 if (!have_digits
|| len
) goto done
;
2933 if ((neg_exp
= exp
< 0)) exp
= -exp
;
2934 for (; exp
; exp
>>= 1)
2936 if (exp
& 1) exp_val
*= exp_mul
;
2940 *ret
= sign
* (neg_exp
? val
/ exp_val
: val
* exp_val
);
2944 restore_fpword( fpword
);
2948 static HRESULT
str_to_guid( const unsigned char *str
, ULONG len
, GUID
*ret
)
2950 static const unsigned char hex
[] =
2952 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x00 */
2953 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x10 */
2954 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x20 */
2955 0,1,2,3,4,5,6,7,8,9,0,0,0,0,0,0, /* 0x30 */
2956 0,10,11,12,13,14,15,0,0,0,0,0,0,0,0,0, /* 0x40 */
2957 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 0x50 */
2958 0,10,11,12,13,14,15 /* 0x60 */
2960 const unsigned char *p
= str
;
2963 while (len
&& read_isspace( *p
)) { p
++; len
--; }
2964 while (len
&& read_isspace( p
[len
- 1] )) { len
--; }
2965 if (len
!= 36) return WS_E_INVALID_FORMAT
;
2967 if (p
[8] != '-' || p
[13] != '-' || p
[18] != '-' || p
[23] != '-')
2968 return WS_E_INVALID_FORMAT
;
2970 for (i
= 0; i
< 36; i
++)
2972 if (i
== 8 || i
== 13 || i
== 18 || i
== 23) continue;
2973 if (p
[i
] > 'f' || (!hex
[p
[i
]] && p
[i
] != '0')) return WS_E_INVALID_FORMAT
;
2976 ret
->Data1
= hex
[p
[0]] << 28 | hex
[p
[1]] << 24 | hex
[p
[2]] << 20 | hex
[p
[3]] << 16 |
2977 hex
[p
[4]] << 12 | hex
[p
[5]] << 8 | hex
[p
[6]] << 4 | hex
[p
[7]];
2979 ret
->Data2
= hex
[p
[9]] << 12 | hex
[p
[10]] << 8 | hex
[p
[11]] << 4 | hex
[p
[12]];
2980 ret
->Data3
= hex
[p
[14]] << 12 | hex
[p
[15]] << 8 | hex
[p
[16]] << 4 | hex
[p
[17]];
2982 ret
->Data4
[0] = hex
[p
[19]] << 4 | hex
[p
[20]];
2983 ret
->Data4
[1] = hex
[p
[21]] << 4 | hex
[p
[22]];
2984 ret
->Data4
[2] = hex
[p
[24]] << 4 | hex
[p
[25]];
2985 ret
->Data4
[3] = hex
[p
[26]] << 4 | hex
[p
[27]];
2986 ret
->Data4
[4] = hex
[p
[28]] << 4 | hex
[p
[29]];
2987 ret
->Data4
[5] = hex
[p
[30]] << 4 | hex
[p
[31]];
2988 ret
->Data4
[6] = hex
[p
[32]] << 4 | hex
[p
[33]];
2989 ret
->Data4
[7] = hex
[p
[34]] << 4 | hex
[p
[35]];
2994 static inline unsigned char decode_char( unsigned char c
)
2996 if (c
>= 'A' && c
<= 'Z') return c
- 'A';
2997 if (c
>= 'a' && c
<= 'z') return c
- 'a' + 26;
2998 if (c
>= '0' && c
<= '9') return c
- '0' + 52;
2999 if (c
== '+') return 62;
3000 if (c
== '/') return 63;
3004 static ULONG
decode_base64( const unsigned char *base64
, ULONG len
, unsigned char *buf
)
3007 unsigned char c0
, c1
, c2
, c3
;
3008 const unsigned char *p
= base64
;
3012 if ((c0
= decode_char( p
[0] )) > 63) return 0;
3013 if ((c1
= decode_char( p
[1] )) > 63) return 0;
3014 if ((c2
= decode_char( p
[2] )) > 63) return 0;
3015 if ((c3
= decode_char( p
[3] )) > 63) return 0;
3016 buf
[i
+ 0] = (c0
<< 2) | (c1
>> 4);
3017 buf
[i
+ 1] = (c1
<< 4) | (c2
>> 2);
3018 buf
[i
+ 2] = (c2
<< 6) | c3
;
3025 if ((c0
= decode_char( p
[0] )) > 63) return 0;
3026 if ((c1
= decode_char( p
[1] )) > 63) return 0;
3027 buf
[i
] = (c0
<< 2) | (c1
>> 4);
3030 else if (p
[3] == '=')
3032 if ((c0
= decode_char( p
[0] )) > 63) return 0;
3033 if ((c1
= decode_char( p
[1] )) > 63) return 0;
3034 if ((c2
= decode_char( p
[2] )) > 63) return 0;
3035 buf
[i
+ 0] = (c0
<< 2) | (c1
>> 4);
3036 buf
[i
+ 1] = (c1
<< 4) | (c2
>> 2);
3041 if ((c0
= decode_char( p
[0] )) > 63) return 0;
3042 if ((c1
= decode_char( p
[1] )) > 63) return 0;
3043 if ((c2
= decode_char( p
[2] )) > 63) return 0;
3044 if ((c3
= decode_char( p
[3] )) > 63) return 0;
3045 buf
[i
+ 0] = (c0
<< 2) | (c1
>> 4);
3046 buf
[i
+ 1] = (c1
<< 4) | (c2
>> 2);
3047 buf
[i
+ 2] = (c2
<< 6) | c3
;
3053 static HRESULT
str_to_bytes( const unsigned char *str
, ULONG len
, WS_HEAP
*heap
, WS_BYTES
*ret
)
3055 const unsigned char *p
= str
;
3057 while (len
&& read_isspace( *p
)) { p
++; len
--; }
3058 while (len
&& read_isspace( p
[len
- 1] )) { len
--; }
3060 if (len
% 4) return WS_E_INVALID_FORMAT
;
3061 if (!(ret
->bytes
= ws_alloc( heap
, len
* 3 / 4 ))) return WS_E_QUOTA_EXCEEDED
;
3062 ret
->length
= decode_base64( p
, len
, ret
->bytes
);
3066 static const int month_offsets
[2][12] =
3068 {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334},
3069 {0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335}
3072 static inline int valid_day( int year
, int month
, int day
)
3074 return day
> 0 && day
<= month_days
[leap_year( year
)][month
- 1];
3077 static inline int leap_days_before( int year
)
3079 return (year
- 1) / 4 - (year
- 1) / 100 + (year
- 1) / 400;
3082 static HRESULT
str_to_datetime( const unsigned char *bytes
, ULONG len
, WS_DATETIME
*ret
)
3084 const unsigned char *p
= bytes
, *q
;
3085 int year
, month
, day
, hour
, min
, sec
, sec_frac
= 0, tz_hour
, tz_min
, tz_neg
;
3087 while (len
&& read_isspace( *p
)) { p
++; len
--; }
3088 while (len
&& read_isspace( p
[len
- 1] )) { len
--; }
3091 while (len
&& isdigit( *q
)) { q
++; len
--; };
3092 if (q
- p
!= 4 || !len
|| *q
!= '-') return WS_E_INVALID_FORMAT
;
3093 year
= (p
[0] - '0') * 1000 + (p
[1] - '0') * 100 + (p
[2] - '0') * 10 + p
[3] - '0';
3094 if (year
< 1) return WS_E_INVALID_FORMAT
;
3097 while (len
&& isdigit( *q
)) { q
++; len
--; };
3098 if (q
- p
!= 2 || !len
|| *q
!= '-') return WS_E_INVALID_FORMAT
;
3099 month
= (p
[0] - '0') * 10 + p
[1] - '0';
3100 if (month
< 1 || month
> 12) return WS_E_INVALID_FORMAT
;
3103 while (len
&& isdigit( *q
)) { q
++; len
--; };
3104 if (q
- p
!= 2 || !len
|| *q
!= 'T') return WS_E_INVALID_FORMAT
;
3105 day
= (p
[0] - '0') * 10 + p
[1] - '0';
3106 if (!valid_day( year
, month
, day
)) return WS_E_INVALID_FORMAT
;
3109 while (len
&& isdigit( *q
)) { q
++; len
--; };
3110 if (q
- p
!= 2 || !len
|| *q
!= ':') return WS_E_INVALID_FORMAT
;
3111 hour
= (p
[0] - '0') * 10 + p
[1] - '0';
3112 if (hour
> 24) return WS_E_INVALID_FORMAT
;
3115 while (len
&& isdigit( *q
)) { q
++; len
--; };
3116 if (q
- p
!= 2 || !len
|| *q
!= ':') return WS_E_INVALID_FORMAT
;
3117 min
= (p
[0] - '0') * 10 + p
[1] - '0';
3118 if (min
> 59 || (min
> 0 && hour
== 24)) return WS_E_INVALID_FORMAT
;
3121 while (len
&& isdigit( *q
)) { q
++; len
--; };
3122 if (q
- p
!= 2 || !len
) return WS_E_INVALID_FORMAT
;
3123 sec
= (p
[0] - '0') * 10 + p
[1] - '0';
3124 if (sec
> 59 || (sec
> 0 && hour
== 24)) return WS_E_INVALID_FORMAT
;
3128 unsigned int i
, nb_digits
, mul
= TICKS_PER_SEC
/ 10;
3130 while (len
&& isdigit( *q
)) { q
++; len
--; };
3132 if (nb_digits
< 1 || nb_digits
> 7) return WS_E_INVALID_FORMAT
;
3133 for (i
= 0; i
< nb_digits
; i
++)
3135 sec_frac
+= (p
[i
] - '0') * mul
;
3141 if (--len
) return WS_E_INVALID_FORMAT
;
3142 tz_hour
= tz_min
= tz_neg
= 0;
3143 ret
->format
= WS_DATETIME_FORMAT_UTC
;
3145 else if (*q
== '+' || *q
== '-')
3147 tz_neg
= (*q
== '-') ? 1 : 0;
3150 while (len
&& isdigit( *q
)) { q
++; len
--; };
3151 if (q
- p
!= 2 || !len
|| *q
!= ':') return WS_E_INVALID_FORMAT
;
3152 tz_hour
= (p
[0] - '0') * 10 + p
[1] - '0';
3153 if (tz_hour
> 14) return WS_E_INVALID_FORMAT
;
3156 while (len
&& isdigit( *q
)) { q
++; len
--; };
3157 if (q
- p
!= 2 || len
) return WS_E_INVALID_FORMAT
;
3158 tz_min
= (p
[0] - '0') * 10 + p
[1] - '0';
3159 if (tz_min
> 59 || (tz_min
> 0 && tz_hour
== 14)) return WS_E_INVALID_FORMAT
;
3161 ret
->format
= WS_DATETIME_FORMAT_LOCAL
;
3163 else return WS_E_INVALID_FORMAT
;
3165 ret
->ticks
= ((year
- 1) * 365 + leap_days_before( year
)) * TICKS_PER_DAY
;
3166 ret
->ticks
+= month_offsets
[leap_year( year
)][month
- 1] * TICKS_PER_DAY
;
3167 ret
->ticks
+= (day
- 1) * TICKS_PER_DAY
;
3168 ret
->ticks
+= hour
* TICKS_PER_HOUR
;
3169 ret
->ticks
+= min
* TICKS_PER_MIN
;
3170 ret
->ticks
+= sec
* TICKS_PER_SEC
;
3171 ret
->ticks
+= sec_frac
;
3175 if (tz_hour
* TICKS_PER_HOUR
+ tz_min
* TICKS_PER_MIN
+ ret
->ticks
> TICKS_MAX
)
3176 return WS_E_INVALID_FORMAT
;
3177 ret
->ticks
+= tz_hour
* TICKS_PER_HOUR
;
3178 ret
->ticks
+= tz_min
* TICKS_PER_MIN
;
3182 if (tz_hour
* TICKS_PER_HOUR
+ tz_min
* TICKS_PER_MIN
> ret
->ticks
)
3183 return WS_E_INVALID_FORMAT
;
3184 ret
->ticks
-= tz_hour
* TICKS_PER_HOUR
;
3185 ret
->ticks
-= tz_min
* TICKS_PER_MIN
;
3191 /**************************************************************************
3192 * WsDateTimeToFileTime [webservices.@]
3194 HRESULT WINAPI
WsDateTimeToFileTime( const WS_DATETIME
*dt
, FILETIME
*ft
, WS_ERROR
*error
)
3196 unsigned __int64 ticks
;
3198 TRACE( "%p %p %p\n", dt
, ft
, error
);
3199 if (error
) FIXME( "ignoring error parameter\n" );
3201 if (!dt
|| !ft
) return E_INVALIDARG
;
3203 if (dt
->ticks
< TICKS_1601_01_01
) return WS_E_INVALID_FORMAT
;
3204 ticks
= dt
->ticks
- TICKS_1601_01_01
;
3205 ft
->dwHighDateTime
= ticks
>> 32;
3206 ft
->dwLowDateTime
= (DWORD
)ticks
;
3210 /**************************************************************************
3211 * WsFileTimeToDateTime [webservices.@]
3213 HRESULT WINAPI
WsFileTimeToDateTime( const FILETIME
*ft
, WS_DATETIME
*dt
, WS_ERROR
*error
)
3215 unsigned __int64 ticks
;
3217 TRACE( "%p %p %p\n", ft
, dt
, error
);
3218 if (error
) FIXME( "ignoring error parameter\n" );
3220 if (!dt
|| !ft
) return E_INVALIDARG
;
3222 ticks
= ((unsigned __int64
)ft
->dwHighDateTime
<< 32) | ft
->dwLowDateTime
;
3223 if (ticks
> MAX_UINT64
- TICKS_1601_01_01
) return WS_E_NUMERIC_OVERFLOW
;
3224 if (ticks
+ TICKS_1601_01_01
> TICKS_MAX
) return WS_E_INVALID_FORMAT
;
3225 dt
->ticks
= ticks
+ TICKS_1601_01_01
;
3226 dt
->format
= WS_DATETIME_FORMAT_UTC
;
3230 static HRESULT
read_get_node_text( struct reader
*reader
, WS_XML_UTF8_TEXT
**ret
)
3232 WS_XML_TEXT_NODE
*text
;
3234 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_TEXT
)
3235 return WS_E_INVALID_FORMAT
;
3237 text
= (WS_XML_TEXT_NODE
*)&reader
->current
->hdr
.node
;
3238 if (text
->text
->textType
!= WS_XML_TEXT_TYPE_UTF8
)
3240 FIXME( "text type %u not supported\n", text
->text
->textType
);
3243 *ret
= (WS_XML_UTF8_TEXT
*)text
->text
;
3247 static HRESULT
read_get_attribute_text( struct reader
*reader
, ULONG index
, WS_XML_UTF8_TEXT
**ret
)
3249 WS_XML_ELEMENT_NODE
*elem
= &reader
->current
->hdr
;
3250 WS_XML_ATTRIBUTE
*attr
;
3252 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_ELEMENT
)
3253 return WS_E_INVALID_FORMAT
;
3255 attr
= elem
->attributes
[index
];
3256 if (attr
->value
->textType
!= WS_XML_TEXT_TYPE_UTF8
)
3258 FIXME( "text type %u not supported\n", attr
->value
->textType
);
3261 *ret
= (WS_XML_UTF8_TEXT
*)attr
->value
;
3265 static BOOL
find_attribute( struct reader
*reader
, const WS_XML_STRING
*localname
,
3266 const WS_XML_STRING
*ns
, ULONG
*index
)
3269 WS_XML_ELEMENT_NODE
*elem
= &reader
->current
->hdr
;
3273 *index
= reader
->current_attr
;
3276 for (i
= 0; i
< elem
->attributeCount
; i
++)
3278 const WS_XML_STRING
*localname2
= elem
->attributes
[i
]->localName
;
3279 const WS_XML_STRING
*ns2
= elem
->attributes
[i
]->ns
;
3281 if (!cmp_name( localname
->bytes
, localname
->length
, localname2
->bytes
, localname2
->length
) &&
3282 !cmp_name( ns
->bytes
, ns
->length
, ns2
->bytes
, ns2
->length
))
3291 /**************************************************************************
3292 * WsFindAttribute [webservices.@]
3294 HRESULT WINAPI
WsFindAttribute( WS_XML_READER
*handle
, const WS_XML_STRING
*localname
,
3295 const WS_XML_STRING
*ns
, BOOL required
, ULONG
*index
,
3298 struct reader
*reader
= (struct reader
*)handle
;
3301 TRACE( "%p %s %s %d %p %p\n", handle
, debugstr_xmlstr(localname
), debugstr_xmlstr(ns
),
3302 required
, index
, error
);
3303 if (error
) FIXME( "ignoring error parameter\n" );
3305 if (!reader
|| !localname
|| !ns
|| !index
) return E_INVALIDARG
;
3307 EnterCriticalSection( &reader
->cs
);
3309 if (reader
->magic
!= READER_MAGIC
)
3311 LeaveCriticalSection( &reader
->cs
);
3312 return E_INVALIDARG
;
3315 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_ELEMENT
)
3317 LeaveCriticalSection( &reader
->cs
);
3318 return WS_E_INVALID_OPERATION
;
3321 if (!find_attribute( reader
, localname
, ns
, index
))
3323 if (required
) hr
= WS_E_INVALID_FORMAT
;
3331 LeaveCriticalSection( &reader
->cs
);
3335 static HRESULT
read_get_text( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3336 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3337 WS_XML_UTF8_TEXT
**ret
, BOOL
*found
)
3341 case WS_ATTRIBUTE_TYPE_MAPPING
:
3344 if (!(*found
= find_attribute( reader
, localname
, ns
, &index
))) return S_OK
;
3345 return read_get_attribute_text( reader
, index
, ret
);
3347 case WS_ELEMENT_TYPE_MAPPING
:
3348 case WS_ELEMENT_CONTENT_TYPE_MAPPING
:
3349 case WS_ANY_ELEMENT_TYPE_MAPPING
:
3355 const WS_XML_ELEMENT_NODE
*elem
= &reader
->current
->hdr
;
3357 if (WsXmlStringEquals( localname
, elem
->localName
, NULL
) != S_OK
||
3358 WsXmlStringEquals( ns
, elem
->ns
, NULL
) != S_OK
)
3363 if ((hr
= read_startelement( reader
)) != S_OK
) return hr
;
3365 return read_get_node_text( reader
, ret
);
3368 FIXME( "mapping %u not supported\n", mapping
);
3373 static HRESULT
read_type_bool( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3374 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3375 const WS_BOOL_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3376 WS_HEAP
*heap
, void *ret
, ULONG size
)
3378 WS_XML_UTF8_TEXT
*utf8
;
3380 BOOL found
, val
= FALSE
;
3384 FIXME( "description not supported\n" );
3387 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3390 ULONG len
= utf8
->value
.length
;
3391 if (len
== 4 && !memcmp( utf8
->value
.bytes
, "true", 4 )) val
= TRUE
;
3392 else if (len
== 1 && !memcmp( utf8
->value
.bytes
, "1", 1 )) val
= TRUE
;
3393 else if (len
== 5 && !memcmp( utf8
->value
.bytes
, "false", 5 )) val
= FALSE
;
3394 else if (len
== 1 && !memcmp( utf8
->value
.bytes
, "0", 1 )) val
= FALSE
;
3395 else return WS_E_INVALID_FORMAT
;
3400 case WS_READ_REQUIRED_VALUE
:
3401 if (!found
) return WS_E_INVALID_FORMAT
;
3404 case WS_READ_NILLABLE_VALUE
:
3405 if (size
!= sizeof(BOOL
)) return E_INVALIDARG
;
3409 case WS_READ_REQUIRED_POINTER
:
3410 if (!found
) return WS_E_INVALID_FORMAT
;
3413 case WS_READ_OPTIONAL_POINTER
:
3414 case WS_READ_NILLABLE_POINTER
:
3416 BOOL
*heap_val
= NULL
;
3417 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3420 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3423 *(BOOL
**)ret
= heap_val
;
3427 FIXME( "read option %u not supported\n", option
);
3434 static HRESULT
read_type_int8( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3435 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3436 const WS_INT8_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3437 WS_HEAP
*heap
, void *ret
, ULONG size
)
3439 WS_XML_UTF8_TEXT
*utf8
;
3446 FIXME( "description not supported\n" );
3449 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3450 if (found
&& (hr
= str_to_int64( utf8
->value
.bytes
, utf8
->value
.length
, MIN_INT8
, MAX_INT8
, &val
)) != S_OK
)
3455 case WS_READ_REQUIRED_VALUE
:
3456 if (!found
) return WS_E_INVALID_FORMAT
;
3459 case WS_READ_NILLABLE_VALUE
:
3460 if (size
!= sizeof(INT8
)) return E_INVALIDARG
;
3464 case WS_READ_REQUIRED_POINTER
:
3465 if (!found
) return WS_E_INVALID_FORMAT
;
3468 case WS_READ_OPTIONAL_POINTER
:
3469 case WS_READ_NILLABLE_POINTER
:
3471 INT8
*heap_val
= NULL
;
3472 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3475 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3478 *(INT8
**)ret
= heap_val
;
3482 FIXME( "read option %u not supported\n", option
);
3489 static HRESULT
read_type_int16( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3490 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3491 const WS_INT16_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3492 WS_HEAP
*heap
, void *ret
, ULONG size
)
3494 WS_XML_UTF8_TEXT
*utf8
;
3501 FIXME( "description not supported\n" );
3504 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3505 if (found
&& (hr
= str_to_int64( utf8
->value
.bytes
, utf8
->value
.length
, MIN_INT16
, MAX_INT16
, &val
)) != S_OK
)
3510 case WS_READ_REQUIRED_VALUE
:
3511 if (!found
) return WS_E_INVALID_FORMAT
;
3514 case WS_READ_NILLABLE_VALUE
:
3515 if (size
!= sizeof(INT16
)) return E_INVALIDARG
;
3516 *(INT16
*)ret
= val
;
3519 case WS_READ_REQUIRED_POINTER
:
3520 if (!found
) return WS_E_INVALID_FORMAT
;
3523 case WS_READ_OPTIONAL_POINTER
:
3524 case WS_READ_NILLABLE_POINTER
:
3526 INT16
*heap_val
= NULL
;
3527 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3530 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3533 *(INT16
**)ret
= heap_val
;
3537 FIXME( "read option %u not supported\n", option
);
3544 static HRESULT
read_type_int32( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3545 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3546 const WS_INT32_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3547 WS_HEAP
*heap
, void *ret
, ULONG size
)
3549 WS_XML_UTF8_TEXT
*utf8
;
3556 FIXME( "description not supported\n" );
3559 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3560 if (found
&& (hr
= str_to_int64( utf8
->value
.bytes
, utf8
->value
.length
, MIN_INT32
, MAX_INT32
, &val
)) != S_OK
)
3565 case WS_READ_REQUIRED_VALUE
:
3566 if (!found
) return WS_E_INVALID_FORMAT
;
3569 case WS_READ_NILLABLE_VALUE
:
3570 if (size
!= sizeof(INT32
)) return E_INVALIDARG
;
3571 *(INT32
*)ret
= val
;
3574 case WS_READ_REQUIRED_POINTER
:
3575 if (!found
) return WS_E_INVALID_FORMAT
;
3578 case WS_READ_OPTIONAL_POINTER
:
3579 case WS_READ_NILLABLE_POINTER
:
3581 INT32
*heap_val
= NULL
;
3582 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3585 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3588 *(INT32
**)ret
= heap_val
;
3592 FIXME( "read option %u not supported\n", option
);
3599 static HRESULT
read_type_int64( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3600 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3601 const WS_INT64_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3602 WS_HEAP
*heap
, void *ret
, ULONG size
)
3604 WS_XML_UTF8_TEXT
*utf8
;
3611 FIXME( "description not supported\n" );
3614 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3615 if (found
&& (hr
= str_to_int64( utf8
->value
.bytes
, utf8
->value
.length
, MIN_INT64
, MAX_INT64
, &val
)) != S_OK
)
3620 case WS_READ_REQUIRED_VALUE
:
3621 if (!found
) return WS_E_INVALID_FORMAT
;
3624 case WS_READ_NILLABLE_VALUE
:
3625 if (size
!= sizeof(INT64
)) return E_INVALIDARG
;
3626 *(INT64
*)ret
= val
;
3629 case WS_READ_REQUIRED_POINTER
:
3630 if (!found
) return WS_E_INVALID_FORMAT
;
3633 case WS_READ_OPTIONAL_POINTER
:
3634 case WS_READ_NILLABLE_POINTER
:
3636 INT64
*heap_val
= NULL
;
3637 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3640 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3643 *(INT64
**)ret
= heap_val
;
3647 FIXME( "read option %u not supported\n", option
);
3654 static HRESULT
read_type_uint8( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3655 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3656 const WS_UINT8_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3657 WS_HEAP
*heap
, void *ret
, ULONG size
)
3659 WS_XML_UTF8_TEXT
*utf8
;
3666 FIXME( "description not supported\n" );
3669 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3670 if (found
&& (hr
= str_to_uint64( utf8
->value
.bytes
, utf8
->value
.length
, MAX_UINT8
, &val
)) != S_OK
)
3675 case WS_READ_REQUIRED_VALUE
:
3676 if (!found
) return WS_E_INVALID_FORMAT
;
3679 case WS_READ_NILLABLE_VALUE
:
3680 if (size
!= sizeof(UINT8
)) return E_INVALIDARG
;
3681 *(UINT8
*)ret
= val
;
3684 case WS_READ_REQUIRED_POINTER
:
3685 if (!found
) return WS_E_INVALID_FORMAT
;
3688 case WS_READ_OPTIONAL_POINTER
:
3689 case WS_READ_NILLABLE_POINTER
:
3691 UINT8
*heap_val
= NULL
;
3692 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3695 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3698 *(UINT8
**)ret
= heap_val
;
3702 FIXME( "read option %u not supported\n", option
);
3709 static HRESULT
read_type_uint16( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3710 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3711 const WS_UINT16_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3712 WS_HEAP
*heap
, void *ret
, ULONG size
)
3714 WS_XML_UTF8_TEXT
*utf8
;
3721 FIXME( "description not supported\n" );
3724 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3725 if (found
&& (hr
= str_to_uint64( utf8
->value
.bytes
, utf8
->value
.length
, MAX_UINT16
, &val
)) != S_OK
)
3730 case WS_READ_REQUIRED_VALUE
:
3731 if (!found
) return WS_E_INVALID_FORMAT
;
3734 case WS_READ_NILLABLE_VALUE
:
3735 if (size
!= sizeof(UINT16
)) return E_INVALIDARG
;
3736 *(UINT16
*)ret
= val
;
3739 case WS_READ_REQUIRED_POINTER
:
3740 if (!found
) return WS_E_INVALID_FORMAT
;
3743 case WS_READ_OPTIONAL_POINTER
:
3744 case WS_READ_NILLABLE_POINTER
:
3746 UINT16
*heap_val
= NULL
;
3747 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3750 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3753 *(UINT16
**)ret
= heap_val
;
3757 FIXME( "read option %u not supported\n", option
);
3764 static HRESULT
read_type_uint32( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3765 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3766 const WS_UINT32_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3767 WS_HEAP
*heap
, void *ret
, ULONG size
)
3769 WS_XML_UTF8_TEXT
*utf8
;
3776 FIXME( "description not supported\n" );
3779 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3780 if (found
&& (hr
= str_to_uint64( utf8
->value
.bytes
, utf8
->value
.length
, MAX_UINT32
, &val
)) != S_OK
)
3785 case WS_READ_REQUIRED_VALUE
:
3786 if (!found
) return WS_E_INVALID_FORMAT
;
3789 case WS_READ_NILLABLE_VALUE
:
3790 if (size
!= sizeof(UINT32
)) return E_INVALIDARG
;
3791 *(UINT32
*)ret
= val
;
3794 case WS_READ_REQUIRED_POINTER
:
3795 if (!found
) return WS_E_INVALID_FORMAT
;
3798 case WS_READ_OPTIONAL_POINTER
:
3799 case WS_READ_NILLABLE_POINTER
:
3801 UINT32
*heap_val
= NULL
;
3802 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3805 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3808 *(UINT32
**)ret
= heap_val
;
3812 FIXME( "read option %u not supported\n", option
);
3819 static HRESULT
read_type_uint64( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3820 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3821 const WS_UINT64_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3822 WS_HEAP
*heap
, void *ret
, ULONG size
)
3824 WS_XML_UTF8_TEXT
*utf8
;
3831 FIXME( "description not supported\n" );
3834 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3835 if (found
&& (hr
= str_to_uint64( utf8
->value
.bytes
, utf8
->value
.length
, MAX_UINT64
, &val
)) != S_OK
)
3840 case WS_READ_REQUIRED_VALUE
:
3841 if (!found
) return WS_E_INVALID_FORMAT
;
3844 case WS_READ_NILLABLE_VALUE
:
3845 if (size
!= sizeof(UINT64
)) return E_INVALIDARG
;
3846 *(UINT64
*)ret
= val
;
3849 case WS_READ_REQUIRED_POINTER
:
3850 if (!found
) return WS_E_INVALID_FORMAT
;
3853 case WS_READ_OPTIONAL_POINTER
:
3854 case WS_READ_NILLABLE_POINTER
:
3856 UINT64
*heap_val
= NULL
;
3857 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3860 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3863 *(UINT64
**)ret
= heap_val
;
3867 FIXME( "read option %u not supported\n", option
);
3874 static HRESULT
read_type_double( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3875 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3876 const WS_DOUBLE_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3877 WS_HEAP
*heap
, void *ret
, ULONG size
)
3879 WS_XML_UTF8_TEXT
*utf8
;
3884 if (desc
) FIXME( "ignoring description\n" );
3886 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3887 if (found
&& (hr
= str_to_double( utf8
->value
.bytes
, utf8
->value
.length
, &val
)) != S_OK
) return hr
;
3891 case WS_READ_REQUIRED_VALUE
:
3892 if (!found
) return WS_E_INVALID_FORMAT
;
3895 case WS_READ_NILLABLE_VALUE
:
3896 if (size
!= sizeof(double)) return E_INVALIDARG
;
3897 *(double *)ret
= val
;
3900 case WS_READ_REQUIRED_POINTER
:
3901 if (!found
) return WS_E_INVALID_FORMAT
;
3904 case WS_READ_OPTIONAL_POINTER
:
3905 case WS_READ_NILLABLE_POINTER
:
3907 double *heap_val
= NULL
;
3908 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
3911 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
3914 *(double **)ret
= heap_val
;
3918 FIXME( "read option %u not supported\n", option
);
3925 static HRESULT
read_type_wsz( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3926 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3927 const WS_WSZ_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3928 WS_HEAP
*heap
, WCHAR
**ret
, ULONG size
)
3930 WS_XML_UTF8_TEXT
*utf8
;
3937 FIXME( "description not supported\n" );
3940 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3941 if (found
&& !(str
= xmltext_to_widechar( heap
, &utf8
->text
))) return WS_E_QUOTA_EXCEEDED
;
3945 case WS_READ_REQUIRED_POINTER
:
3946 if (!found
) return WS_E_INVALID_FORMAT
;
3949 case WS_READ_OPTIONAL_POINTER
:
3950 case WS_READ_NILLABLE_POINTER
:
3951 if (size
!= sizeof(str
)) return E_INVALIDARG
;
3956 FIXME( "read option %u not supported\n", option
);
3963 static HRESULT
get_enum_value( const WS_XML_UTF8_TEXT
*text
, const WS_ENUM_DESCRIPTION
*desc
, int *ret
)
3966 for (i
= 0; i
< desc
->valueCount
; i
++)
3968 if (WsXmlStringEquals( &text
->value
, desc
->values
[i
].name
, NULL
) == S_OK
)
3970 *ret
= desc
->values
[i
].value
;
3974 return WS_E_INVALID_FORMAT
;
3977 static HRESULT
read_type_enum( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
3978 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
3979 const WS_ENUM_DESCRIPTION
*desc
, WS_READ_OPTION option
,
3980 WS_HEAP
*heap
, void *ret
, ULONG size
)
3982 WS_XML_UTF8_TEXT
*utf8
;
3987 if (!desc
) return E_INVALIDARG
;
3989 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
3990 if (found
&& (hr
= get_enum_value( utf8
, desc
, &val
)) != S_OK
) return hr
;
3994 case WS_READ_REQUIRED_VALUE
:
3995 if (!found
) return WS_E_INVALID_FORMAT
;
3998 case WS_READ_NILLABLE_VALUE
:
3999 if (size
!= sizeof(int)) return E_INVALIDARG
;
4003 case WS_READ_REQUIRED_POINTER
:
4004 if (!found
) return WS_E_INVALID_FORMAT
;
4007 case WS_READ_OPTIONAL_POINTER
:
4008 case WS_READ_NILLABLE_POINTER
:
4010 int *heap_val
= NULL
;
4011 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
4014 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
4017 *(int **)ret
= heap_val
;
4021 FIXME( "read option %u not supported\n", option
);
4028 static HRESULT
read_type_datetime( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
4029 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4030 const WS_DATETIME_DESCRIPTION
*desc
, WS_READ_OPTION option
,
4031 WS_HEAP
*heap
, void *ret
, ULONG size
)
4033 WS_XML_UTF8_TEXT
*utf8
;
4035 WS_DATETIME val
= {0, WS_DATETIME_FORMAT_UTC
};
4038 if (desc
) FIXME( "ignoring description\n" );
4040 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
4041 if (found
&& (hr
= str_to_datetime( utf8
->value
.bytes
, utf8
->value
.length
, &val
)) != S_OK
) return hr
;
4045 case WS_READ_REQUIRED_VALUE
:
4046 if (!found
) return WS_E_INVALID_FORMAT
;
4049 case WS_READ_NILLABLE_VALUE
:
4050 if (size
!= sizeof(WS_DATETIME
)) return E_INVALIDARG
;
4051 *(WS_DATETIME
*)ret
= val
;
4054 case WS_READ_REQUIRED_POINTER
:
4055 if (!found
) return WS_E_INVALID_FORMAT
;
4058 case WS_READ_OPTIONAL_POINTER
:
4059 case WS_READ_NILLABLE_POINTER
:
4061 WS_DATETIME
*heap_val
= NULL
;
4062 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
4065 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
4068 *(WS_DATETIME
**)ret
= heap_val
;
4072 FIXME( "read option %u not supported\n", option
);
4079 static HRESULT
read_type_guid( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
4080 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4081 const WS_GUID_DESCRIPTION
*desc
, WS_READ_OPTION option
,
4082 WS_HEAP
*heap
, void *ret
, ULONG size
)
4084 WS_XML_UTF8_TEXT
*utf8
;
4089 if (desc
) FIXME( "ignoring description\n" );
4091 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
4092 if (found
&& (hr
= str_to_guid( utf8
->value
.bytes
, utf8
->value
.length
, &val
)) != S_OK
) return hr
;
4096 case WS_READ_REQUIRED_VALUE
:
4097 if (!found
) return WS_E_INVALID_FORMAT
;
4100 case WS_READ_NILLABLE_VALUE
:
4101 if (size
!= sizeof(GUID
)) return E_INVALIDARG
;
4105 case WS_READ_REQUIRED_POINTER
:
4106 if (!found
) return WS_E_INVALID_FORMAT
;
4109 case WS_READ_OPTIONAL_POINTER
:
4110 case WS_READ_NILLABLE_POINTER
:
4112 GUID
*heap_val
= NULL
;
4113 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
4116 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
4119 *(GUID
**)ret
= heap_val
;
4123 FIXME( "read option %u not supported\n", option
);
4130 static HRESULT
read_type_bytes( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
4131 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4132 const WS_BYTES_DESCRIPTION
*desc
, WS_READ_OPTION option
,
4133 WS_HEAP
*heap
, void *ret
, ULONG size
)
4135 WS_XML_UTF8_TEXT
*utf8
;
4140 if (desc
) FIXME( "ignoring description\n" );
4142 if ((hr
= read_get_text( reader
, mapping
, localname
, ns
, &utf8
, &found
)) != S_OK
) return hr
;
4143 if (found
&& (hr
= str_to_bytes( utf8
->value
.bytes
, utf8
->value
.length
, heap
, &val
)) != S_OK
)
4148 case WS_READ_REQUIRED_VALUE
:
4149 if (!found
) return WS_E_INVALID_FORMAT
;
4152 case WS_READ_NILLABLE_VALUE
:
4153 if (size
!= sizeof(WS_BYTES
)) return E_INVALIDARG
;
4154 *(WS_BYTES
*)ret
= val
;
4157 case WS_READ_REQUIRED_POINTER
:
4158 if (!found
) return WS_E_INVALID_FORMAT
;
4161 case WS_READ_OPTIONAL_POINTER
:
4162 case WS_READ_NILLABLE_POINTER
:
4164 WS_BYTES
*heap_val
= NULL
;
4165 if (size
!= sizeof(heap_val
)) return E_INVALIDARG
;
4168 if (!(heap_val
= ws_alloc( heap
, sizeof(*heap_val
) ))) return WS_E_QUOTA_EXCEEDED
;
4171 *(WS_BYTES
**)ret
= heap_val
;
4175 FIXME( "read option %u not supported\n", option
);
4182 static BOOL
is_empty_text_node( const struct node
*node
)
4184 const WS_XML_TEXT_NODE
*text
= (const WS_XML_TEXT_NODE
*)node
;
4185 const WS_XML_UTF8_TEXT
*utf8
;
4188 if (node_type( node
) != WS_XML_NODE_TYPE_TEXT
) return FALSE
;
4189 if (text
->text
->textType
!= WS_XML_TEXT_TYPE_UTF8
)
4191 ERR( "unhandled text type %u\n", text
->text
->textType
);
4194 utf8
= (const WS_XML_UTF8_TEXT
*)text
->text
;
4195 for (i
= 0; i
< utf8
->value
.length
; i
++) if (!read_isspace( utf8
->value
.bytes
[i
] )) return FALSE
;
4199 static HRESULT
read_next_node( struct reader
*reader
)
4201 if (reader
->current
== reader
->last
) return read_node( reader
);
4202 if (move_to_child_node( &reader
->current
)) return S_OK
;
4203 if (move_to_next_node( &reader
->current
)) return S_OK
;
4204 if (!move_to_parent_node( &reader
->current
)) return WS_E_INVALID_FORMAT
;
4205 if (move_to_next_node( &reader
->current
)) return S_OK
;
4206 return WS_E_INVALID_FORMAT
;
4209 /* skips comment and empty text nodes */
4210 static HRESULT
read_type_next_node( struct reader
*reader
)
4215 WS_XML_NODE_TYPE type
;
4217 if ((hr
= read_next_node( reader
)) != S_OK
) return hr
;
4218 type
= node_type( reader
->current
);
4219 if (type
== WS_XML_NODE_TYPE_COMMENT
||
4220 (type
== WS_XML_NODE_TYPE_TEXT
&& is_empty_text_node( reader
->current
))) continue;
4225 static BOOL
match_current_element( struct reader
*reader
, const WS_XML_STRING
*localname
,
4226 const WS_XML_STRING
*ns
)
4228 const WS_XML_ELEMENT_NODE
*elem
= &reader
->current
->hdr
;
4229 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_ELEMENT
) return FALSE
;
4230 return WsXmlStringEquals( localname
, elem
->localName
, NULL
) == S_OK
&&
4231 WsXmlStringEquals( ns
, elem
->ns
, NULL
) == S_OK
;
4234 static HRESULT
read_type_next_element_node( struct reader
*reader
, const WS_XML_STRING
*localname
,
4235 const WS_XML_STRING
*ns
)
4241 if (!localname
) return S_OK
; /* assume reader is already correctly positioned */
4242 if (reader
->current
== reader
->last
)
4245 if ((hr
= read_to_startelement( reader
, &found
)) != S_OK
) return hr
;
4246 if (!found
) return WS_E_INVALID_FORMAT
;
4248 if (match_current_element( reader
, localname
, ns
)) return S_OK
;
4250 node
= reader
->current
;
4251 attr
= reader
->current_attr
;
4253 if ((hr
= read_type_next_node( reader
)) != S_OK
) return hr
;
4254 if (match_current_element( reader
, localname
, ns
)) return S_OK
;
4256 reader
->current
= node
;
4257 reader
->current_attr
= attr
;
4259 return WS_E_INVALID_FORMAT
;
4262 ULONG
get_type_size( WS_TYPE type
, const WS_STRUCT_DESCRIPTION
*desc
)
4268 return sizeof(INT8
);
4271 case WS_UINT16_TYPE
:
4272 return sizeof(INT16
);
4276 case WS_UINT32_TYPE
:
4278 return sizeof(INT32
);
4281 case WS_UINT64_TYPE
:
4282 return sizeof(INT64
);
4284 case WS_DOUBLE_TYPE
:
4285 return sizeof(double);
4287 case WS_DATETIME_TYPE
:
4288 return sizeof(WS_DATETIME
);
4291 return sizeof(GUID
);
4293 case WS_STRING_TYPE
:
4294 return sizeof(WS_STRING
);
4297 return sizeof(WCHAR
*);
4300 return sizeof(WS_BYTES
);
4302 case WS_XML_STRING_TYPE
:
4303 return sizeof(WS_XML_STRING
);
4305 case WS_STRUCT_TYPE
:
4308 case WS_DESCRIPTION_TYPE
:
4309 return sizeof(WS_STRUCT_DESCRIPTION
*);
4312 ERR( "unhandled type %u\n", type
);
4317 static WS_READ_OPTION
get_field_read_option( WS_TYPE type
, ULONG options
)
4319 if (options
& WS_FIELD_POINTER
)
4321 if (options
& WS_FIELD_NILLABLE
) return WS_READ_NILLABLE_POINTER
;
4322 if (options
& WS_FIELD_OPTIONAL
) return WS_READ_OPTIONAL_POINTER
;
4323 return WS_READ_REQUIRED_POINTER
;
4334 case WS_UINT16_TYPE
:
4335 case WS_UINT32_TYPE
:
4336 case WS_UINT64_TYPE
:
4337 case WS_DOUBLE_TYPE
:
4338 case WS_DATETIME_TYPE
:
4340 case WS_STRING_TYPE
:
4342 case WS_XML_STRING_TYPE
:
4343 case WS_STRUCT_TYPE
:
4345 if (options
& (WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
)) return WS_READ_NILLABLE_VALUE
;
4346 return WS_READ_REQUIRED_VALUE
;
4349 case WS_DESCRIPTION_TYPE
:
4350 if (options
& WS_FIELD_NILLABLE
) return WS_READ_NILLABLE_POINTER
;
4351 if (options
& WS_FIELD_OPTIONAL
) return WS_READ_OPTIONAL_POINTER
;
4352 return WS_READ_REQUIRED_POINTER
;
4355 FIXME( "unhandled type %u\n", type
);
4360 static HRESULT
read_type( struct reader
*, WS_TYPE_MAPPING
, WS_TYPE
, const WS_XML_STRING
*,
4361 const WS_XML_STRING
*, const void *, WS_READ_OPTION
, WS_HEAP
*,
4364 static HRESULT
read_type_repeating_element( struct reader
*reader
, const WS_FIELD_DESCRIPTION
*desc
,
4365 WS_HEAP
*heap
, void **ret
, ULONG
*count
)
4368 ULONG item_size
, nb_items
= 0, nb_allocated
= 1, offset
= 0;
4369 WS_READ_OPTION option
;
4372 if (!(option
= get_field_read_option( desc
->type
, desc
->options
))) return E_INVALIDARG
;
4374 /* wrapper element */
4375 if (desc
->localName
&& ((hr
= read_type_next_element_node( reader
, desc
->localName
, desc
->ns
)) != S_OK
))
4378 if (option
== WS_READ_REQUIRED_VALUE
|| option
== WS_READ_NILLABLE_VALUE
)
4379 item_size
= get_type_size( desc
->type
, desc
->typeDescription
);
4381 item_size
= sizeof(void *);
4383 if (!(buf
= ws_alloc_zero( heap
, item_size
))) return WS_E_QUOTA_EXCEEDED
;
4386 if (nb_items
>= nb_allocated
)
4388 SIZE_T old_size
= nb_allocated
* item_size
, new_size
= old_size
* 2;
4389 if (!(buf
= ws_realloc_zero( heap
, buf
, old_size
, new_size
)))
4390 return WS_E_QUOTA_EXCEEDED
;
4393 hr
= read_type( reader
, WS_ELEMENT_TYPE_MAPPING
, desc
->type
, desc
->itemLocalName
, desc
->itemNs
,
4394 desc
->typeDescription
, option
, heap
, buf
+ offset
, item_size
);
4395 if (hr
== WS_E_INVALID_FORMAT
) break;
4398 ws_free( heap
, buf
, nb_allocated
* item_size
);
4401 offset
+= item_size
;
4405 if (desc
->localName
&& ((hr
= read_type_next_node( reader
)) != S_OK
)) return hr
;
4407 if (desc
->itemRange
&& (nb_items
< desc
->itemRange
->minItemCount
|| nb_items
> desc
->itemRange
->maxItemCount
))
4409 TRACE( "number of items %u out of range (%u-%u)\n", nb_items
, desc
->itemRange
->minItemCount
,
4410 desc
->itemRange
->maxItemCount
);
4411 ws_free( heap
, buf
, nb_allocated
* item_size
);
4412 return WS_E_INVALID_FORMAT
;
4421 static HRESULT
read_type_text( struct reader
*reader
, const WS_FIELD_DESCRIPTION
*desc
,
4422 WS_READ_OPTION option
, WS_HEAP
*heap
, void *ret
, ULONG size
)
4425 if (reader
->current
== reader
->last
)
4428 if ((hr
= read_to_startelement( reader
, &found
)) != S_OK
) return S_OK
;
4429 if (!found
) return WS_E_INVALID_FORMAT
;
4431 if ((hr
= read_next_node( reader
)) != S_OK
) return hr
;
4432 if (node_type( reader
->current
) != WS_XML_NODE_TYPE_TEXT
) return WS_E_INVALID_FORMAT
;
4434 return read_type( reader
, WS_ANY_ELEMENT_TYPE_MAPPING
, desc
->type
, NULL
, NULL
,
4435 desc
->typeDescription
, option
, heap
, ret
, size
);
4438 static HRESULT
read_type_struct_field( struct reader
*reader
, const WS_FIELD_DESCRIPTION
*desc
,
4439 WS_HEAP
*heap
, char *buf
, ULONG offset
)
4442 WS_READ_OPTION option
;
4446 if (!desc
) return E_INVALIDARG
;
4447 if (desc
->options
& ~(WS_FIELD_POINTER
|WS_FIELD_OPTIONAL
|WS_FIELD_NILLABLE
|WS_FIELD_NILLABLE_ITEM
))
4449 FIXME( "options %08x not supported\n", desc
->options
);
4452 if (!(option
= get_field_read_option( desc
->type
, desc
->options
))) return E_INVALIDARG
;
4454 if (option
== WS_READ_REQUIRED_VALUE
|| option
== WS_READ_NILLABLE_VALUE
)
4455 size
= get_type_size( desc
->type
, desc
->typeDescription
);
4457 size
= sizeof(void *);
4460 switch (desc
->mapping
)
4462 case WS_TYPE_ATTRIBUTE_FIELD_MAPPING
:
4463 FIXME( "WS_TYPE_ATTRIBUTE_FIELD_MAPPING not supported\n" );
4466 case WS_ATTRIBUTE_FIELD_MAPPING
:
4467 hr
= read_type( reader
, WS_ATTRIBUTE_TYPE_MAPPING
, desc
->type
, desc
->localName
, desc
->ns
,
4468 desc
->typeDescription
, option
, heap
, ptr
, size
);
4471 case WS_ELEMENT_FIELD_MAPPING
:
4472 hr
= read_type( reader
, WS_ELEMENT_TYPE_MAPPING
, desc
->type
, desc
->localName
, desc
->ns
,
4473 desc
->typeDescription
, option
, heap
, ptr
, size
);
4476 case WS_REPEATING_ELEMENT_FIELD_MAPPING
:
4479 hr
= read_type_repeating_element( reader
, desc
, heap
, (void **)ptr
, &count
);
4480 if (hr
== S_OK
) *(ULONG
*)(buf
+ desc
->countOffset
) = count
;
4483 case WS_TEXT_FIELD_MAPPING
:
4484 hr
= read_type_text( reader
, desc
, option
, heap
, ptr
, size
);
4488 FIXME( "unhandled field mapping %u\n", desc
->mapping
);
4492 if (hr
== WS_E_INVALID_FORMAT
)
4496 case WS_READ_REQUIRED_VALUE
:
4497 case WS_READ_REQUIRED_POINTER
:
4498 return WS_E_INVALID_FORMAT
;
4500 case WS_READ_NILLABLE_VALUE
:
4501 if (desc
->defaultValue
) memcpy( ptr
, desc
->defaultValue
->value
, desc
->defaultValue
->valueSize
);
4504 case WS_READ_OPTIONAL_POINTER
:
4505 case WS_READ_NILLABLE_POINTER
:
4506 *(void **)ptr
= NULL
;
4510 ERR( "unhandled option %u\n", option
);
4518 static HRESULT
read_type_struct( struct reader
*reader
, WS_TYPE_MAPPING mapping
,
4519 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
,
4520 const WS_STRUCT_DESCRIPTION
*desc
, WS_READ_OPTION option
,
4521 WS_HEAP
*heap
, void *ret
, ULONG size
)
4527 if (!desc
) return E_INVALIDARG
;
4528 if (desc
->structOptions
& ~WS_STRUCT_IGNORE_TRAILING_ELEMENT_CONTENT
)
4530 FIXME( "struct options %08x not supported\n",
4531 desc
->structOptions
& ~WS_STRUCT_IGNORE_TRAILING_ELEMENT_CONTENT
);
4536 case WS_READ_REQUIRED_POINTER
:
4537 case WS_READ_OPTIONAL_POINTER
:
4538 case WS_READ_NILLABLE_POINTER
:
4539 if (size
!= sizeof(void *)) return E_INVALIDARG
;
4540 if (!(buf
= ws_alloc_zero( heap
, desc
->size
))) return WS_E_QUOTA_EXCEEDED
;
4543 case WS_READ_REQUIRED_VALUE
:
4544 case WS_READ_NILLABLE_VALUE
:
4545 if (size
!= desc
->size
) return E_INVALIDARG
;
4550 FIXME( "unhandled read option %u\n", option
);
4554 for (i
= 0; i
< desc
->fieldCount
; i
++)
4556 offset
= desc
->fields
[i
]->offset
;
4557 if ((hr
= read_type_struct_field( reader
, desc
->fields
[i
], heap
, buf
, offset
)) != S_OK
)
4563 case WS_READ_REQUIRED_POINTER
:
4566 ws_free( heap
, buf
, desc
->size
);
4569 *(char **)ret
= buf
;
4572 case WS_READ_OPTIONAL_POINTER
:
4573 case WS_READ_NILLABLE_POINTER
:
4574 if (is_nil_value( buf
, desc
->size
))
4576 ws_free( heap
, buf
, desc
->size
);
4579 *(char **)ret
= buf
;
4582 case WS_READ_REQUIRED_VALUE
:
4583 case WS_READ_NILLABLE_VALUE
:
4584 if (hr
!= S_OK
) return hr
;
4588 ERR( "unhandled read option %u\n", option
);
4592 if (desc
->structOptions
& WS_STRUCT_IGNORE_TRAILING_ELEMENT_CONTENT
)
4594 struct node
*parent
= find_parent( reader
);
4595 parent
->flags
|= NODE_FLAG_IGNORE_TRAILING_ELEMENT_CONTENT
;
4600 static HRESULT
start_mapping( struct reader
*reader
, WS_TYPE_MAPPING mapping
, const WS_XML_STRING
*localname
,
4601 const WS_XML_STRING
*ns
)
4605 case WS_ELEMENT_TYPE_MAPPING
:
4606 case WS_ELEMENT_CONTENT_TYPE_MAPPING
:
4607 return read_type_next_element_node( reader
, localname
, ns
);
4609 case WS_ANY_ELEMENT_TYPE_MAPPING
:
4610 case WS_ATTRIBUTE_TYPE_MAPPING
:
4614 FIXME( "unhandled mapping %u\n", mapping
);
4619 static HRESULT
read_type_endelement_node( struct reader
*reader
)
4621 const struct node
*parent
= find_parent( reader
);
4626 if ((hr
= read_type_next_node( reader
)) != S_OK
) return hr
;
4627 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_END_ELEMENT
&& reader
->current
->parent
== parent
)
4631 if (read_end_of_data( reader
) || !(parent
->flags
& NODE_FLAG_IGNORE_TRAILING_ELEMENT_CONTENT
)) break;
4634 return WS_E_INVALID_FORMAT
;
4637 static HRESULT
end_mapping( struct reader
*reader
, WS_TYPE_MAPPING mapping
)
4641 case WS_ELEMENT_TYPE_MAPPING
:
4642 return read_type_endelement_node( reader
);
4644 case WS_ELEMENT_CONTENT_TYPE_MAPPING
:
4645 return read_type_next_node( reader
);
4647 case WS_ATTRIBUTE_TYPE_MAPPING
:
4653 static HRESULT
is_nil_element( const WS_XML_ELEMENT_NODE
*elem
)
4655 static const WS_XML_STRING localname
= {3, (BYTE
*)"nil"};
4656 static const WS_XML_STRING ns
= {41, (BYTE
*)"http://www.w3.org/2001/XMLSchema-instance"};
4659 for (i
= 0; i
< elem
->attributeCount
; i
++)
4661 const WS_XML_UTF8_TEXT
*text
= (WS_XML_UTF8_TEXT
*)elem
->attributes
[i
]->value
;
4663 if (elem
->attributes
[i
]->isXmlNs
) continue;
4664 if (WsXmlStringEquals( elem
->attributes
[i
]->localName
, &localname
, NULL
) == S_OK
&&
4665 WsXmlStringEquals( elem
->attributes
[i
]->ns
, &ns
, NULL
) == S_OK
&&
4666 text
->value
.length
== 4 && !memcmp( text
->value
.bytes
, "true", 4 )) return TRUE
;
4671 static HRESULT
read_type( struct reader
*reader
, WS_TYPE_MAPPING mapping
, WS_TYPE type
,
4672 const WS_XML_STRING
*localname
, const WS_XML_STRING
*ns
, const void *desc
,
4673 WS_READ_OPTION option
, WS_HEAP
*heap
, void *value
, ULONG size
)
4677 if ((hr
= start_mapping( reader
, mapping
, localname
, ns
)) != S_OK
) return hr
;
4679 if (mapping
== WS_ELEMENT_TYPE_MAPPING
&& is_nil_element( &reader
->current
->hdr
))
4681 if (option
!= WS_READ_NILLABLE_POINTER
&& option
!= WS_READ_NILLABLE_VALUE
) return WS_E_INVALID_FORMAT
;
4682 return end_mapping( reader
, mapping
);
4688 if ((hr
= read_type_bool( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4693 if ((hr
= read_type_int8( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4698 if ((hr
= read_type_int16( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4703 if ((hr
= read_type_int32( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4708 if ((hr
= read_type_int64( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4713 if ((hr
= read_type_uint8( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4717 case WS_UINT16_TYPE
:
4718 if ((hr
= read_type_uint16( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4722 case WS_UINT32_TYPE
:
4723 if ((hr
= read_type_uint32( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4727 case WS_UINT64_TYPE
:
4728 if ((hr
= read_type_uint64( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4732 case WS_DOUBLE_TYPE
:
4733 if ((hr
= read_type_double( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4737 case WS_DATETIME_TYPE
:
4738 if ((hr
= read_type_datetime( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4743 if ((hr
= read_type_guid( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4748 if ((hr
= read_type_wsz( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4753 if ((hr
= read_type_bytes( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4757 case WS_STRUCT_TYPE
:
4758 if ((hr
= read_type_struct( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4763 if ((hr
= read_type_enum( reader
, mapping
, localname
, ns
, desc
, option
, heap
, value
, size
)) != S_OK
)
4768 FIXME( "type %u not supported\n", type
);
4772 return end_mapping( reader
, mapping
);
4775 /**************************************************************************
4776 * WsReadType [webservices.@]
4778 HRESULT WINAPI
WsReadType( WS_XML_READER
*handle
, WS_TYPE_MAPPING mapping
, WS_TYPE type
,
4779 const void *desc
, WS_READ_OPTION option
, WS_HEAP
*heap
, void *value
,
4780 ULONG size
, WS_ERROR
*error
)
4782 struct reader
*reader
= (struct reader
*)handle
;
4785 TRACE( "%p %u %u %p %u %p %p %u %p\n", handle
, mapping
, type
, desc
, option
, heap
, value
,
4787 if (error
) FIXME( "ignoring error parameter\n" );
4789 if (!reader
|| !value
) return E_INVALIDARG
;
4791 EnterCriticalSection( &reader
->cs
);
4793 if (reader
->magic
!= READER_MAGIC
)
4795 LeaveCriticalSection( &reader
->cs
);
4796 return E_INVALIDARG
;
4799 if ((hr
= read_type( reader
, mapping
, type
, NULL
, NULL
, desc
, option
, heap
, value
, size
)) != S_OK
)
4801 LeaveCriticalSection( &reader
->cs
);
4807 case WS_ELEMENT_TYPE_MAPPING
:
4808 hr
= read_node( reader
);
4815 if (hr
== S_OK
&& !read_end_of_data( reader
)) hr
= WS_E_INVALID_FORMAT
;
4817 LeaveCriticalSection( &reader
->cs
);
4821 /**************************************************************************
4822 * WsReadElement [webservices.@]
4824 HRESULT WINAPI
WsReadElement( WS_XML_READER
*handle
, const WS_ELEMENT_DESCRIPTION
*desc
,
4825 WS_READ_OPTION option
, WS_HEAP
*heap
, void *value
, ULONG size
,
4828 struct reader
*reader
= (struct reader
*)handle
;
4831 TRACE( "%p %p %u %p %p %u %p\n", handle
, desc
, option
, heap
, value
, size
, error
);
4832 if (error
) FIXME( "ignoring error parameter\n" );
4834 if (!reader
|| !desc
|| !value
) return E_INVALIDARG
;
4836 EnterCriticalSection( &reader
->cs
);
4838 if (reader
->magic
!= READER_MAGIC
)
4840 LeaveCriticalSection( &reader
->cs
);
4841 return E_INVALIDARG
;
4844 hr
= read_type( reader
, WS_ELEMENT_TYPE_MAPPING
, desc
->type
, desc
->elementLocalName
,
4845 desc
->elementNs
, desc
->typeDescription
, option
, heap
, value
, size
);
4847 LeaveCriticalSection( &reader
->cs
);
4851 /**************************************************************************
4852 * WsReadValue [webservices.@]
4854 HRESULT WINAPI
WsReadValue( WS_XML_READER
*handle
, WS_VALUE_TYPE value_type
, void *value
, ULONG size
,
4857 struct reader
*reader
= (struct reader
*)handle
;
4858 WS_TYPE type
= map_value_type( value_type
);
4861 TRACE( "%p %u %p %u %p\n", handle
, type
, value
, size
, error
);
4862 if (error
) FIXME( "ignoring error parameter\n" );
4864 if (!reader
|| !value
|| type
== ~0u) return E_INVALIDARG
;
4866 EnterCriticalSection( &reader
->cs
);
4868 if (reader
->magic
!= READER_MAGIC
)
4870 LeaveCriticalSection( &reader
->cs
);
4871 return E_INVALIDARG
;
4874 hr
= read_type( reader
, WS_ELEMENT_TYPE_MAPPING
, type
, NULL
, NULL
, NULL
, WS_READ_REQUIRED_VALUE
,
4875 NULL
, value
, size
);
4877 LeaveCriticalSection( &reader
->cs
);
4881 /**************************************************************************
4882 * WsReadAttribute [webservices.@]
4884 HRESULT WINAPI
WsReadAttribute( WS_XML_READER
*handle
, const WS_ATTRIBUTE_DESCRIPTION
*desc
,
4885 WS_READ_OPTION option
, WS_HEAP
*heap
, void *value
, ULONG size
,
4888 struct reader
*reader
= (struct reader
*)handle
;
4891 TRACE( "%p %p %u %p %p %u %p\n", handle
, desc
, option
, heap
, value
, size
, error
);
4892 if (error
) FIXME( "ignoring error parameter\n" );
4894 if (!reader
|| !desc
|| !value
) return E_INVALIDARG
;
4896 EnterCriticalSection( &reader
->cs
);
4898 if (reader
->magic
!= READER_MAGIC
)
4900 LeaveCriticalSection( &reader
->cs
);
4901 return E_INVALIDARG
;
4904 if (!reader
->input_type
)
4906 LeaveCriticalSection( &reader
->cs
);
4907 return WS_E_INVALID_OPERATION
;
4910 hr
= read_type( reader
, WS_ATTRIBUTE_TYPE_MAPPING
, desc
->type
, desc
->attributeLocalName
,
4911 desc
->attributeNs
, desc
->typeDescription
, option
, heap
, value
, size
);
4913 LeaveCriticalSection( &reader
->cs
);
4917 static inline BOOL
is_utf8( const unsigned char *data
, ULONG size
, ULONG
*offset
)
4919 static const char bom
[] = {0xef,0xbb,0xbf};
4920 const unsigned char *p
= data
;
4922 return (size
>= sizeof(bom
) && !memcmp( p
, bom
, sizeof(bom
) ) && (*offset
= sizeof(bom
))) ||
4923 (size
> 2 && !(*offset
= 0));
4926 static inline BOOL
is_utf16le( const unsigned char *data
, ULONG size
, ULONG
*offset
)
4928 static const char bom
[] = {0xff,0xfe};
4929 const unsigned char *p
= data
;
4931 return (size
>= sizeof(bom
) && !memcmp( p
, bom
, sizeof(bom
) ) && (*offset
= sizeof(bom
))) ||
4932 (size
>= 4 && p
[0] == '<' && !p
[1] && !(*offset
= 0));
4935 static WS_CHARSET
detect_charset( const unsigned char *data
, ULONG size
, ULONG
*offset
)
4939 /* FIXME: parse xml declaration */
4941 if (is_utf16le( data
, size
, offset
)) ret
= WS_CHARSET_UTF16LE
;
4942 else if (is_utf8( data
, size
, offset
)) ret
= WS_CHARSET_UTF8
;
4945 FIXME( "charset not recognized\n" );
4949 TRACE( "detected charset %u\n", ret
);
4953 static void set_input_buffer( struct reader
*reader
, struct xmlbuf
*buf
, const unsigned char *data
, ULONG size
)
4955 reader
->input_type
= WS_XML_READER_INPUT_TYPE_BUFFER
;
4956 reader
->input_buf
= buf
;
4957 reader
->input_data
= data
;
4958 reader
->input_size
= size
;
4960 reader
->read_size
= reader
->input_size
;
4961 reader
->read_pos
= 0;
4962 reader
->read_bufptr
= reader
->input_data
;
4964 reader
->text_conv_offset
= 0;
4967 /**************************************************************************
4968 * WsSetInput [webservices.@]
4970 HRESULT WINAPI
WsSetInput( WS_XML_READER
*handle
, const WS_XML_READER_ENCODING
*encoding
,
4971 const WS_XML_READER_INPUT
*input
, const WS_XML_READER_PROPERTY
*properties
,
4972 ULONG count
, WS_ERROR
*error
)
4974 struct reader
*reader
= (struct reader
*)handle
;
4976 ULONG i
, offset
= 0;
4979 TRACE( "%p %p %p %p %u %p\n", handle
, encoding
, input
, properties
, count
, error
);
4980 if (error
) FIXME( "ignoring error parameter\n" );
4982 if (!reader
) return E_INVALIDARG
;
4984 EnterCriticalSection( &reader
->cs
);
4986 if (reader
->magic
!= READER_MAGIC
)
4988 LeaveCriticalSection( &reader
->cs
);
4989 return E_INVALIDARG
;
4992 for (i
= 0; i
< count
; i
++)
4994 hr
= prop_set( reader
->prop
, reader
->prop_count
, properties
[i
].id
, properties
[i
].value
,
4995 properties
[i
].valueSize
);
4996 if (hr
!= S_OK
) goto done
;
4999 if ((hr
= init_reader( reader
)) != S_OK
) goto done
;
5001 switch (encoding
->encodingType
)
5003 case WS_XML_READER_ENCODING_TYPE_TEXT
:
5005 WS_XML_READER_TEXT_ENCODING
*text
= (WS_XML_READER_TEXT_ENCODING
*)encoding
;
5006 WS_XML_READER_BUFFER_INPUT
*buf
= (WS_XML_READER_BUFFER_INPUT
*)input
;
5007 WS_CHARSET charset
= text
->charSet
;
5009 if (input
->inputType
!= WS_XML_READER_INPUT_TYPE_BUFFER
)
5011 FIXME( "charset detection on input type %u not supported\n", input
->inputType
);
5016 if (charset
== WS_CHARSET_AUTO
)
5017 charset
= detect_charset( buf
->encodedData
, buf
->encodedDataSize
, &offset
);
5019 hr
= prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_CHARSET
,
5020 &charset
, sizeof(charset
) );
5021 if (hr
!= S_OK
) goto done
;
5023 reader
->input_enc
= WS_XML_READER_ENCODING_TYPE_TEXT
;
5026 case WS_XML_READER_ENCODING_TYPE_BINARY
:
5027 reader
->input_enc
= WS_XML_READER_ENCODING_TYPE_BINARY
;
5031 FIXME( "encoding type %u not supported\n", encoding
->encodingType
);
5036 switch (input
->inputType
)
5038 case WS_XML_READER_INPUT_TYPE_BUFFER
:
5040 WS_XML_READER_BUFFER_INPUT
*buf
= (WS_XML_READER_BUFFER_INPUT
*)input
;
5041 set_input_buffer( reader
, NULL
, (const unsigned char *)buf
->encodedData
+ offset
,
5042 buf
->encodedDataSize
- offset
);
5046 FIXME( "input type %u not supported\n", input
->inputType
);
5051 if (!(node
= alloc_node( WS_XML_NODE_TYPE_BOF
))) hr
= E_OUTOFMEMORY
;
5052 else read_insert_bof( reader
, node
);
5055 LeaveCriticalSection( &reader
->cs
);
5059 /**************************************************************************
5060 * WsSetInputToBuffer [webservices.@]
5062 HRESULT WINAPI
WsSetInputToBuffer( WS_XML_READER
*handle
, WS_XML_BUFFER
*buffer
,
5063 const WS_XML_READER_PROPERTY
*properties
, ULONG count
,
5066 struct reader
*reader
= (struct reader
*)handle
;
5067 struct xmlbuf
*xmlbuf
= (struct xmlbuf
*)buffer
;
5070 ULONG i
, offset
= 0;
5073 TRACE( "%p %p %p %u %p\n", handle
, buffer
, properties
, count
, error
);
5074 if (error
) FIXME( "ignoring error parameter\n" );
5076 if (!reader
|| !xmlbuf
) return E_INVALIDARG
;
5078 EnterCriticalSection( &reader
->cs
);
5080 if (reader
->magic
!= READER_MAGIC
)
5082 LeaveCriticalSection( &reader
->cs
);
5083 return E_INVALIDARG
;
5086 for (i
= 0; i
< count
; i
++)
5088 hr
= prop_set( reader
->prop
, reader
->prop_count
, properties
[i
].id
, properties
[i
].value
,
5089 properties
[i
].valueSize
);
5090 if (hr
!= S_OK
) goto done
;
5093 if ((hr
= init_reader( reader
)) != S_OK
) goto done
;
5095 charset
= detect_charset( xmlbuf
->bytes
.bytes
, xmlbuf
->bytes
.length
, &offset
);
5096 hr
= prop_set( reader
->prop
, reader
->prop_count
, WS_XML_READER_PROPERTY_CHARSET
, &charset
,
5098 if (hr
!= S_OK
) goto done
;
5100 set_input_buffer( reader
, xmlbuf
, xmlbuf
->bytes
.bytes
+ offset
, xmlbuf
->bytes
.length
- offset
);
5101 if (!(node
= alloc_node( WS_XML_NODE_TYPE_BOF
))) hr
= E_OUTOFMEMORY
;
5102 else read_insert_bof( reader
, node
);
5105 LeaveCriticalSection( &reader
->cs
);
5109 /**************************************************************************
5110 * WsXmlStringEquals [webservices.@]
5112 HRESULT WINAPI
WsXmlStringEquals( const WS_XML_STRING
*str1
, const WS_XML_STRING
*str2
, WS_ERROR
*error
)
5114 TRACE( "%s %s %p\n", debugstr_xmlstr(str1
), debugstr_xmlstr(str2
), error
);
5115 if (error
) FIXME( "ignoring error parameter\n" );
5117 if (!str1
|| !str2
) return E_INVALIDARG
;
5119 if (str1
->length
!= str2
->length
) return S_FALSE
;
5120 if (!memcmp( str1
->bytes
, str2
->bytes
, str1
->length
)) return S_OK
;
5124 /**************************************************************************
5125 * WsGetReaderPosition [webservices.@]
5127 HRESULT WINAPI
WsGetReaderPosition( WS_XML_READER
*handle
, WS_XML_NODE_POSITION
*pos
, WS_ERROR
*error
)
5129 struct reader
*reader
= (struct reader
*)handle
;
5131 TRACE( "%p %p %p\n", handle
, pos
, error
);
5132 if (error
) FIXME( "ignoring error parameter\n" );
5134 if (!reader
|| !pos
) return E_INVALIDARG
;
5136 EnterCriticalSection( &reader
->cs
);
5138 if (reader
->magic
!= READER_MAGIC
)
5140 LeaveCriticalSection( &reader
->cs
);
5141 return E_INVALIDARG
;
5144 if (!reader
->input_buf
)
5146 LeaveCriticalSection( &reader
->cs
);
5147 return WS_E_INVALID_OPERATION
;
5150 pos
->buffer
= (WS_XML_BUFFER
*)reader
->input_buf
;
5151 pos
->node
= reader
->current
;
5153 LeaveCriticalSection( &reader
->cs
);
5157 /**************************************************************************
5158 * WsSetReaderPosition [webservices.@]
5160 HRESULT WINAPI
WsSetReaderPosition( WS_XML_READER
*handle
, const WS_XML_NODE_POSITION
*pos
, WS_ERROR
*error
)
5162 struct reader
*reader
= (struct reader
*)handle
;
5164 TRACE( "%p %p %p\n", handle
, pos
, error
);
5165 if (error
) FIXME( "ignoring error parameter\n" );
5167 if (!reader
|| !pos
) return E_INVALIDARG
;
5169 EnterCriticalSection( &reader
->cs
);
5171 if (reader
->magic
!= READER_MAGIC
|| (struct xmlbuf
*)pos
->buffer
!= reader
->input_buf
)
5173 LeaveCriticalSection( &reader
->cs
);
5174 return E_INVALIDARG
;
5177 if (!reader
->input_buf
)
5179 LeaveCriticalSection( &reader
->cs
);
5180 return WS_E_INVALID_OPERATION
;
5183 reader
->current
= pos
->node
;
5185 LeaveCriticalSection( &reader
->cs
);
5189 static HRESULT
utf8_to_base64( const WS_XML_UTF8_TEXT
*utf8
, WS_XML_BASE64_TEXT
*base64
)
5191 if (utf8
->value
.length
% 4) return WS_E_INVALID_FORMAT
;
5192 if (!(base64
->bytes
= heap_alloc( utf8
->value
.length
* 3 / 4 ))) return E_OUTOFMEMORY
;
5193 base64
->length
= decode_base64( utf8
->value
.bytes
, utf8
->value
.length
, base64
->bytes
);
5197 /**************************************************************************
5198 * WsReadBytes [webservices.@]
5200 HRESULT WINAPI
WsReadBytes( WS_XML_READER
*handle
, void *bytes
, ULONG max_count
, ULONG
*count
, WS_ERROR
*error
)
5202 struct reader
*reader
= (struct reader
*)handle
;
5205 TRACE( "%p %p %u %p %p\n", handle
, bytes
, max_count
, count
, error
);
5206 if (error
) FIXME( "ignoring error parameter\n" );
5208 if (!reader
) return E_INVALIDARG
;
5210 EnterCriticalSection( &reader
->cs
);
5212 if (reader
->magic
!= READER_MAGIC
)
5214 LeaveCriticalSection( &reader
->cs
);
5215 return E_INVALIDARG
;
5218 if (!reader
->input_type
)
5220 LeaveCriticalSection( &reader
->cs
);
5221 return WS_E_INVALID_OPERATION
;
5226 LeaveCriticalSection( &reader
->cs
);
5227 return E_INVALIDARG
;
5231 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_TEXT
&& bytes
)
5233 const WS_XML_TEXT_NODE
*text
= (const WS_XML_TEXT_NODE
*)reader
->current
;
5234 WS_XML_BASE64_TEXT base64
;
5236 if ((hr
= utf8_to_base64( (const WS_XML_UTF8_TEXT
*)text
->text
, &base64
)) != S_OK
)
5238 LeaveCriticalSection( &reader
->cs
);
5241 if (reader
->text_conv_offset
== base64
.length
)
5243 heap_free( base64
.bytes
);
5244 hr
= read_node( reader
);
5245 LeaveCriticalSection( &reader
->cs
);
5248 *count
= min( base64
.length
- reader
->text_conv_offset
, max_count
);
5249 memcpy( bytes
, base64
.bytes
+ reader
->text_conv_offset
, *count
);
5250 reader
->text_conv_offset
+= *count
;
5251 heap_free( base64
.bytes
);
5254 LeaveCriticalSection( &reader
->cs
);
5258 static HRESULT
utf8_to_utf16( const WS_XML_UTF8_TEXT
*utf8
, WS_XML_UTF16_TEXT
*utf16
)
5260 int len
= MultiByteToWideChar( CP_UTF8
, 0, (char *)utf8
->value
.bytes
, utf8
->value
.length
, NULL
, 0 );
5261 if (!(utf16
->bytes
= heap_alloc( len
* sizeof(WCHAR
) ))) return E_OUTOFMEMORY
;
5262 MultiByteToWideChar( CP_UTF8
, 0, (char *)utf8
->value
.bytes
, utf8
->value
.length
, (WCHAR
*)utf16
->bytes
, len
);
5263 utf16
->byteCount
= len
* sizeof(WCHAR
);
5267 /**************************************************************************
5268 * WsReadChars [webservices.@]
5270 HRESULT WINAPI
WsReadChars( WS_XML_READER
*handle
, WCHAR
*chars
, ULONG max_count
, ULONG
*count
, WS_ERROR
*error
)
5272 struct reader
*reader
= (struct reader
*)handle
;
5274 TRACE( "%p %p %u %p %p\n", handle
, chars
, max_count
, count
, error
);
5275 if (error
) FIXME( "ignoring error parameter\n" );
5277 if (!reader
) return E_INVALIDARG
;
5279 EnterCriticalSection( &reader
->cs
);
5281 if (reader
->magic
!= READER_MAGIC
)
5283 LeaveCriticalSection( &reader
->cs
);
5284 return E_INVALIDARG
;
5287 if (!reader
->input_type
)
5289 LeaveCriticalSection( &reader
->cs
);
5290 return WS_E_INVALID_OPERATION
;
5295 LeaveCriticalSection( &reader
->cs
);
5296 return E_INVALIDARG
;
5300 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_TEXT
&& chars
)
5302 const WS_XML_TEXT_NODE
*text
= (const WS_XML_TEXT_NODE
*)reader
->current
;
5303 WS_XML_UTF16_TEXT utf16
;
5306 if ((hr
= utf8_to_utf16( (const WS_XML_UTF8_TEXT
*)text
->text
, &utf16
)) != S_OK
)
5308 LeaveCriticalSection( &reader
->cs
);
5311 if (reader
->text_conv_offset
== utf16
.byteCount
/ sizeof(WCHAR
))
5313 heap_free( utf16
.bytes
);
5314 hr
= read_node( reader
);
5315 LeaveCriticalSection( &reader
->cs
);
5318 *count
= min( utf16
.byteCount
/ sizeof(WCHAR
) - reader
->text_conv_offset
, max_count
);
5319 memcpy( chars
, utf16
.bytes
+ reader
->text_conv_offset
* sizeof(WCHAR
), *count
* sizeof(WCHAR
) );
5320 reader
->text_conv_offset
+= *count
;
5321 heap_free( utf16
.bytes
);
5324 LeaveCriticalSection( &reader
->cs
);
5328 /**************************************************************************
5329 * WsReadCharsUtf8 [webservices.@]
5331 HRESULT WINAPI
WsReadCharsUtf8( WS_XML_READER
*handle
, BYTE
*bytes
, ULONG max_count
, ULONG
*count
, WS_ERROR
*error
)
5333 struct reader
*reader
= (struct reader
*)handle
;
5336 TRACE( "%p %p %u %p %p\n", handle
, bytes
, max_count
, count
, error
);
5337 if (error
) FIXME( "ignoring error parameter\n" );
5339 if (!reader
) return E_INVALIDARG
;
5341 EnterCriticalSection( &reader
->cs
);
5343 if (reader
->magic
!= READER_MAGIC
)
5345 LeaveCriticalSection( &reader
->cs
);
5346 return E_INVALIDARG
;
5349 if (!reader
->input_type
)
5351 LeaveCriticalSection( &reader
->cs
);
5352 return WS_E_INVALID_OPERATION
;
5357 LeaveCriticalSection( &reader
->cs
);
5358 return E_INVALIDARG
;
5362 if (node_type( reader
->current
) == WS_XML_NODE_TYPE_TEXT
&& bytes
)
5364 const WS_XML_TEXT_NODE
*text
= (const WS_XML_TEXT_NODE
*)reader
->current
;
5365 const WS_XML_UTF8_TEXT
*utf8
= (const WS_XML_UTF8_TEXT
*)text
->text
;
5367 if (reader
->text_conv_offset
== utf8
->value
.length
)
5369 hr
= read_node( reader
);
5370 LeaveCriticalSection( &reader
->cs
);
5373 *count
= min( utf8
->value
.length
- reader
->text_conv_offset
, max_count
);
5374 memcpy( bytes
, utf8
->value
.bytes
+ reader
->text_conv_offset
, *count
);
5375 reader
->text_conv_offset
+= *count
;
5378 LeaveCriticalSection( &reader
->cs
);
5382 HRESULT
get_param_desc( const WS_STRUCT_DESCRIPTION
*desc
, USHORT index
, const WS_FIELD_DESCRIPTION
**ret
)
5384 if (index
>= desc
->fieldCount
) return E_INVALIDARG
;
5385 *ret
= desc
->fields
[index
];
5389 static ULONG
get_field_size( const WS_FIELD_DESCRIPTION
*desc
)
5391 WS_READ_OPTION option
;
5394 switch ((option
= get_field_read_option( desc
->type
, desc
->options
)))
5396 case WS_READ_REQUIRED_POINTER
:
5397 case WS_READ_OPTIONAL_POINTER
:
5398 case WS_READ_NILLABLE_POINTER
:
5399 size
= sizeof(void *);
5402 case WS_READ_REQUIRED_VALUE
:
5403 case WS_READ_NILLABLE_VALUE
:
5404 size
= get_type_size( desc
->type
, desc
->typeDescription
);
5408 WARN( "unhandled option %u\n", option
);
5415 static HRESULT
read_param( struct reader
*reader
, const WS_FIELD_DESCRIPTION
*desc
, WS_HEAP
*heap
, void *ret
)
5417 if (!ret
&& !(ret
= ws_alloc_zero( heap
, get_field_size(desc
) ))) return WS_E_QUOTA_EXCEEDED
;
5418 return read_type_struct_field( reader
, desc
, heap
, ret
, 0 );
5421 static HRESULT
read_param_array( struct reader
*reader
, const WS_FIELD_DESCRIPTION
*desc
, WS_HEAP
*heap
,
5422 void **ret
, ULONG
*count
)
5424 if (!ret
&& !(ret
= ws_alloc_zero( heap
, sizeof(void **) ))) return WS_E_QUOTA_EXCEEDED
;
5425 return read_type_repeating_element( reader
, desc
, heap
, ret
, count
);
5428 static void set_array_len( const WS_PARAMETER_DESCRIPTION
*params
, ULONG count
, ULONG index
, ULONG len
,
5432 for (i
= 0; i
< count
; i
++)
5434 if (params
[i
].outputMessageIndex
!= index
|| params
[i
].parameterType
!= WS_PARAMETER_TYPE_ARRAY_COUNT
)
5436 if ((ptr
= *(ULONG
**)args
[i
])) *ptr
= len
;
5441 HRESULT
read_output_params( WS_XML_READER
*handle
, WS_HEAP
*heap
, const WS_ELEMENT_DESCRIPTION
*desc
,
5442 const WS_PARAMETER_DESCRIPTION
*params
, ULONG count
, const void **args
)
5444 struct reader
*reader
= (struct reader
*)handle
;
5445 const WS_STRUCT_DESCRIPTION
*desc_struct
;
5446 const WS_FIELD_DESCRIPTION
*desc_field
;
5450 if (desc
->type
!= WS_STRUCT_TYPE
|| !(desc_struct
= desc
->typeDescription
)) return E_INVALIDARG
;
5452 EnterCriticalSection( &reader
->cs
);
5454 if (reader
->magic
!= READER_MAGIC
)
5456 LeaveCriticalSection( &reader
->cs
);
5457 return E_INVALIDARG
;
5460 if ((hr
= start_mapping( reader
, WS_ELEMENT_TYPE_MAPPING
, desc
->elementLocalName
, desc
->elementNs
)) != S_OK
)
5463 for (i
= 0; i
< count
; i
++)
5465 if (params
[i
].outputMessageIndex
== INVALID_PARAMETER_INDEX
) continue;
5466 if (params
[i
].parameterType
== WS_PARAMETER_TYPE_MESSAGES
)
5468 FIXME( "messages type not supported\n" );
5472 if ((hr
= get_param_desc( desc_struct
, params
[i
].outputMessageIndex
, &desc_field
)) != S_OK
) goto done
;
5473 if (params
[i
].parameterType
== WS_PARAMETER_TYPE_NORMAL
)
5475 void *ptr
= *(void **)args
[i
];
5476 if ((hr
= read_param( reader
, desc_field
, heap
, ptr
)) != S_OK
) goto done
;
5478 else if (params
[i
].parameterType
== WS_PARAMETER_TYPE_ARRAY
)
5480 void **ptr
= *(void ***)args
[i
];
5481 if ((hr
= read_param_array( reader
, desc_field
, heap
, ptr
, &len
)) != S_OK
) goto done
;
5482 set_array_len( params
, count
, params
[i
].outputMessageIndex
, len
, args
);
5486 if (desc_struct
->structOptions
& WS_STRUCT_IGNORE_TRAILING_ELEMENT_CONTENT
)
5488 struct node
*parent
= find_parent( reader
);
5489 parent
->flags
|= NODE_FLAG_IGNORE_TRAILING_ELEMENT_CONTENT
;
5492 hr
= end_mapping( reader
, WS_ELEMENT_TYPE_MAPPING
);
5495 LeaveCriticalSection( &reader
->cs
);