blob: b3b6fc4f992b9f8328a4200cde6031c77fff9274 [file] [log] [blame]
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11001/*
2 * SAX Reader implementation
3 *
4 * Copyright 2008 Alistair Leslie-Hughes
Piotr Caband3e9ca72008-07-17 00:40:32 +02005 * Copyright 2008 Piotr Caban
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11006 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 */
21#define COBJMACROS
22
23#include "config.h"
24
25#include <stdarg.h>
Alexandre Julliarda1c8f6d2011-02-24 13:11:53 +010026#ifdef HAVE_LIBXML2
27# include <libxml/parser.h>
28# include <libxml/xmlerror.h>
29# include <libxml/SAX2.h>
30# include <libxml/parserInternals.h>
31#endif
32
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +110033#include "windef.h"
34#include "winbase.h"
35#include "winuser.h"
36#include "winnls.h"
37#include "ole2.h"
Nikolay Sivov62e521a2010-08-30 01:31:49 +040038#include "msxml6.h"
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +110039#include "wininet.h"
40#include "urlmon.h"
41#include "winreg.h"
42#include "shlwapi.h"
43
44#include "wine/debug.h"
Nikolay Sivov9d662922011-12-24 16:10:33 +030045#include "wine/list.h"
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +110046
47#include "msxml_private.h"
48
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +110049WINE_DEFAULT_DEBUG_CHANNEL(msxml);
50
51#ifdef HAVE_LIBXML2
52
Nikolay Sivovf999f4a2011-12-16 14:33:32 +030053typedef enum
Nikolay Sivovad336be2011-10-04 16:27:39 -050054{
Nikolay Sivov137a21d2012-04-18 22:39:38 +040055 FeatureUnknown = 0,
Nikolay Sivovad336be2011-10-04 16:27:39 -050056 ExhaustiveErrors = 1 << 1,
57 ExternalGeneralEntities = 1 << 2,
58 ExternalParameterEntities = 1 << 3,
59 ForcedResync = 1 << 4,
60 NamespacePrefixes = 1 << 5,
Nikolay Sivov3d3786c2011-10-05 13:29:37 -050061 Namespaces = 1 << 6,
Nikolay Sivovad336be2011-10-04 16:27:39 -050062 ParameterEntities = 1 << 7,
63 PreserveSystemIndentifiers = 1 << 8,
64 ProhibitDTD = 1 << 9,
65 SchemaValidation = 1 << 10,
66 ServerHttpRequest = 1 << 11,
67 SuppressValidationfatalError = 1 << 12,
68 UseInlineSchema = 1 << 13,
Nikolay Sivovcc4c6942011-10-05 09:37:40 -050069 UseSchemaLocation = 1 << 14,
70 LexicalHandlerParEntities = 1 << 15
Nikolay Sivov137a21d2012-04-18 22:39:38 +040071} saxreader_feature;
72
73/* feature names */
74static const WCHAR FeatureExternalGeneralEntitiesW[] = {
75 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/','s','a','x','/',
76 'f','e','a','t','u','r','e','s','/','e','x','t','e','r','n','a','l','-','g','e','n','e','r','a','l',
77 '-','e','n','t','i','t','i','e','s',0
78};
79
80static const WCHAR FeatureExternalParameterEntitiesW[] = {
81 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/','s','a','x','/','f','e','a','t','u','r','e','s',
82 '/','e','x','t','e','r','n','a','l','-','p','a','r','a','m','e','t','e','r','-','e','n','t','i','t','i','e','s',0
83};
84
85static const WCHAR FeatureLexicalHandlerParEntitiesW[] = {
86 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/','s','a','x','/','f','e','a','t','u','r','e','s',
87 '/','l','e','x','i','c','a','l','-','h','a','n','d','l','e','r','/','p','a','r','a','m','e','t','e','r','-','e','n','t','i','t','i','e','s',0
88};
89
90static const WCHAR FeatureProhibitDTDW[] = {
91 'p','r','o','h','i','b','i','t','-','d','t','d',0
92};
93
94static const WCHAR FeatureNamespacesW[] = {
95 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/','s','a','x','/','f','e','a','t','u','r','e','s',
96 '/','n','a','m','e','s','p','a','c','e','s',0
97};
98
99static const WCHAR FeatureNamespacePrefixesW[] = {
100 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/','s','a','x','/','f','e','a','t','u','r','e','s',
101 '/','n','a','m','e','s','p','a','c','e','-','p','r','e','f','i','x','e','s',0
102};
103
104struct saxreader_feature_pair
105{
106 saxreader_feature feature;
107 const WCHAR *name;
108};
109
110static const struct saxreader_feature_pair saxreader_feature_map[] = {
111 { ExternalGeneralEntities, FeatureExternalGeneralEntitiesW },
112 { ExternalParameterEntities, FeatureExternalParameterEntitiesW },
113 { LexicalHandlerParEntities, FeatureLexicalHandlerParEntitiesW },
114 { NamespacePrefixes, FeatureNamespacePrefixesW },
115 { Namespaces, FeatureNamespacesW },
116 { ProhibitDTD, FeatureProhibitDTDW }
117};
118
119static saxreader_feature get_saxreader_feature(const WCHAR *name)
120{
121 int min, max, n, c;
122
123 min = 0;
124 max = sizeof(saxreader_feature_map)/sizeof(struct saxreader_feature_pair) - 1;
125
126 while (min <= max)
127 {
128 n = (min+max)/2;
129
130 c = strcmpW(saxreader_feature_map[n].name, name);
131 if (!c)
132 return saxreader_feature_map[n].feature;
133
134 if (c > 0)
135 max = n-1;
136 else
137 min = n+1;
138 }
139
140 return FeatureUnknown;
141}
Nikolay Sivovad336be2011-10-04 16:27:39 -0500142
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -0500143struct bstrpool
144{
145 BSTR *pool;
146 unsigned int index;
147 unsigned int len;
148};
149
Nikolay Sivovf999f4a2011-12-16 14:33:32 +0300150typedef struct
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100151{
Nikolay Sivov9d662922011-12-24 16:10:33 +0300152 BSTR prefix;
153 BSTR uri;
154} ns;
155
156typedef struct
157{
158 struct list entry;
159 BSTR prefix;
160 BSTR local;
161 BSTR qname;
162 ns *ns; /* namespaces defined in this particular element */
163 int ns_count;
164} element_entry;
165
166typedef struct
167{
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +0300168 DispatchEx dispex;
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100169 IVBSAXXMLReader IVBSAXXMLReader_iface;
170 ISAXXMLReader ISAXXMLReader_iface;
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100171 LONG ref;
Nikolay Sivovf999f4a2011-12-16 14:33:32 +0300172 ISAXContentHandler *contentHandler;
173 IVBSAXContentHandler *vbcontentHandler;
174 ISAXErrorHandler *errorHandler;
175 IVBSAXErrorHandler *vberrorHandler;
176 ISAXLexicalHandler *lexicalHandler;
177 IVBSAXLexicalHandler *vblexicalHandler;
178 ISAXDeclHandler *declHandler;
179 IVBSAXDeclHandler *vbdeclHandler;
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100180 xmlSAXHandler sax;
Piotr Cabanc7fc9262008-10-01 19:52:36 +0200181 BOOL isParsing;
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -0500182 struct bstrpool pool;
Nikolay Sivov137a21d2012-04-18 22:39:38 +0400183 saxreader_feature features;
Piotr Cabane84c4102011-10-24 20:08:46 +0200184 MSXML_VERSION version;
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100185} saxreader;
186
Nikolay Sivovf999f4a2011-12-16 14:33:32 +0300187typedef struct
Piotr Caband3e9ca72008-07-17 00:40:32 +0200188{
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100189 IVBSAXLocator IVBSAXLocator_iface;
190 ISAXLocator ISAXLocator_iface;
Piotr Caban1b462062011-10-31 12:32:39 +0100191 IVBSAXAttributes IVBSAXAttributes_iface;
192 ISAXAttributes ISAXAttributes_iface;
Piotr Caband3e9ca72008-07-17 00:40:32 +0200193 LONG ref;
194 saxreader *saxreader;
Piotr Caban51c95272008-07-17 00:40:53 +0200195 HRESULT ret;
196 xmlParserCtxtPtr pParserCtxt;
Piotr Caban60ca0a72008-07-17 00:42:01 +0200197 WCHAR *publicId;
Piotr Caban3196f782008-07-17 00:42:07 +0200198 WCHAR *systemId;
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200199 int line;
200 int column;
Piotr Cabanc52e0912008-07-31 16:48:03 +0200201 BOOL vbInterface;
Nikolay Sivov9d662922011-12-24 16:10:33 +0300202 struct list elements;
Piotr Caband3e9ca72008-07-17 00:40:32 +0200203
Piotr Cabana95b35d2011-10-31 12:33:18 +0100204 BSTR namespaceUri;
Piotr Caban1b462062011-10-31 12:32:39 +0100205 int attributesSize;
Piotr Caban6df49d52008-07-30 20:24:12 +0200206 int nb_attributes;
Piotr Caban1b462062011-10-31 12:32:39 +0100207 struct _attributes
208 {
209 BSTR szLocalname;
210 BSTR szURI;
211 BSTR szValue;
212 BSTR szQName;
213 } *attributes;
214} saxlocator;
Piotr Caban5816c382008-07-27 19:55:05 +0200215
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100216static inline saxreader *impl_from_IVBSAXXMLReader( IVBSAXXMLReader *iface )
217{
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100218 return CONTAINING_RECORD(iface, saxreader, IVBSAXXMLReader_iface);
Andrew Talbot7142fc12008-03-31 20:42:18 +0100219}
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +1100220
Piotr Caban19267602008-07-08 20:52:04 +0200221static inline saxreader *impl_from_ISAXXMLReader( ISAXXMLReader *iface )
222{
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100223 return CONTAINING_RECORD(iface, saxreader, ISAXXMLReader_iface);
Piotr Caban19267602008-07-08 20:52:04 +0200224}
225
Piotr Caban4c69e862008-07-31 16:47:26 +0200226static inline saxlocator *impl_from_IVBSAXLocator( IVBSAXLocator *iface )
227{
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100228 return CONTAINING_RECORD(iface, saxlocator, IVBSAXLocator_iface);
Piotr Caban4c69e862008-07-31 16:47:26 +0200229}
230
Piotr Caband3e9ca72008-07-17 00:40:32 +0200231static inline saxlocator *impl_from_ISAXLocator( ISAXLocator *iface )
232{
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100233 return CONTAINING_RECORD(iface, saxlocator, ISAXLocator_iface);
Piotr Caband3e9ca72008-07-17 00:40:32 +0200234}
235
Piotr Caban1b462062011-10-31 12:32:39 +0100236static inline saxlocator *impl_from_IVBSAXAttributes( IVBSAXAttributes *iface )
Piotr Caban4c69e862008-07-31 16:47:26 +0200237{
Piotr Caban1b462062011-10-31 12:32:39 +0100238 return CONTAINING_RECORD(iface, saxlocator, IVBSAXAttributes_iface);
Piotr Caban4c69e862008-07-31 16:47:26 +0200239}
240
Piotr Caban1b462062011-10-31 12:32:39 +0100241static inline saxlocator *impl_from_ISAXAttributes( ISAXAttributes *iface )
Piotr Caban5816c382008-07-27 19:55:05 +0200242{
Piotr Caban1b462062011-10-31 12:32:39 +0100243 return CONTAINING_RECORD(iface, saxlocator, ISAXAttributes_iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200244}
245
Nikolay Sivov864f1702011-09-21 21:01:10 +0400246/* property names */
247static const WCHAR PropertyCharsetW[] = {
248 'c','h','a','r','s','e','t',0
249};
250static const WCHAR PropertyDeclHandlerW[] = {
251 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/',
252 's','a','x','/','p','r','o','p','e','r','t','i','e','s','/',
253 'd','e','c','l','a','r','a','t','i','o','n',
254 '-','h','a','n','d','l','e','r',0
255};
256static const WCHAR PropertyDomNodeW[] = {
257 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/',
258 's','a','x','/','p','r','o','p','e','r','t','i','e','s','/',
259 'd','o','m','-','n','o','d','e',0
260};
261static const WCHAR PropertyInputSourceW[] = {
262 'i','n','p','u','t','-','s','o','u','r','c','e',0
263};
264static const WCHAR PropertyLexicalHandlerW[] = {
265 'h','t','t','p',':','/','/','x','m','l','.','o','r','g','/',
266 's','a','x','/','p','r','o','p','e','r','t','i','e','s','/',
267 'l','e','x','i','c','a','l','-','h','a','n','d','l','e','r',0
268};
269static const WCHAR PropertyMaxElementDepthW[] = {
270 'm','a','x','-','e','l','e','m','e','n','t','-','d','e','p','t','h',0
271};
272static const WCHAR PropertyMaxXMLSizeW[] = {
273 'm','a','x','-','x','m','l','-','s','i','z','e',0
274};
275static const WCHAR PropertySchemaDeclHandlerW[] = {
276 's','c','h','e','m','a','-','d','e','c','l','a','r','a','t','i','o','n','-',
277 'h','a','n','d','l','e','r',0
278};
279static const WCHAR PropertyXMLDeclEncodingW[] = {
280 'x','m','l','d','e','c','l','-','e','n','c','o','d','i','n','g',0
281};
282static const WCHAR PropertyXMLDeclStandaloneW[] = {
283 'x','m','l','d','e','c','l','-','s','t','a','n','d','a','l','o','n','e',0
284};
285static const WCHAR PropertyXMLDeclVersionW[] = {
286 'x','m','l','d','e','c','l','-','v','e','r','s','i','o','n',0
287};
288
Nikolay Sivov137a21d2012-04-18 22:39:38 +0400289static inline HRESULT set_feature_value(saxreader *reader, saxreader_feature feature, VARIANT_BOOL value)
Nikolay Sivovad336be2011-10-04 16:27:39 -0500290{
Nikolay Sivov54df4812012-04-19 10:53:15 +0400291 /* handling of non-VARIANT_* values is version dependent */
292 if ((reader->version < MSXML4) && (value != VARIANT_TRUE))
293 value = VARIANT_FALSE;
294 if ((reader->version >= MSXML4) && (value != VARIANT_FALSE))
295 value = VARIANT_TRUE;
296
Nikolay Sivovad336be2011-10-04 16:27:39 -0500297 if (value == VARIANT_TRUE)
298 reader->features |= feature;
299 else
300 reader->features &= ~feature;
301
302 return S_OK;
303}
304
Nikolay Sivov137a21d2012-04-18 22:39:38 +0400305static inline HRESULT get_feature_value(const saxreader *reader, saxreader_feature feature, VARIANT_BOOL *value)
Nikolay Sivov3d3786c2011-10-05 13:29:37 -0500306{
307 *value = reader->features & feature ? VARIANT_TRUE : VARIANT_FALSE;
308 return S_OK;
309}
310
Nikolay Sivovf27159a2012-04-19 10:46:03 +0400311static BOOL is_namespaces_enabled(const saxreader *reader)
312{
313 return (reader->version < MSXML4) || (reader->features & Namespaces);
314}
315
Nikolay Sivov2dc67112012-05-08 15:50:19 +0400316static inline int has_content_handler(const saxlocator *locator)
Nikolay Sivovff992282010-01-18 23:30:50 +0300317{
318 return (locator->vbInterface && locator->saxreader->vbcontentHandler) ||
319 (!locator->vbInterface && locator->saxreader->contentHandler);
320}
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200321
Nikolay Sivovfc4dab42012-05-08 15:56:20 +0400322static inline int has_lexical_handler(const saxlocator *locator)
323{
324 return (locator->vbInterface && locator->saxreader->vblexicalHandler) ||
325 (!locator->vbInterface && locator->saxreader->lexicalHandler);
326}
327
328static inline int has_error_handler(const saxlocator *locator)
Nikolay Sivov8f9804e2010-10-29 01:04:44 +0400329{
330 return (locator->vbInterface && locator->saxreader->vberrorHandler) ||
331 (!locator->vbInterface && locator->saxreader->errorHandler);
332}
333
Nikolay Sivov9d662922011-12-24 16:10:33 +0300334static BSTR build_qname(BSTR prefix, BSTR local)
Piotr Caban03f79092008-08-20 18:20:50 +0200335{
Nikolay Sivov9d662922011-12-24 16:10:33 +0300336 if (prefix && *prefix)
Piotr Caban03f79092008-08-20 18:20:50 +0200337 {
Nikolay Sivov9d662922011-12-24 16:10:33 +0300338 BSTR qname = SysAllocStringLen(NULL, SysStringLen(prefix) + SysStringLen(local) + 1);
339 WCHAR *ptr;
Piotr Caban03f79092008-08-20 18:20:50 +0200340
Nikolay Sivov9d662922011-12-24 16:10:33 +0300341 ptr = qname;
342 strcpyW(ptr, prefix);
343 ptr += SysStringLen(prefix);
344 *ptr++ = ':';
345 strcpyW(ptr, local);
346 return qname;
Piotr Caban619c2dc2011-10-31 12:32:59 +0100347 }
348 else
Nikolay Sivov9d662922011-12-24 16:10:33 +0300349 return SysAllocString(local);
Piotr Caban03f79092008-08-20 18:20:50 +0200350}
351
Nikolay Sivov9d662922011-12-24 16:10:33 +0300352static element_entry* alloc_element_entry(const xmlChar *local, const xmlChar *prefix, int nb_ns,
353 const xmlChar **namespaces)
Piotr Caban03f79092008-08-20 18:20:50 +0200354{
Nikolay Sivov9d662922011-12-24 16:10:33 +0300355 element_entry *ret;
356 int i;
357
358 ret = heap_alloc(sizeof(*ret));
359 if (!ret) return ret;
360
361 ret->local = bstr_from_xmlChar(local);
362 ret->prefix = bstr_from_xmlChar(prefix);
363 ret->qname = build_qname(ret->prefix, ret->local);
364 ret->ns = nb_ns ? heap_alloc(nb_ns*sizeof(ns)) : NULL;
365 ret->ns_count = nb_ns;
366
367 for (i=0; i < nb_ns; i++)
Piotr Caban619c2dc2011-10-31 12:32:59 +0100368 {
Nikolay Sivov9d662922011-12-24 16:10:33 +0300369 ret->ns[i].prefix = bstr_from_xmlChar(namespaces[2*i]);
370 ret->ns[i].uri = bstr_from_xmlChar(namespaces[2*i+1]);
Piotr Caban619c2dc2011-10-31 12:32:59 +0100371 }
372
Nikolay Sivov9d662922011-12-24 16:10:33 +0300373 return ret;
Piotr Caban03f79092008-08-20 18:20:50 +0200374}
375
Nikolay Sivov9d662922011-12-24 16:10:33 +0300376static void free_element_entry(element_entry *element)
Piotr Caban216b6482011-10-31 12:33:08 +0100377{
378 int i;
379
Nikolay Sivov9d662922011-12-24 16:10:33 +0300380 for (i=0; i<element->ns_count;i++)
Piotr Caban216b6482011-10-31 12:33:08 +0100381 {
Nikolay Sivov9d662922011-12-24 16:10:33 +0300382 SysFreeString(element->ns[i].prefix);
383 SysFreeString(element->ns[i].uri);
Piotr Caban216b6482011-10-31 12:33:08 +0100384 }
385
Nikolay Sivov9d662922011-12-24 16:10:33 +0300386 SysFreeString(element->prefix);
387 SysFreeString(element->local);
388
389 heap_free(element->ns);
390 heap_free(element);
391}
392
393static void push_element_ns(saxlocator *locator, element_entry *element)
394{
395 list_add_head(&locator->elements, &element->entry);
396}
397
398static element_entry * pop_element_ns(saxlocator *locator)
399{
400 element_entry *element = LIST_ENTRY(list_head(&locator->elements), element_entry, entry);
401
402 if (element)
403 list_remove(&element->entry);
404
405 return element;
406}
407
408static BSTR find_element_uri(saxlocator *locator, const xmlChar *uri)
409{
410 element_entry *element;
411 BSTR uriW;
412 int i;
413
414 if (!uri) return NULL;
415
416 uriW = bstr_from_xmlChar(uri);
417
418 LIST_FOR_EACH_ENTRY(element, &locator->elements, element_entry, entry)
419 {
420 for (i=0; i < element->ns_count; i++)
421 if (!strcmpW(uriW, element->ns[i].uri))
422 {
423 SysFreeString(uriW);
424 return element->ns[i].uri;
425 }
426 }
427
428 SysFreeString(uriW);
429 ERR("namespace uri not found, %s\n", debugstr_a((char*)uri));
Piotr Caban216b6482011-10-31 12:33:08 +0100430 return NULL;
431}
432
Nikolay Sivov9d662922011-12-24 16:10:33 +0300433/* used to localize version dependent error check behaviour */
434static inline BOOL sax_callback_failed(saxlocator *This, HRESULT hr)
435{
Nikolay Sivovbcd85852012-04-15 13:47:30 +0400436 return This->saxreader->version >= MSXML4 ? FAILED(hr) : hr != S_OK;
Nikolay Sivov9d662922011-12-24 16:10:33 +0300437}
438
439/* index value -1 means it tries to loop for a first time */
440static inline BOOL iterate_endprefix_index(saxlocator *This, const element_entry *element, int *i)
441{
Nikolay Sivovf9c3a512012-04-16 02:29:18 +0400442 if (This->saxreader->version >= MSXML4)
Nikolay Sivov9d662922011-12-24 16:10:33 +0300443 {
444 if (*i == -1) *i = 0; else ++*i;
445 return *i < element->ns_count;
446 }
447 else
448 {
449 if (*i == -1) *i = element->ns_count-1; else --*i;
450 return *i >= 0;
451 }
452}
453
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -0500454static BOOL bstr_pool_insert(struct bstrpool *pool, BSTR pool_entry)
455{
456 if (!pool->pool)
457 {
458 pool->pool = HeapAlloc(GetProcessHeap(), 0, 16 * sizeof(*pool->pool));
459 if (!pool->pool)
460 return FALSE;
461
462 pool->index = 0;
463 pool->len = 16;
464 }
465 else if (pool->index == pool->len)
466 {
467 BSTR *realloc = HeapReAlloc(GetProcessHeap(), 0, pool->pool, pool->len * 2 * sizeof(*realloc));
468
469 if (!realloc)
470 return FALSE;
471
472 pool->pool = realloc;
473 pool->len *= 2;
474 }
475
476 pool->pool[pool->index++] = pool_entry;
477 return TRUE;
478}
479
480static void free_bstr_pool(struct bstrpool *pool)
481{
482 unsigned int i;
483
484 for (i = 0; i < pool->index; i++)
485 SysFreeString(pool->pool[i]);
486
487 HeapFree(GetProcessHeap(), 0, pool->pool);
488
489 pool->pool = NULL;
490 pool->index = pool->len = 0;
491}
492
Piotr Caban319d86c2008-08-03 14:37:37 +0200493static BSTR bstr_from_xmlCharN(const xmlChar *buf, int len)
Piotr Caban6df49d52008-07-30 20:24:12 +0200494{
495 DWORD dLen;
Piotr Caban6df49d52008-07-30 20:24:12 +0200496 BSTR bstr;
497
498 if (!buf)
499 return NULL;
500
501 dLen = MultiByteToWideChar(CP_UTF8, 0, (LPCSTR)buf, len, NULL, 0);
502 if(len != -1) dLen++;
Piotr Caban7a223652010-07-17 14:06:53 +0200503 bstr = SysAllocStringLen(NULL, dLen-1);
504 if (!bstr)
Piotr Caban6df49d52008-07-30 20:24:12 +0200505 return NULL;
Piotr Caban7a223652010-07-17 14:06:53 +0200506 MultiByteToWideChar(CP_UTF8, 0, (LPCSTR)buf, len, bstr, dLen);
507 if(len != -1) bstr[dLen-1] = '\0';
Piotr Caban6df49d52008-07-30 20:24:12 +0200508
509 return bstr;
510}
511
Piotr Caban319d86c2008-08-03 14:37:37 +0200512static BSTR QName_from_xmlChar(const xmlChar *prefix, const xmlChar *name)
513{
Nikolay Sivov2329d2a2010-10-29 01:14:08 +0400514 xmlChar *qname;
Piotr Caban319d86c2008-08-03 14:37:37 +0200515 BSTR bstr;
516
517 if(!name) return NULL;
518
Nikolay Sivov2329d2a2010-10-29 01:14:08 +0400519 if(!prefix || !*prefix)
Piotr Caban319d86c2008-08-03 14:37:37 +0200520 return bstr_from_xmlChar(name);
521
Nikolay Sivov2329d2a2010-10-29 01:14:08 +0400522 qname = xmlBuildQName(name, prefix, NULL, 0);
523 bstr = bstr_from_xmlChar(qname);
524 xmlFree(qname);
Piotr Caban319d86c2008-08-03 14:37:37 +0200525
526 return bstr;
527}
528
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -0500529static BSTR pooled_bstr_from_xmlChar(struct bstrpool *pool, const xmlChar *buf)
530{
531 BSTR pool_entry = bstr_from_xmlChar(buf);
532
533 if (pool_entry && !bstr_pool_insert(pool, pool_entry))
534 {
535 SysFreeString(pool_entry);
536 return NULL;
537 }
538
539 return pool_entry;
540}
541
542static BSTR pooled_bstr_from_xmlCharN(struct bstrpool *pool, const xmlChar *buf, int len)
543{
544 BSTR pool_entry = bstr_from_xmlCharN(buf, len);
545
546 if (pool_entry && !bstr_pool_insert(pool, pool_entry))
547 {
548 SysFreeString(pool_entry);
549 return NULL;
550 }
551
552 return pool_entry;
553}
554
Piotr Cabanbba7eb52008-07-19 22:32:10 +0200555static void format_error_message_from_id(saxlocator *This, HRESULT hr)
556{
557 xmlStopParser(This->pParserCtxt);
558 This->ret = hr;
559
Nikolay Sivov8f9804e2010-10-29 01:04:44 +0400560 if(has_error_handler(This))
Piotr Cabanbba7eb52008-07-19 22:32:10 +0200561 {
562 WCHAR msg[1024];
563 if(!FormatMessageW(FORMAT_MESSAGE_FROM_SYSTEM,
564 NULL, hr, 0, msg, sizeof(msg), NULL))
565 {
566 FIXME("MSXML errors not yet supported.\n");
567 msg[0] = '\0';
568 }
569
Piotr Cabanc52e0912008-07-31 16:48:03 +0200570 if(This->vbInterface)
571 {
572 BSTR bstrMsg = SysAllocString(msg);
573 IVBSAXErrorHandler_fatalError(This->saxreader->vberrorHandler,
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100574 &This->IVBSAXLocator_iface, &bstrMsg, hr);
Nikolay Sivovbd7787c2010-10-27 01:15:38 +0400575 SysFreeString(bstrMsg);
Piotr Cabanc52e0912008-07-31 16:48:03 +0200576 }
577 else
578 ISAXErrorHandler_fatalError(This->saxreader->errorHandler,
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100579 &This->ISAXLocator_iface, msg, hr);
Piotr Cabanbba7eb52008-07-19 22:32:10 +0200580 }
581}
582
Piotr Cabanc253bab2011-11-28 15:05:25 +0100583static void update_position(saxlocator *This, BOOL fix_column)
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200584{
Piotr Cabanc253bab2011-11-28 15:05:25 +0100585 const xmlChar *p = This->pParserCtxt->input->cur-1;
586
587 This->line = xmlSAX2GetLineNumber(This->pParserCtxt);
588 if(fix_column)
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200589 {
Piotr Cabanc253bab2011-11-28 15:05:25 +0100590 This->column = 1;
591 for(; *p!='\n' && *p!='\r' && p>=This->pParserCtxt->input->base; p--)
592 This->column++;
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200593 }
Piotr Cabanc253bab2011-11-28 15:05:25 +0100594 else
Piotr Cabandf5e40e2008-07-31 16:45:30 +0200595 {
Piotr Cabanc253bab2011-11-28 15:05:25 +0100596 This->column = xmlSAX2GetColumnNumber(This->pParserCtxt);
Piotr Cabanc801c182008-10-07 22:10:43 +0200597 }
Piotr Cabanbb849dc2008-07-23 16:39:04 +0200598}
599
Piotr Cabancafbf852008-07-31 16:47:41 +0200600/*** IVBSAXAttributes interface ***/
601/*** IUnknown methods ***/
602static HRESULT WINAPI ivbsaxattributes_QueryInterface(
603 IVBSAXAttributes* iface,
604 REFIID riid,
605 void **ppvObject)
606{
Piotr Caban1b462062011-10-31 12:32:39 +0100607 saxlocator *This = impl_from_IVBSAXAttributes(iface);
Piotr Cabancafbf852008-07-31 16:47:41 +0200608 TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject);
Piotr Caban1b462062011-10-31 12:32:39 +0100609 return IVBSAXLocator_QueryInterface(&This->IVBSAXLocator_iface, riid, ppvObject);
Piotr Cabancafbf852008-07-31 16:47:41 +0200610}
611
612static ULONG WINAPI ivbsaxattributes_AddRef(IVBSAXAttributes* iface)
613{
Piotr Caban1b462062011-10-31 12:32:39 +0100614 saxlocator *This = impl_from_IVBSAXAttributes(iface);
615 return ISAXLocator_AddRef(&This->ISAXLocator_iface);
Piotr Cabancafbf852008-07-31 16:47:41 +0200616}
617
618static ULONG WINAPI ivbsaxattributes_Release(IVBSAXAttributes* iface)
619{
Piotr Caban1b462062011-10-31 12:32:39 +0100620 saxlocator *This = impl_from_IVBSAXAttributes(iface);
621 return ISAXLocator_Release(&This->ISAXLocator_iface);
Piotr Cabancafbf852008-07-31 16:47:41 +0200622}
623
624/*** IDispatch methods ***/
625static HRESULT WINAPI ivbsaxattributes_GetTypeInfoCount( IVBSAXAttributes *iface, UINT* pctinfo )
626{
Piotr Caban1b462062011-10-31 12:32:39 +0100627 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Piotr Cabancafbf852008-07-31 16:47:41 +0200628
629 TRACE("(%p)->(%p)\n", This, pctinfo);
630
631 *pctinfo = 1;
632
633 return S_OK;
634}
635
636static HRESULT WINAPI ivbsaxattributes_GetTypeInfo(
637 IVBSAXAttributes *iface,
638 UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo )
639{
Piotr Caban1b462062011-10-31 12:32:39 +0100640 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Piotr Cabancafbf852008-07-31 16:47:41 +0200641 HRESULT hr;
642
643 TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
644
645 hr = get_typeinfo(IVBSAXAttributes_tid, ppTInfo);
646
647 return hr;
648}
649
650static HRESULT WINAPI ivbsaxattributes_GetIDsOfNames(
651 IVBSAXAttributes *iface,
652 REFIID riid,
653 LPOLESTR* rgszNames,
654 UINT cNames,
655 LCID lcid,
656 DISPID* rgDispId)
657{
Piotr Caban1b462062011-10-31 12:32:39 +0100658 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Piotr Cabancafbf852008-07-31 16:47:41 +0200659 ITypeInfo *typeinfo;
660 HRESULT hr;
661
662 TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
663 lcid, rgDispId);
664
665 if(!rgszNames || cNames == 0 || !rgDispId)
666 return E_INVALIDARG;
667
668 hr = get_typeinfo(IVBSAXAttributes_tid, &typeinfo);
669 if(SUCCEEDED(hr))
670 {
671 hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
672 ITypeInfo_Release(typeinfo);
673 }
674
675 return hr;
676}
677
678static HRESULT WINAPI ivbsaxattributes_Invoke(
679 IVBSAXAttributes *iface,
680 DISPID dispIdMember,
681 REFIID riid,
682 LCID lcid,
683 WORD wFlags,
684 DISPPARAMS* pDispParams,
685 VARIANT* pVarResult,
686 EXCEPINFO* pExcepInfo,
687 UINT* puArgErr)
688{
Piotr Caban1b462062011-10-31 12:32:39 +0100689 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Piotr Cabancafbf852008-07-31 16:47:41 +0200690 ITypeInfo *typeinfo;
691 HRESULT hr;
692
693 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
694 lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
695
696 hr = get_typeinfo(IVBSAXAttributes_tid, &typeinfo);
697 if(SUCCEEDED(hr))
698 {
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100699 hr = ITypeInfo_Invoke(typeinfo, &This->IVBSAXAttributes_iface, dispIdMember, wFlags,
700 pDispParams, pVarResult, pExcepInfo, puArgErr);
Piotr Cabancafbf852008-07-31 16:47:41 +0200701 ITypeInfo_Release(typeinfo);
702 }
703
704 return hr;
705}
706
707/*** IVBSAXAttributes methods ***/
708static HRESULT WINAPI ivbsaxattributes_get_length(
709 IVBSAXAttributes* iface,
710 int *nLength)
711{
Piotr Caban1b462062011-10-31 12:32:39 +0100712 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100713 return ISAXAttributes_getLength(&This->ISAXAttributes_iface, nLength);
Piotr Cabancafbf852008-07-31 16:47:41 +0200714}
715
716static HRESULT WINAPI ivbsaxattributes_getURI(
717 IVBSAXAttributes* iface,
718 int nIndex,
719 BSTR *uri)
720{
721 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100722 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100723 return ISAXAttributes_getURI(&This->ISAXAttributes_iface, nIndex, (const WCHAR**)uri, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200724}
725
726static HRESULT WINAPI ivbsaxattributes_getLocalName(
727 IVBSAXAttributes* iface,
728 int nIndex,
729 BSTR *localName)
730{
731 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100732 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100733 return ISAXAttributes_getLocalName(&This->ISAXAttributes_iface, nIndex,
734 (const WCHAR**)localName, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200735}
736
737static HRESULT WINAPI ivbsaxattributes_getQName(
738 IVBSAXAttributes* iface,
739 int nIndex,
740 BSTR *QName)
741{
742 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100743 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100744 return ISAXAttributes_getQName(&This->ISAXAttributes_iface, nIndex, (const WCHAR**)QName, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200745}
746
747static HRESULT WINAPI ivbsaxattributes_getIndexFromName(
748 IVBSAXAttributes* iface,
749 BSTR uri,
750 BSTR localName,
751 int *index)
752{
Piotr Caban1b462062011-10-31 12:32:39 +0100753 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100754 return ISAXAttributes_getIndexFromName(&This->ISAXAttributes_iface, uri, SysStringLen(uri),
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +0100755 localName, SysStringLen(localName), index);
Piotr Cabancafbf852008-07-31 16:47:41 +0200756}
757
758static HRESULT WINAPI ivbsaxattributes_getIndexFromQName(
759 IVBSAXAttributes* iface,
760 BSTR QName,
761 int *index)
762{
Piotr Caban1b462062011-10-31 12:32:39 +0100763 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100764 return ISAXAttributes_getIndexFromQName(&This->ISAXAttributes_iface, QName,
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +0100765 SysStringLen(QName), index);
Piotr Cabancafbf852008-07-31 16:47:41 +0200766}
767
768static HRESULT WINAPI ivbsaxattributes_getType(
769 IVBSAXAttributes* iface,
770 int nIndex,
771 BSTR *type)
772{
773 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100774 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100775 return ISAXAttributes_getType(&This->ISAXAttributes_iface, nIndex, (const WCHAR**)type, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200776}
777
778static HRESULT WINAPI ivbsaxattributes_getTypeFromName(
779 IVBSAXAttributes* iface,
780 BSTR uri,
781 BSTR localName,
782 BSTR *type)
783{
784 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100785 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100786 return ISAXAttributes_getTypeFromName(&This->ISAXAttributes_iface, uri, SysStringLen(uri),
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +0100787 localName, SysStringLen(localName), (const WCHAR**)type, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200788}
789
790static HRESULT WINAPI ivbsaxattributes_getTypeFromQName(
791 IVBSAXAttributes* iface,
792 BSTR QName,
793 BSTR *type)
794{
795 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100796 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100797 return ISAXAttributes_getTypeFromQName(&This->ISAXAttributes_iface, QName, SysStringLen(QName),
798 (const WCHAR**)type, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200799}
800
801static HRESULT WINAPI ivbsaxattributes_getValue(
802 IVBSAXAttributes* iface,
803 int nIndex,
804 BSTR *value)
805{
806 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100807 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100808 return ISAXAttributes_getValue(&This->ISAXAttributes_iface, nIndex, (const WCHAR**)value, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200809}
810
811static HRESULT WINAPI ivbsaxattributes_getValueFromName(
812 IVBSAXAttributes* iface,
813 BSTR uri,
814 BSTR localName,
815 BSTR *value)
816{
817 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100818 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100819 return ISAXAttributes_getValueFromName(&This->ISAXAttributes_iface, uri, SysStringLen(uri),
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +0100820 localName, SysStringLen(localName), (const WCHAR**)value, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200821}
822
823static HRESULT WINAPI ivbsaxattributes_getValueFromQName(
824 IVBSAXAttributes* iface,
825 BSTR QName,
826 BSTR *value)
827{
828 int len;
Piotr Caban1b462062011-10-31 12:32:39 +0100829 saxlocator *This = impl_from_IVBSAXAttributes( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +0100830 return ISAXAttributes_getValueFromQName(&This->ISAXAttributes_iface, QName,
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +0100831 SysStringLen(QName), (const WCHAR**)value, &len);
Piotr Cabancafbf852008-07-31 16:47:41 +0200832}
833
834static const struct IVBSAXAttributesVtbl ivbsaxattributes_vtbl =
835{
836 ivbsaxattributes_QueryInterface,
837 ivbsaxattributes_AddRef,
838 ivbsaxattributes_Release,
839 ivbsaxattributes_GetTypeInfoCount,
840 ivbsaxattributes_GetTypeInfo,
841 ivbsaxattributes_GetIDsOfNames,
842 ivbsaxattributes_Invoke,
843 ivbsaxattributes_get_length,
844 ivbsaxattributes_getURI,
845 ivbsaxattributes_getLocalName,
846 ivbsaxattributes_getQName,
847 ivbsaxattributes_getIndexFromName,
848 ivbsaxattributes_getIndexFromQName,
849 ivbsaxattributes_getType,
850 ivbsaxattributes_getTypeFromName,
851 ivbsaxattributes_getTypeFromQName,
852 ivbsaxattributes_getValue,
853 ivbsaxattributes_getValueFromName,
854 ivbsaxattributes_getValueFromQName
855};
856
Piotr Caban5816c382008-07-27 19:55:05 +0200857/*** ISAXAttributes interface ***/
858/*** IUnknown methods ***/
859static HRESULT WINAPI isaxattributes_QueryInterface(
860 ISAXAttributes* iface,
861 REFIID riid,
862 void **ppvObject)
863{
Piotr Caban1b462062011-10-31 12:32:39 +0100864 saxlocator *This = impl_from_ISAXAttributes(iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200865 TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject);
Piotr Caban1b462062011-10-31 12:32:39 +0100866 return ISAXLocator_QueryInterface(&This->ISAXLocator_iface, riid, ppvObject);
Piotr Caban5816c382008-07-27 19:55:05 +0200867}
868
869static ULONG WINAPI isaxattributes_AddRef(ISAXAttributes* iface)
870{
Piotr Caban1b462062011-10-31 12:32:39 +0100871 saxlocator *This = impl_from_ISAXAttributes(iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200872 TRACE("%p\n", This);
Piotr Caban1b462062011-10-31 12:32:39 +0100873 return ISAXLocator_AddRef(&This->ISAXLocator_iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200874}
875
876static ULONG WINAPI isaxattributes_Release(ISAXAttributes* iface)
877{
Piotr Caban1b462062011-10-31 12:32:39 +0100878 saxlocator *This = impl_from_ISAXAttributes(iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200879
880 TRACE("%p\n", This);
Piotr Caban1b462062011-10-31 12:32:39 +0100881 return ISAXLocator_Release(&This->ISAXLocator_iface);
Piotr Caban5816c382008-07-27 19:55:05 +0200882}
883
884/*** ISAXAttributes methods ***/
885static HRESULT WINAPI isaxattributes_getLength(
886 ISAXAttributes* iface,
887 int *length)
888{
Piotr Caban1b462062011-10-31 12:32:39 +0100889 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban5816c382008-07-27 19:55:05 +0200890
Piotr Caban58b26f32008-07-30 20:26:50 +0200891 *length = This->nb_attributes;
892 TRACE("Length set to %d\n", *length);
893 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +0200894}
895
896static HRESULT WINAPI isaxattributes_getURI(
897 ISAXAttributes* iface,
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500898 int index,
899 const WCHAR **url,
900 int *size)
Piotr Caban5816c382008-07-27 19:55:05 +0200901{
Piotr Caban1b462062011-10-31 12:32:39 +0100902 saxlocator *This = impl_from_ISAXAttributes( iface );
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500903 TRACE("(%p)->(%d)\n", This, index);
Piotr Caban5816c382008-07-27 19:55:05 +0200904
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500905 if(index >= This->nb_attributes || index < 0) return E_INVALIDARG;
906 if(!url || !size) return E_POINTER;
Piotr Caban7be136b2008-08-03 14:37:52 +0200907
Piotr Caban1b462062011-10-31 12:32:39 +0100908 *size = SysStringLen(This->attributes[index].szURI);
909 *url = This->attributes[index].szURI;
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500910
Piotr Caban1b462062011-10-31 12:32:39 +0100911 TRACE("(%s:%d)\n", debugstr_w(This->attributes[index].szURI), *size);
Piotr Caban7be136b2008-08-03 14:37:52 +0200912
913 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +0200914}
915
916static HRESULT WINAPI isaxattributes_getLocalName(
917 ISAXAttributes* iface,
918 int nIndex,
919 const WCHAR **pLocalName,
920 int *pLocalNameLength)
921{
Piotr Caban1b462062011-10-31 12:32:39 +0100922 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban421a9a62008-07-30 20:27:07 +0200923 TRACE("(%p)->(%d)\n", This, nIndex);
Piotr Caban5816c382008-07-27 19:55:05 +0200924
Piotr Caban7a5a7812008-08-03 14:37:45 +0200925 if(nIndex>=This->nb_attributes || nIndex<0) return E_INVALIDARG;
926 if(!pLocalName || !pLocalNameLength) return E_POINTER;
Piotr Caban421a9a62008-07-30 20:27:07 +0200927
Piotr Caban1b462062011-10-31 12:32:39 +0100928 *pLocalNameLength = SysStringLen(This->attributes[nIndex].szLocalname);
929 *pLocalName = This->attributes[nIndex].szLocalname;
Piotr Caban421a9a62008-07-30 20:27:07 +0200930
931 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +0200932}
933
934static HRESULT WINAPI isaxattributes_getQName(
935 ISAXAttributes* iface,
936 int nIndex,
937 const WCHAR **pQName,
938 int *pQNameLength)
939{
Piotr Caban1b462062011-10-31 12:32:39 +0100940 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban6818cc62008-07-31 16:45:42 +0200941 TRACE("(%p)->(%d)\n", This, nIndex);
Piotr Caban5816c382008-07-27 19:55:05 +0200942
Piotr Caban7a5a7812008-08-03 14:37:45 +0200943 if(nIndex>=This->nb_attributes || nIndex<0) return E_INVALIDARG;
944 if(!pQName || !pQNameLength) return E_POINTER;
Piotr Caban6818cc62008-07-31 16:45:42 +0200945
Piotr Caban1b462062011-10-31 12:32:39 +0100946 *pQNameLength = SysStringLen(This->attributes[nIndex].szQName);
947 *pQName = This->attributes[nIndex].szQName;
Piotr Caban6818cc62008-07-31 16:45:42 +0200948
949 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +0200950}
951
952static HRESULT WINAPI isaxattributes_getName(
953 ISAXAttributes* iface,
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500954 int index,
955 const WCHAR **uri,
Piotr Caban5816c382008-07-27 19:55:05 +0200956 int *pUriLength,
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500957 const WCHAR **localName,
Piotr Caban5816c382008-07-27 19:55:05 +0200958 int *pLocalNameSize,
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500959 const WCHAR **QName,
Piotr Caban5816c382008-07-27 19:55:05 +0200960 int *pQNameLength)
961{
Piotr Caban1b462062011-10-31 12:32:39 +0100962 saxlocator *This = impl_from_ISAXAttributes( iface );
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500963 TRACE("(%p)->(%d)\n", This, index);
Piotr Caban5816c382008-07-27 19:55:05 +0200964
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500965 if(index>=This->nb_attributes || index<0) return E_INVALIDARG;
966 if(!uri || !pUriLength || !localName || !pLocalNameSize
967 || !QName || !pQNameLength) return E_POINTER;
Piotr Caban5e1af9b2008-08-20 18:30:16 +0200968
Piotr Caban1b462062011-10-31 12:32:39 +0100969 *pUriLength = SysStringLen(This->attributes[index].szURI);
970 *uri = This->attributes[index].szURI;
971 *pLocalNameSize = SysStringLen(This->attributes[index].szLocalname);
972 *localName = This->attributes[index].szLocalname;
973 *pQNameLength = SysStringLen(This->attributes[index].szQName);
974 *QName = This->attributes[index].szQName;
Nikolay Sivovd65f2982011-10-05 14:19:29 -0500975
976 TRACE("(%s, %s, %s)\n", debugstr_w(*uri), debugstr_w(*localName), debugstr_w(*QName));
Piotr Caban5e1af9b2008-08-20 18:30:16 +0200977
978 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +0200979}
980
981static HRESULT WINAPI isaxattributes_getIndexFromName(
982 ISAXAttributes* iface,
983 const WCHAR *pUri,
984 int cUriLength,
985 const WCHAR *pLocalName,
986 int cocalNameLength,
987 int *index)
988{
Piotr Caban1b462062011-10-31 12:32:39 +0100989 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban0afbcd52008-08-20 18:30:34 +0200990 int i;
991 TRACE("(%p)->(%s, %d, %s, %d)\n", This, debugstr_w(pUri), cUriLength,
Piotr Caban5816c382008-07-27 19:55:05 +0200992 debugstr_w(pLocalName), cocalNameLength);
Piotr Caban0afbcd52008-08-20 18:30:34 +0200993
994 if(!pUri || !pLocalName || !index) return E_POINTER;
995
996 for(i=0; i<This->nb_attributes; i++)
997 {
Piotr Caban1b462062011-10-31 12:32:39 +0100998 if(cUriLength!=SysStringLen(This->attributes[i].szURI)
999 || cocalNameLength!=SysStringLen(This->attributes[i].szLocalname))
Piotr Caban0afbcd52008-08-20 18:30:34 +02001000 continue;
Piotr Caban1b462062011-10-31 12:32:39 +01001001 if(cUriLength && memcmp(pUri, This->attributes[i].szURI,
Piotr Caban0afbcd52008-08-20 18:30:34 +02001002 sizeof(WCHAR)*cUriLength))
1003 continue;
Piotr Caban1b462062011-10-31 12:32:39 +01001004 if(cocalNameLength && memcmp(pLocalName, This->attributes[i].szLocalname,
Piotr Caban0afbcd52008-08-20 18:30:34 +02001005 sizeof(WCHAR)*cocalNameLength))
1006 continue;
1007
1008 *index = i;
1009 return S_OK;
1010 }
1011
1012 return E_INVALIDARG;
Piotr Caban5816c382008-07-27 19:55:05 +02001013}
1014
1015static HRESULT WINAPI isaxattributes_getIndexFromQName(
1016 ISAXAttributes* iface,
1017 const WCHAR *pQName,
1018 int nQNameLength,
1019 int *index)
1020{
Piotr Caban1b462062011-10-31 12:32:39 +01001021 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Cabanfa221f52008-08-20 18:30:58 +02001022 int i;
1023 TRACE("(%p)->(%s, %d)\n", This, debugstr_w(pQName), nQNameLength);
Piotr Caban5816c382008-07-27 19:55:05 +02001024
Piotr Cabanfa221f52008-08-20 18:30:58 +02001025 if(!pQName || !index) return E_POINTER;
1026 if(!nQNameLength) return E_INVALIDARG;
1027
1028 for(i=0; i<This->nb_attributes; i++)
1029 {
Piotr Caban1b462062011-10-31 12:32:39 +01001030 if(nQNameLength!=SysStringLen(This->attributes[i].szQName)) continue;
1031 if(memcmp(pQName, This->attributes[i].szQName, sizeof(WCHAR)*nQNameLength)) continue;
Piotr Cabanfa221f52008-08-20 18:30:58 +02001032
1033 *index = i;
1034 return S_OK;
1035 }
1036
1037 return E_INVALIDARG;
Piotr Caban5816c382008-07-27 19:55:05 +02001038}
1039
1040static HRESULT WINAPI isaxattributes_getType(
1041 ISAXAttributes* iface,
1042 int nIndex,
1043 const WCHAR **pType,
1044 int *pTypeLength)
1045{
Piotr Caban1b462062011-10-31 12:32:39 +01001046 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban5816c382008-07-27 19:55:05 +02001047
1048 FIXME("(%p)->(%d) stub\n", This, nIndex);
1049 return E_NOTIMPL;
1050}
1051
1052static HRESULT WINAPI isaxattributes_getTypeFromName(
1053 ISAXAttributes* iface,
1054 const WCHAR *pUri,
1055 int nUri,
1056 const WCHAR *pLocalName,
1057 int nLocalName,
1058 const WCHAR **pType,
1059 int *nType)
1060{
Piotr Caban1b462062011-10-31 12:32:39 +01001061 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban5816c382008-07-27 19:55:05 +02001062
1063 FIXME("(%p)->(%s, %d, %s, %d) stub\n", This, debugstr_w(pUri), nUri,
1064 debugstr_w(pLocalName), nLocalName);
1065 return E_NOTIMPL;
1066}
1067
1068static HRESULT WINAPI isaxattributes_getTypeFromQName(
1069 ISAXAttributes* iface,
1070 const WCHAR *pQName,
1071 int nQName,
1072 const WCHAR **pType,
1073 int *nType)
1074{
Piotr Caban1b462062011-10-31 12:32:39 +01001075 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban5816c382008-07-27 19:55:05 +02001076
1077 FIXME("(%p)->(%s, %d) stub\n", This, debugstr_w(pQName), nQName);
1078 return E_NOTIMPL;
1079}
1080
1081static HRESULT WINAPI isaxattributes_getValue(
1082 ISAXAttributes* iface,
Nikolay Sivovd65f2982011-10-05 14:19:29 -05001083 int index,
1084 const WCHAR **value,
Piotr Caban5816c382008-07-27 19:55:05 +02001085 int *nValue)
1086{
Piotr Caban1b462062011-10-31 12:32:39 +01001087 saxlocator *This = impl_from_ISAXAttributes( iface );
Nikolay Sivovd65f2982011-10-05 14:19:29 -05001088 TRACE("(%p)->(%d)\n", This, index);
Piotr Caban5816c382008-07-27 19:55:05 +02001089
Nikolay Sivovd65f2982011-10-05 14:19:29 -05001090 if(index>=This->nb_attributes || index<0) return E_INVALIDARG;
1091 if(!value || !nValue) return E_POINTER;
Piotr Caban25168402008-07-30 20:27:25 +02001092
Piotr Caban1b462062011-10-31 12:32:39 +01001093 *nValue = SysStringLen(This->attributes[index].szValue);
1094 *value = This->attributes[index].szValue;
Nikolay Sivovd65f2982011-10-05 14:19:29 -05001095
1096 TRACE("(%s:%d)\n", debugstr_w(*value), *nValue);
Piotr Caban25168402008-07-30 20:27:25 +02001097
1098 return S_OK;
Piotr Caban5816c382008-07-27 19:55:05 +02001099}
1100
1101static HRESULT WINAPI isaxattributes_getValueFromName(
1102 ISAXAttributes* iface,
1103 const WCHAR *pUri,
1104 int nUri,
1105 const WCHAR *pLocalName,
1106 int nLocalName,
1107 const WCHAR **pValue,
1108 int *nValue)
1109{
Piotr Cabanc9893d12008-08-20 18:31:17 +02001110 HRESULT hr;
1111 int index;
Piotr Caban1b462062011-10-31 12:32:39 +01001112 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Cabanc9893d12008-08-20 18:31:17 +02001113 TRACE("(%p)->(%s, %d, %s, %d)\n", This, debugstr_w(pUri), nUri,
Piotr Caban5816c382008-07-27 19:55:05 +02001114 debugstr_w(pLocalName), nLocalName);
Piotr Cabanc9893d12008-08-20 18:31:17 +02001115
1116 hr = ISAXAttributes_getIndexFromName(iface,
1117 pUri, nUri, pLocalName, nLocalName, &index);
1118 if(hr==S_OK) hr = ISAXAttributes_getValue(iface, index, pValue, nValue);
1119
1120 return hr;
Piotr Caban5816c382008-07-27 19:55:05 +02001121}
1122
1123static HRESULT WINAPI isaxattributes_getValueFromQName(
1124 ISAXAttributes* iface,
1125 const WCHAR *pQName,
1126 int nQName,
1127 const WCHAR **pValue,
1128 int *nValue)
1129{
Piotr Caban655155f2008-08-20 18:31:39 +02001130 HRESULT hr;
1131 int index;
Piotr Caban1b462062011-10-31 12:32:39 +01001132 saxlocator *This = impl_from_ISAXAttributes( iface );
Piotr Caban655155f2008-08-20 18:31:39 +02001133 TRACE("(%p)->(%s, %d)\n", This, debugstr_w(pQName), nQName);
Piotr Caban5816c382008-07-27 19:55:05 +02001134
Piotr Caban655155f2008-08-20 18:31:39 +02001135 hr = ISAXAttributes_getIndexFromQName(iface, pQName, nQName, &index);
1136 if(hr==S_OK) hr = ISAXAttributes_getValue(iface, index, pValue, nValue);
1137
1138 return hr;
Piotr Caban5816c382008-07-27 19:55:05 +02001139}
1140
1141static const struct ISAXAttributesVtbl isaxattributes_vtbl =
1142{
1143 isaxattributes_QueryInterface,
1144 isaxattributes_AddRef,
1145 isaxattributes_Release,
1146 isaxattributes_getLength,
1147 isaxattributes_getURI,
1148 isaxattributes_getLocalName,
1149 isaxattributes_getQName,
1150 isaxattributes_getName,
1151 isaxattributes_getIndexFromName,
1152 isaxattributes_getIndexFromQName,
1153 isaxattributes_getType,
1154 isaxattributes_getTypeFromName,
1155 isaxattributes_getTypeFromQName,
1156 isaxattributes_getValue,
1157 isaxattributes_getValueFromName,
1158 isaxattributes_getValueFromQName
1159};
1160
Piotr Caban1b462062011-10-31 12:32:39 +01001161static HRESULT SAXAttributes_populate(saxlocator *locator,
Piotr Cabana94e4c12008-08-03 14:38:10 +02001162 int nb_namespaces, const xmlChar **xmlNamespaces,
Piotr Caban6df49d52008-07-30 20:24:12 +02001163 int nb_attributes, const xmlChar **xmlAttributes)
Piotr Caban5816c382008-07-27 19:55:05 +02001164{
Piotr Cabana94e4c12008-08-03 14:38:10 +02001165 static const xmlChar xmlns[] = "xmlns";
Piotr Caban685be402011-10-26 13:25:27 +02001166 static const WCHAR xmlnsW[] = { 'x','m','l','n','s',0 };
Piotr Caban5816c382008-07-27 19:55:05 +02001167
Piotr Caban1b462062011-10-31 12:32:39 +01001168 struct _attributes *attrs;
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001169 int i;
Piotr Caban5816c382008-07-27 19:55:05 +02001170
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001171 /* skip namespace definitions */
1172 if ((locator->saxreader->features & NamespacePrefixes) == 0)
1173 nb_namespaces = 0;
1174
1175 locator->nb_attributes = nb_namespaces + nb_attributes;
Piotr Caban1b462062011-10-31 12:32:39 +01001176 if(locator->nb_attributes > locator->attributesSize)
Piotr Caban6df49d52008-07-30 20:24:12 +02001177 {
Piotr Caban1b462062011-10-31 12:32:39 +01001178 attrs = heap_realloc(locator->attributes, sizeof(struct _attributes)*locator->nb_attributes*2);
1179 if(!attrs)
1180 {
1181 locator->nb_attributes = 0;
1182 return E_OUTOFMEMORY;
1183 }
1184 locator->attributes = attrs;
1185 }
1186 else
1187 {
1188 attrs = locator->attributes;
Piotr Caban6df49d52008-07-30 20:24:12 +02001189 }
1190
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001191 for (i = 0; i < nb_namespaces; i++)
Piotr Cabana94e4c12008-08-03 14:38:10 +02001192 {
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001193 attrs[nb_attributes+i].szLocalname = SysAllocStringLen(NULL, 0);
1194 attrs[nb_attributes+i].szURI = locator->namespaceUri;
1195 attrs[nb_attributes+i].szValue = bstr_from_xmlChar(xmlNamespaces[2*i+1]);
1196 if(!xmlNamespaces[2*i])
1197 attrs[nb_attributes+i].szQName = SysAllocString(xmlnsW);
Piotr Caban685be402011-10-26 13:25:27 +02001198 else
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001199 attrs[nb_attributes+i].szQName = QName_from_xmlChar(xmlns, xmlNamespaces[2*i]);
Piotr Cabana94e4c12008-08-03 14:38:10 +02001200 }
1201
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001202 for (i = 0; i < nb_attributes; i++)
Piotr Caban6df49d52008-07-30 20:24:12 +02001203 {
Nikolay Sivov973f62f2012-03-01 21:06:39 +03001204 static const xmlChar xmlA[] = "xml";
1205
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001206 if (xmlStrEqual(xmlAttributes[i*5+1], xmlA))
1207 attrs[i].szURI = bstr_from_xmlChar(xmlAttributes[i*5+2]);
Nikolay Sivov973f62f2012-03-01 21:06:39 +03001208 else
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001209 attrs[i].szURI = find_element_uri(locator, xmlAttributes[i*5+2]);
Nikolay Sivov973f62f2012-03-01 21:06:39 +03001210
Nikolay Sivov4fb58722012-04-20 11:03:18 +04001211 attrs[i].szLocalname = bstr_from_xmlChar(xmlAttributes[i*5]);
1212 attrs[i].szValue = bstr_from_xmlCharN(xmlAttributes[i*5+3],
1213 xmlAttributes[i*5+4]-xmlAttributes[i*5+3]);
1214 attrs[i].szQName = QName_from_xmlChar(xmlAttributes[i*5+1],
1215 xmlAttributes[i*5]);
Piotr Caban6df49d52008-07-30 20:24:12 +02001216 }
1217
Piotr Caban5816c382008-07-27 19:55:05 +02001218 return S_OK;
1219}
1220
Piotr Caban072383e2008-07-17 00:41:01 +02001221/*** LibXML callbacks ***/
1222static void libxmlStartDocument(void *ctx)
1223{
1224 saxlocator *This = ctx;
1225 HRESULT hr;
1226
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001227 if (This->saxreader->version >= MSXML4)
Piotr Cabanc253bab2011-11-28 15:05:25 +01001228 {
1229 const xmlChar *p = This->pParserCtxt->input->cur-1;
1230 update_position(This, FALSE);
1231 while(p>This->pParserCtxt->input->base && *p!='>')
1232 {
1233 if(*p=='\n' || (*p=='\r' && *(p+1)!='\n'))
1234 This->line--;
1235 p--;
1236 }
1237 This->column = 0;
1238 for(; p>=This->pParserCtxt->input->base && *p!='\n' && *p!='\r'; p--)
1239 This->column++;
Piotr Cabane84c4102011-10-24 20:08:46 +02001240 }
1241
Nikolay Sivovff992282010-01-18 23:30:50 +03001242 if(has_content_handler(This))
Piotr Caban072383e2008-07-17 00:41:01 +02001243 {
Piotr Cabanc52e0912008-07-31 16:48:03 +02001244 if(This->vbInterface)
1245 hr = IVBSAXContentHandler_startDocument(This->saxreader->vbcontentHandler);
1246 else
1247 hr = ISAXContentHandler_startDocument(This->saxreader->contentHandler);
1248
Nikolay Sivov9d662922011-12-24 16:10:33 +03001249 if (sax_callback_failed(This, hr))
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001250 format_error_message_from_id(This, hr);
Piotr Caban072383e2008-07-17 00:41:01 +02001251 }
1252}
1253
Piotr Cabanc58b24b2008-07-17 00:41:19 +02001254static void libxmlEndDocument(void *ctx)
1255{
1256 saxlocator *This = ctx;
1257 HRESULT hr;
1258
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001259 if (This->saxreader->version >= MSXML4) {
Piotr Cabanc253bab2011-11-28 15:05:25 +01001260 update_position(This, FALSE);
Piotr Cabane84c4102011-10-24 20:08:46 +02001261 if(This->column > 1)
1262 This->line++;
1263 This->column = 0;
1264 } else {
1265 This->column = 0;
1266 This->line = 0;
1267 }
Piotr Cabanc58b24b2008-07-17 00:41:19 +02001268
Piotr Caban2cb215b2008-07-19 22:31:56 +02001269 if(This->ret != S_OK) return;
1270
Nikolay Sivovff992282010-01-18 23:30:50 +03001271 if(has_content_handler(This))
Piotr Cabanc58b24b2008-07-17 00:41:19 +02001272 {
Piotr Cabanc52e0912008-07-31 16:48:03 +02001273 if(This->vbInterface)
1274 hr = IVBSAXContentHandler_endDocument(This->saxreader->vbcontentHandler);
1275 else
1276 hr = ISAXContentHandler_endDocument(This->saxreader->contentHandler);
1277
Nikolay Sivov9d662922011-12-24 16:10:33 +03001278 if (sax_callback_failed(This, hr))
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001279 format_error_message_from_id(This, hr);
Piotr Cabanc58b24b2008-07-17 00:41:19 +02001280 }
1281}
1282
Piotr Caban965a2ea2008-07-17 00:41:27 +02001283static void libxmlStartElementNS(
1284 void *ctx,
1285 const xmlChar *localname,
1286 const xmlChar *prefix,
1287 const xmlChar *URI,
1288 int nb_namespaces,
1289 const xmlChar **namespaces,
1290 int nb_attributes,
1291 int nb_defaulted,
1292 const xmlChar **attributes)
1293{
Piotr Caban965a2ea2008-07-17 00:41:27 +02001294 saxlocator *This = ctx;
Nikolay Sivov9d662922011-12-24 16:10:33 +03001295 element_entry *element;
1296 HRESULT hr = S_OK;
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001297 BSTR uri;
Piotr Caban965a2ea2008-07-17 00:41:27 +02001298
Piotr Cabanc253bab2011-11-28 15:05:25 +01001299 update_position(This, TRUE);
Piotr Cabande91f262011-10-31 12:32:22 +01001300 if(*(This->pParserCtxt->input->cur) == '/')
Piotr Cabanc253bab2011-11-28 15:05:25 +01001301 This->column++;
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001302 if(This->saxreader->version < MSXML4)
Piotr Cabanc253bab2011-11-28 15:05:25 +01001303 This->column++;
Piotr Caban965a2ea2008-07-17 00:41:27 +02001304
Nikolay Sivov9d662922011-12-24 16:10:33 +03001305 element = alloc_element_entry(localname, prefix, nb_namespaces, namespaces);
1306 push_element_ns(This, element);
Piotr Caban619c2dc2011-10-31 12:32:59 +01001307
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001308 if (is_namespaces_enabled(This->saxreader))
Piotr Caban965a2ea2008-07-17 00:41:27 +02001309 {
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001310 int i;
Nikolay Sivov9d662922011-12-24 16:10:33 +03001311
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001312 for (i = 0; i < nb_namespaces && has_content_handler(This); i++)
Piotr Cabane9d34b72008-08-03 17:03:28 +02001313 {
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001314 if (This->vbInterface)
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001315 hr = IVBSAXContentHandler_startPrefixMapping(
1316 This->saxreader->vbcontentHandler,
1317 &element->ns[i].prefix,
1318 &element->ns[i].uri);
Piotr Cabanc52e0912008-07-31 16:48:03 +02001319 else
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001320 hr = ISAXContentHandler_startPrefixMapping(
1321 This->saxreader->contentHandler,
1322 element->ns[i].prefix,
1323 SysStringLen(element->ns[i].prefix),
1324 element->ns[i].uri,
1325 SysStringLen(element->ns[i].uri));
1326
1327 if (sax_callback_failed(This, hr))
1328 {
1329 format_error_message_from_id(This, hr);
1330 return;
1331 }
Piotr Caban6df49d52008-07-30 20:24:12 +02001332 }
Piotr Caban965a2ea2008-07-17 00:41:27 +02001333 }
Piotr Caban03f79092008-08-20 18:20:50 +02001334
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001335 uri = find_element_uri(This, URI);
1336 hr = SAXAttributes_populate(This, nb_namespaces, namespaces, nb_attributes, attributes);
1337 if (hr == S_OK && has_content_handler(This))
1338 {
1339 BSTR local;
1340
1341 if (is_namespaces_enabled(This->saxreader))
1342 local = element->local;
1343 else
1344 uri = local = NULL;
1345
1346 if (This->vbInterface)
1347 hr = IVBSAXContentHandler_startElement(This->saxreader->vbcontentHandler,
1348 &uri, &local, &element->qname, &This->IVBSAXAttributes_iface);
1349 else
1350 hr = ISAXContentHandler_startElement(This->saxreader->contentHandler,
1351 uri, SysStringLen(uri),
1352 local, SysStringLen(local),
1353 element->qname, SysStringLen(element->qname),
1354 &This->ISAXAttributes_iface);
1355
1356 if (sax_callback_failed(This, hr))
1357 format_error_message_from_id(This, hr);
1358 }
Piotr Caban965a2ea2008-07-17 00:41:27 +02001359}
1360
Piotr Caban66932632008-07-17 00:41:34 +02001361static void libxmlEndElementNS(
1362 void *ctx,
1363 const xmlChar *localname,
1364 const xmlChar *prefix,
1365 const xmlChar *URI)
1366{
Piotr Caban66932632008-07-17 00:41:34 +02001367 saxlocator *This = ctx;
Nikolay Sivov9d662922011-12-24 16:10:33 +03001368 element_entry *element;
Piotr Cabanc253bab2011-11-28 15:05:25 +01001369 const xmlChar *p;
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001370 BSTR uri, local;
Nikolay Sivov9d662922011-12-24 16:10:33 +03001371 HRESULT hr;
Piotr Caban66932632008-07-17 00:41:34 +02001372
Piotr Cabanc253bab2011-11-28 15:05:25 +01001373 update_position(This, FALSE);
1374 p = This->pParserCtxt->input->cur;
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001375
1376 if (This->saxreader->version >= MSXML4)
Piotr Cabanc253bab2011-11-28 15:05:25 +01001377 {
1378 p--;
1379 while(p>This->pParserCtxt->input->base && *p!='>')
1380 {
1381 if(*p=='\n' || (*p=='\r' && *(p+1)!='\n'))
1382 This->line--;
1383 p--;
1384 }
Piotr Cabane84c4102011-10-24 20:08:46 +02001385 }
Piotr Cabanc253bab2011-11-28 15:05:25 +01001386 else if(*(p-1)!='>' || *(p-2)!='/')
1387 {
1388 p--;
1389 while(p-2>=This->pParserCtxt->input->base
1390 && *(p-2)!='<' && *(p-1)!='/')
1391 {
1392 if(*p=='\n' || (*p=='\r' && *(p+1)!='\n'))
1393 This->line--;
1394 p--;
1395 }
1396 }
1397 This->column = 0;
1398 for(; p>=This->pParserCtxt->input->base && *p!='\n' && *p!='\r'; p--)
1399 This->column++;
Piotr Caban66932632008-07-17 00:41:34 +02001400
Nikolay Sivov9d662922011-12-24 16:10:33 +03001401 uri = find_element_uri(This, URI);
1402 element = pop_element_ns(This);
1403
1404 if (!has_content_handler(This))
Piotr Caban66932632008-07-17 00:41:34 +02001405 {
Piotr Caban1b462062011-10-31 12:32:39 +01001406 This->nb_attributes = 0;
Nikolay Sivov9d662922011-12-24 16:10:33 +03001407 free_element_entry(element);
1408 return;
Piotr Caban66932632008-07-17 00:41:34 +02001409 }
Nikolay Sivov9d662922011-12-24 16:10:33 +03001410
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001411 if (is_namespaces_enabled(This->saxreader))
1412 local = element->local;
1413 else
1414 uri = local = NULL;
1415
1416 if (This->vbInterface)
Nikolay Sivov9d662922011-12-24 16:10:33 +03001417 hr = IVBSAXContentHandler_endElement(
1418 This->saxreader->vbcontentHandler,
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001419 &uri, &local, &element->qname);
Piotr Caban1b462062011-10-31 12:32:39 +01001420 else
Nikolay Sivov9d662922011-12-24 16:10:33 +03001421 hr = ISAXContentHandler_endElement(
1422 This->saxreader->contentHandler,
1423 uri, SysStringLen(uri),
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001424 local, SysStringLen(local),
Nikolay Sivov9d662922011-12-24 16:10:33 +03001425 element->qname, SysStringLen(element->qname));
1426
1427 This->nb_attributes = 0;
1428
1429 if (sax_callback_failed(This, hr))
Piotr Caban1b462062011-10-31 12:32:39 +01001430 {
Nikolay Sivov9d662922011-12-24 16:10:33 +03001431 format_error_message_from_id(This, hr);
Nikolay Sivovf39209c2012-02-05 22:44:44 +03001432 free_element_entry(element);
Nikolay Sivov9d662922011-12-24 16:10:33 +03001433 return;
Piotr Caban1b462062011-10-31 12:32:39 +01001434 }
Nikolay Sivov9d662922011-12-24 16:10:33 +03001435
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001436 if (is_namespaces_enabled(This->saxreader))
Nikolay Sivov9d662922011-12-24 16:10:33 +03001437 {
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001438 int i = -1;
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001439 while (iterate_endprefix_index(This, element, &i) && has_content_handler(This))
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001440 {
1441 if (This->vbInterface)
1442 hr = IVBSAXContentHandler_endPrefixMapping(
1443 This->saxreader->vbcontentHandler, &element->ns[i].prefix);
1444 else
1445 hr = ISAXContentHandler_endPrefixMapping(
1446 This->saxreader->contentHandler,
1447 element->ns[i].prefix, SysStringLen(element->ns[i].prefix));
Nikolay Sivov9d662922011-12-24 16:10:33 +03001448
Nikolay Sivovf27159a2012-04-19 10:46:03 +04001449 if (sax_callback_failed(This, hr)) break;
1450 }
Nikolay Sivov9d662922011-12-24 16:10:33 +03001451
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001452 if (sax_callback_failed(This, hr))
1453 format_error_message_from_id(This, hr);
1454 }
Nikolay Sivov9d662922011-12-24 16:10:33 +03001455
1456 free_element_entry(element);
Piotr Caban66932632008-07-17 00:41:34 +02001457}
1458
Piotr Caban40388632008-07-17 00:41:40 +02001459static void libxmlCharacters(
1460 void *ctx,
1461 const xmlChar *ch,
1462 int len)
1463{
Piotr Caban40388632008-07-17 00:41:40 +02001464 saxlocator *This = ctx;
Piotr Cabanc801c182008-10-07 22:10:43 +02001465 BSTR Chars;
Piotr Caban40388632008-07-17 00:41:40 +02001466 HRESULT hr;
Piotr Cabanc253bab2011-11-28 15:05:25 +01001467 xmlChar *cur, *end;
Piotr Cabanc801c182008-10-07 22:10:43 +02001468 BOOL lastEvent = FALSE;
Piotr Caban3d4ad642008-07-23 16:39:12 +02001469
Nikolay Sivovff992282010-01-18 23:30:50 +03001470 if(!(has_content_handler(This))) return;
Piotr Caban40388632008-07-17 00:41:40 +02001471
Piotr Cabanc253bab2011-11-28 15:05:25 +01001472 update_position(This, FALSE);
1473 cur = (xmlChar*)This->pParserCtxt->input->cur;
1474 while(cur>=This->pParserCtxt->input->base && *cur!='>')
1475 {
1476 if(*cur=='\n' || (*cur=='\r' && *(cur+1)!='\n'))
1477 This->line--;
1478 cur--;
1479 }
1480 This->column = 1;
1481 for(; cur>=This->pParserCtxt->input->base && *cur!='\n' && *cur!='\r'; cur--)
1482 This->column++;
1483
Piotr Cabanc801c182008-10-07 22:10:43 +02001484 cur = (xmlChar*)ch;
1485 if(*(ch-1)=='\r') cur--;
1486 end = cur;
1487
Piotr Cabanc801c182008-10-07 22:10:43 +02001488 while(1)
Piotr Caban40388632008-07-17 00:41:40 +02001489 {
Piotr Cabanc801c182008-10-07 22:10:43 +02001490 while(end-ch<len && *end!='\r') end++;
1491 if(end-ch==len)
Piotr Caban3d4ad642008-07-23 16:39:12 +02001492 {
Piotr Cabanc801c182008-10-07 22:10:43 +02001493 lastEvent = TRUE;
Piotr Caban3d4ad642008-07-23 16:39:12 +02001494 }
Piotr Cabanc253bab2011-11-28 15:05:25 +01001495 else
1496 {
1497 *end = '\n';
1498 end++;
1499 }
Piotr Caban3d4ad642008-07-23 16:39:12 +02001500
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001501 if (This->saxreader->version >= MSXML4)
Piotr Cabanc253bab2011-11-28 15:05:25 +01001502 {
1503 xmlChar *p;
Piotr Cabanc801c182008-10-07 22:10:43 +02001504
Piotr Cabanc253bab2011-11-28 15:05:25 +01001505 for(p=cur; p!=end; p++)
1506 {
1507 if(*p=='\n')
1508 {
1509 This->line++;
1510 This->column = 1;
1511 }
1512 else
1513 {
1514 This->column++;
1515 }
1516 }
Piotr Cabane84c4102011-10-24 20:08:46 +02001517
1518 if(!lastEvent)
1519 This->column = 0;
1520 }
1521
Piotr Cabanc253bab2011-11-28 15:05:25 +01001522 Chars = pooled_bstr_from_xmlCharN(&This->saxreader->pool, cur, end-cur);
Piotr Cabanc801c182008-10-07 22:10:43 +02001523 if(This->vbInterface)
1524 hr = IVBSAXContentHandler_characters(
1525 This->saxreader->vbcontentHandler, &Chars);
1526 else
1527 hr = ISAXContentHandler_characters(
1528 This->saxreader->contentHandler,
1529 Chars, SysStringLen(Chars));
Piotr Cabanc801c182008-10-07 22:10:43 +02001530
Nikolay Sivov9d662922011-12-24 16:10:33 +03001531 if (sax_callback_failed(This, hr))
Ricardo Filipee5e301b2008-11-13 13:33:27 +00001532 {
1533 format_error_message_from_id(This, hr);
1534 return;
1535 }
1536
Nikolay Sivovbcd85852012-04-15 13:47:30 +04001537 if (This->saxreader->version < MSXML4)
Piotr Cabanc253bab2011-11-28 15:05:25 +01001538 This->column += end-cur;
Piotr Cabanc801c182008-10-07 22:10:43 +02001539
1540 if(lastEvent)
1541 break;
1542
Piotr Cabanc253bab2011-11-28 15:05:25 +01001543 *(end-1) = '\r';
Piotr Cabanc801c182008-10-07 22:10:43 +02001544 if(*end == '\n')
1545 {
1546 end++;
1547 This->column++;
1548 }
1549 cur = end;
1550
1551 if(end-ch == len) break;
Piotr Caban40388632008-07-17 00:41:40 +02001552 }
1553}
1554
Piotr Cabane47cbd12008-07-17 00:41:52 +02001555static void libxmlSetDocumentLocator(
1556 void *ctx,
1557 xmlSAXLocatorPtr loc)
1558{
1559 saxlocator *This = ctx;
Nikolay Sivov843bd252010-10-29 01:00:44 +04001560 HRESULT hr = S_OK;
Piotr Cabane47cbd12008-07-17 00:41:52 +02001561
Nikolay Sivov843bd252010-10-29 01:00:44 +04001562 if(has_content_handler(This))
1563 {
1564 if(This->vbInterface)
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001565 hr = IVBSAXContentHandler_putref_documentLocator(This->saxreader->vbcontentHandler,
1566 &This->IVBSAXLocator_iface);
Nikolay Sivov843bd252010-10-29 01:00:44 +04001567 else
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001568 hr = ISAXContentHandler_putDocumentLocator(This->saxreader->contentHandler,
1569 &This->ISAXLocator_iface);
Nikolay Sivov843bd252010-10-29 01:00:44 +04001570 }
Piotr Cabane47cbd12008-07-17 00:41:52 +02001571
1572 if(FAILED(hr))
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001573 format_error_message_from_id(This, hr);
1574}
1575
Piotr Cabandb2761a2008-10-07 22:11:04 +02001576static void libxmlComment(void *ctx, const xmlChar *value)
1577{
1578 saxlocator *This = ctx;
1579 BSTR bValue;
1580 HRESULT hr;
Piotr Cabanc253bab2011-11-28 15:05:25 +01001581 const xmlChar *p = This->pParserCtxt->input->cur;
Piotr Cabandb2761a2008-10-07 22:11:04 +02001582
Piotr Cabanc253bab2011-11-28 15:05:25 +01001583 update_position(This, FALSE);
1584 while(p-4>=This->pParserCtxt->input->base
1585 && memcmp(p-4, "<!--", sizeof(char[4])))
1586 {
1587 if(*p=='\n' || (*p=='\r' && *(p+1)!='\n'))
1588 This->line--;
1589 p--;
1590 }
Nikolay Sivov2dc67112012-05-08 15:50:19 +04001591
Piotr Cabanc253bab2011-11-28 15:05:25 +01001592 This->column = 0;
1593 for(; p>=This->pParserCtxt->input->base && *p!='\n' && *p!='\r'; p--)
1594 This->column++;
Piotr Cabandb2761a2008-10-07 22:11:04 +02001595
Nikolay Sivovfc4dab42012-05-08 15:56:20 +04001596 if (!has_lexical_handler(This)) return;
Piotr Cabandb2761a2008-10-07 22:11:04 +02001597
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -05001598 bValue = pooled_bstr_from_xmlChar(&This->saxreader->pool, value);
Piotr Cabandb2761a2008-10-07 22:11:04 +02001599
1600 if(This->vbInterface)
1601 hr = IVBSAXLexicalHandler_comment(
1602 This->saxreader->vblexicalHandler, &bValue);
1603 else
1604 hr = ISAXLexicalHandler_comment(
1605 This->saxreader->lexicalHandler,
1606 bValue, SysStringLen(bValue));
1607
Piotr Cabandb2761a2008-10-07 22:11:04 +02001608 if(FAILED(hr))
1609 format_error_message_from_id(This, hr);
Piotr Cabandb2761a2008-10-07 22:11:04 +02001610}
1611
Piotr Cabanc801c182008-10-07 22:10:43 +02001612static void libxmlFatalError(void *ctx, const char *msg, ...)
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001613{
1614 saxlocator *This = ctx;
1615 char message[1024];
Nikolay Sivov9a3bfce2010-10-27 01:25:26 +04001616 WCHAR *error;
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001617 DWORD len;
1618 va_list args;
1619
Piotr Caban6927a112011-10-24 12:55:13 +02001620 if(This->ret != S_OK) {
1621 xmlStopParser(This->pParserCtxt);
Piotr Cabanf8112cc2011-10-24 12:54:47 +02001622 return;
Piotr Caban6927a112011-10-24 12:55:13 +02001623 }
Piotr Cabanf8112cc2011-10-24 12:54:47 +02001624
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001625 va_start(args, msg);
1626 vsprintf(message, msg, args);
1627 va_end(args);
1628
Rob Shearmanfd75c212008-08-26 20:06:09 +01001629 len = MultiByteToWideChar(CP_UNIXCP, 0, message, -1, NULL, 0);
Nikolay Sivov9a3bfce2010-10-27 01:25:26 +04001630 error = heap_alloc(sizeof(WCHAR)*len);
1631 if(error)
1632 {
1633 MultiByteToWideChar(CP_UNIXCP, 0, message, -1, error, len);
1634 TRACE("fatal error for %p: %s\n", This, debugstr_w(error));
1635 }
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001636
Nikolay Sivovb1aaa862010-10-29 02:06:33 +04001637 if(!has_error_handler(This))
1638 {
1639 xmlStopParser(This->pParserCtxt);
1640 This->ret = E_FAIL;
1641 heap_free(error);
1642 return;
1643 }
1644
1645 FIXME("Error handling is not compatible.\n");
1646
Piotr Cabanc52e0912008-07-31 16:48:03 +02001647 if(This->vbInterface)
1648 {
Nikolay Sivov9a3bfce2010-10-27 01:25:26 +04001649 BSTR bstrError = SysAllocString(error);
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001650 IVBSAXErrorHandler_fatalError(This->saxreader->vberrorHandler, &This->IVBSAXLocator_iface,
1651 &bstrError, E_FAIL);
Nikolay Sivovbd7787c2010-10-27 01:15:38 +04001652 SysFreeString(bstrError);
Piotr Cabanc52e0912008-07-31 16:48:03 +02001653 }
1654 else
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001655 ISAXErrorHandler_fatalError(This->saxreader->errorHandler, &This->ISAXLocator_iface,
1656 error, E_FAIL);
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001657
Nikolay Sivov9a3bfce2010-10-27 01:25:26 +04001658 heap_free(error);
Piotr Cabanbba7eb52008-07-19 22:32:10 +02001659
1660 xmlStopParser(This->pParserCtxt);
1661 This->ret = E_FAIL;
Piotr Cabane47cbd12008-07-17 00:41:52 +02001662}
1663
Piotr Cabanb84e4782008-10-07 22:11:11 +02001664static void libxmlCDataBlock(void *ctx, const xmlChar *value, int len)
1665{
1666 saxlocator *This = ctx;
1667 HRESULT hr = S_OK;
1668 xmlChar *beg = (xmlChar*)This->pParserCtxt->input->cur-len;
1669 xmlChar *cur, *end;
1670 int realLen;
1671 BSTR Chars;
1672 BOOL lastEvent = FALSE, change;
1673
Piotr Cabanc253bab2011-11-28 15:05:25 +01001674 update_position(This, FALSE);
Piotr Cabana42a0912010-06-27 15:21:28 +02001675 while(beg-9>=This->pParserCtxt->input->base
Piotr Cabanc253bab2011-11-28 15:05:25 +01001676 && memcmp(beg-9, "<![CDATA[", sizeof(char[9])))
1677 {
1678 if(*beg=='\n' || (*beg=='\r' && *(beg+1)!='\n'))
1679 This->line--;
1680 beg--;
1681 }
1682 This->column = 0;
1683 for(; beg>=This->pParserCtxt->input->base && *beg!='\n' && *beg!='\r'; beg--)
1684 This->column++;
Piotr Cabanb84e4782008-10-07 22:11:11 +02001685
Nikolay Sivovfc4dab42012-05-08 15:56:20 +04001686 if (has_lexical_handler(This))
1687 {
1688 if (This->vbInterface)
1689 hr = IVBSAXLexicalHandler_startCDATA(This->saxreader->vblexicalHandler);
1690 else
1691 hr = ISAXLexicalHandler_startCDATA(This->saxreader->lexicalHandler);
1692 }
Piotr Cabanb84e4782008-10-07 22:11:11 +02001693
1694 if(FAILED(hr))
Andrew Talbotce3af502008-11-13 22:20:52 +00001695 {
1696 format_error_message_from_id(This, hr);
1697 return;
1698 }
Piotr Cabanb84e4782008-10-07 22:11:11 +02001699
1700 realLen = This->pParserCtxt->input->cur-beg-3;
1701 cur = beg;
1702 end = beg;
1703
1704 while(1)
1705 {
1706 while(end-beg<realLen && *end!='\r') end++;
1707 if(end-beg==realLen)
1708 {
1709 end--;
1710 lastEvent = TRUE;
1711 }
1712 else if(end-beg==realLen-1 && *end=='\r' && *(end+1)=='\n')
1713 lastEvent = TRUE;
1714
1715 if(*end == '\r') change = TRUE;
1716 else change = FALSE;
1717
1718 if(change) *end = '\n';
1719
Nikolay Sivovff992282010-01-18 23:30:50 +03001720 if(has_content_handler(This))
Piotr Cabanb84e4782008-10-07 22:11:11 +02001721 {
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -05001722 Chars = pooled_bstr_from_xmlCharN(&This->saxreader->pool, cur, end-cur+1);
Piotr Cabanb84e4782008-10-07 22:11:11 +02001723 if(This->vbInterface)
1724 hr = IVBSAXContentHandler_characters(
1725 This->saxreader->vbcontentHandler, &Chars);
1726 else
1727 hr = ISAXContentHandler_characters(
1728 This->saxreader->contentHandler,
1729 Chars, SysStringLen(Chars));
Piotr Cabanb84e4782008-10-07 22:11:11 +02001730 }
1731
1732 if(change) *end = '\r';
1733
1734 if(lastEvent)
1735 break;
1736
1737 This->column += end-cur+2;
1738 end += 2;
1739 cur = end;
1740 }
1741
Nikolay Sivovfc4dab42012-05-08 15:56:20 +04001742 if (has_lexical_handler(This))
1743 {
1744 if (This->vbInterface)
1745 hr = IVBSAXLexicalHandler_endCDATA(This->saxreader->vblexicalHandler);
1746 else
1747 hr = ISAXLexicalHandler_endCDATA(This->saxreader->lexicalHandler);
1748 }
Piotr Cabanb84e4782008-10-07 22:11:11 +02001749
1750 if(FAILED(hr))
1751 format_error_message_from_id(This, hr);
1752
1753 This->column += 4+end-cur;
1754}
1755
Piotr Caband439ca52008-07-31 16:48:11 +02001756/*** IVBSAXLocator interface ***/
1757/*** IUnknown methods ***/
1758static HRESULT WINAPI ivbsaxlocator_QueryInterface(IVBSAXLocator* iface, REFIID riid, void **ppvObject)
1759{
1760 saxlocator *This = impl_from_IVBSAXLocator( iface );
1761
1762 TRACE("%p %s %p\n", This, debugstr_guid( riid ), ppvObject);
1763
1764 *ppvObject = NULL;
1765
1766 if ( IsEqualGUID( riid, &IID_IUnknown ) ||
1767 IsEqualGUID( riid, &IID_IDispatch) ||
1768 IsEqualGUID( riid, &IID_IVBSAXLocator ))
1769 {
1770 *ppvObject = iface;
1771 }
Piotr Caban1b462062011-10-31 12:32:39 +01001772 else if ( IsEqualGUID( riid, &IID_IVBSAXAttributes ))
1773 {
1774 *ppvObject = &This->IVBSAXAttributes_iface;
1775 }
Piotr Caband439ca52008-07-31 16:48:11 +02001776 else
1777 {
1778 FIXME("interface %s not implemented\n", debugstr_guid(riid));
1779 return E_NOINTERFACE;
1780 }
1781
1782 IVBSAXLocator_AddRef( iface );
1783
1784 return S_OK;
1785}
1786
1787static ULONG WINAPI ivbsaxlocator_AddRef(IVBSAXLocator* iface)
1788{
1789 saxlocator *This = impl_from_IVBSAXLocator( iface );
1790 TRACE("%p\n", This );
1791 return InterlockedIncrement( &This->ref );
1792}
1793
1794static ULONG WINAPI ivbsaxlocator_Release(
1795 IVBSAXLocator* iface)
1796{
1797 saxlocator *This = impl_from_IVBSAXLocator( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001798 return ISAXLocator_Release((ISAXLocator*)&This->IVBSAXLocator_iface);
Piotr Caband439ca52008-07-31 16:48:11 +02001799}
1800
1801/*** IDispatch methods ***/
1802static HRESULT WINAPI ivbsaxlocator_GetTypeInfoCount( IVBSAXLocator *iface, UINT* pctinfo )
1803{
1804 saxlocator *This = impl_from_IVBSAXLocator( iface );
1805
1806 TRACE("(%p)->(%p)\n", This, pctinfo);
1807
1808 *pctinfo = 1;
1809
1810 return S_OK;
1811}
1812
1813static HRESULT WINAPI ivbsaxlocator_GetTypeInfo(
1814 IVBSAXLocator *iface,
1815 UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo )
1816{
1817 saxlocator *This = impl_from_IVBSAXLocator( iface );
1818 HRESULT hr;
1819
1820 TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
1821
1822 hr = get_typeinfo(IVBSAXLocator_tid, ppTInfo);
1823
1824 return hr;
1825}
1826
1827static HRESULT WINAPI ivbsaxlocator_GetIDsOfNames(
1828 IVBSAXLocator *iface,
1829 REFIID riid,
1830 LPOLESTR* rgszNames,
1831 UINT cNames,
1832 LCID lcid,
1833 DISPID* rgDispId)
1834{
1835 saxlocator *This = impl_from_IVBSAXLocator( iface );
1836 ITypeInfo *typeinfo;
1837 HRESULT hr;
1838
1839 TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
1840 lcid, rgDispId);
1841
1842 if(!rgszNames || cNames == 0 || !rgDispId)
1843 return E_INVALIDARG;
1844
1845 hr = get_typeinfo(IVBSAXLocator_tid, &typeinfo);
1846 if(SUCCEEDED(hr))
1847 {
1848 hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
1849 ITypeInfo_Release(typeinfo);
1850 }
1851
1852 return hr;
1853}
1854
1855static HRESULT WINAPI ivbsaxlocator_Invoke(
1856 IVBSAXLocator *iface,
1857 DISPID dispIdMember,
1858 REFIID riid,
1859 LCID lcid,
1860 WORD wFlags,
1861 DISPPARAMS* pDispParams,
1862 VARIANT* pVarResult,
1863 EXCEPINFO* pExcepInfo,
1864 UINT* puArgErr)
1865{
1866 saxlocator *This = impl_from_IVBSAXLocator( iface );
1867 ITypeInfo *typeinfo;
1868 HRESULT hr;
1869
1870 TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
1871 lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
1872
1873 hr = get_typeinfo(IVBSAXLocator_tid, &typeinfo);
1874 if(SUCCEEDED(hr))
1875 {
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001876 hr = ITypeInfo_Invoke(typeinfo, &This->IVBSAXLocator_iface, dispIdMember, wFlags,
1877 pDispParams, pVarResult, pExcepInfo, puArgErr);
Piotr Caband439ca52008-07-31 16:48:11 +02001878 ITypeInfo_Release(typeinfo);
1879 }
1880
1881 return hr;
1882}
1883
1884/*** IVBSAXLocator methods ***/
1885static HRESULT WINAPI ivbsaxlocator_get_columnNumber(
1886 IVBSAXLocator* iface,
1887 int *pnColumn)
1888{
1889 saxlocator *This = impl_from_IVBSAXLocator( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001890 return ISAXLocator_getColumnNumber((ISAXLocator*)&This->IVBSAXLocator_iface, pnColumn);
Piotr Caband439ca52008-07-31 16:48:11 +02001891}
1892
1893static HRESULT WINAPI ivbsaxlocator_get_lineNumber(
1894 IVBSAXLocator* iface,
1895 int *pnLine)
1896{
1897 saxlocator *This = impl_from_IVBSAXLocator( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001898 return ISAXLocator_getLineNumber((ISAXLocator*)&This->IVBSAXLocator_iface, pnLine);
Piotr Caband439ca52008-07-31 16:48:11 +02001899}
1900
1901static HRESULT WINAPI ivbsaxlocator_get_publicId(
1902 IVBSAXLocator* iface,
1903 BSTR* publicId)
1904{
1905 saxlocator *This = impl_from_IVBSAXLocator( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001906 return ISAXLocator_getPublicId((ISAXLocator*)&This->IVBSAXLocator_iface,
Piotr Caband439ca52008-07-31 16:48:11 +02001907 (const WCHAR**)publicId);
1908}
1909
1910static HRESULT WINAPI ivbsaxlocator_get_systemId(
1911 IVBSAXLocator* iface,
1912 BSTR* systemId)
1913{
1914 saxlocator *This = impl_from_IVBSAXLocator( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01001915 return ISAXLocator_getSystemId((ISAXLocator*)&This->IVBSAXLocator_iface,
Piotr Caband439ca52008-07-31 16:48:11 +02001916 (const WCHAR**)systemId);
1917}
1918
Nikolay Sivov9d662922011-12-24 16:10:33 +03001919static const struct IVBSAXLocatorVtbl VBSAXLocatorVtbl =
Piotr Caband439ca52008-07-31 16:48:11 +02001920{
1921 ivbsaxlocator_QueryInterface,
1922 ivbsaxlocator_AddRef,
1923 ivbsaxlocator_Release,
1924 ivbsaxlocator_GetTypeInfoCount,
1925 ivbsaxlocator_GetTypeInfo,
1926 ivbsaxlocator_GetIDsOfNames,
1927 ivbsaxlocator_Invoke,
1928 ivbsaxlocator_get_columnNumber,
1929 ivbsaxlocator_get_lineNumber,
1930 ivbsaxlocator_get_publicId,
1931 ivbsaxlocator_get_systemId
1932};
1933
Piotr Caband3e9ca72008-07-17 00:40:32 +02001934/*** ISAXLocator interface ***/
1935/*** IUnknown methods ***/
1936static HRESULT WINAPI isaxlocator_QueryInterface(ISAXLocator* iface, REFIID riid, void **ppvObject)
1937{
1938 saxlocator *This = impl_from_ISAXLocator( iface );
1939
1940 TRACE("%p %s %p\n", This, debugstr_guid( riid ), ppvObject );
1941
1942 *ppvObject = NULL;
1943
1944 if ( IsEqualGUID( riid, &IID_IUnknown ) ||
1945 IsEqualGUID( riid, &IID_ISAXLocator ))
1946 {
1947 *ppvObject = iface;
1948 }
Piotr Caban1b462062011-10-31 12:32:39 +01001949 else if ( IsEqualGUID( riid, &IID_ISAXAttributes ))
1950 {
1951 *ppvObject = &This->ISAXAttributes_iface;
1952 }
Piotr Caband3e9ca72008-07-17 00:40:32 +02001953 else
1954 {
Nikolay Sivov1554cbc2012-04-19 10:55:17 +04001955 WARN("interface %s not implemented\n", debugstr_guid(riid));
Piotr Caband3e9ca72008-07-17 00:40:32 +02001956 return E_NOINTERFACE;
1957 }
1958
1959 ISAXLocator_AddRef( iface );
1960
1961 return S_OK;
1962}
1963
1964static ULONG WINAPI isaxlocator_AddRef(ISAXLocator* iface)
1965{
1966 saxlocator *This = impl_from_ISAXLocator( iface );
Nikolay Sivov9d662922011-12-24 16:10:33 +03001967 ULONG ref = InterlockedIncrement( &This->ref );
1968 TRACE("(%p)->(%d)\n", This, ref);
1969 return ref;
Piotr Caband3e9ca72008-07-17 00:40:32 +02001970}
1971
1972static ULONG WINAPI isaxlocator_Release(
1973 ISAXLocator* iface)
1974{
1975 saxlocator *This = impl_from_ISAXLocator( iface );
Nikolay Sivov9d662922011-12-24 16:10:33 +03001976 LONG ref = InterlockedDecrement( &This->ref );
Piotr Caband3e9ca72008-07-17 00:40:32 +02001977
Nikolay Sivov9d662922011-12-24 16:10:33 +03001978 TRACE("(%p)->(%d)\n", This, ref );
Piotr Caband3e9ca72008-07-17 00:40:32 +02001979
Nikolay Sivov9d662922011-12-24 16:10:33 +03001980 if (ref == 0)
Piotr Caband3e9ca72008-07-17 00:40:32 +02001981 {
Nikolay Sivov9d662922011-12-24 16:10:33 +03001982 element_entry *element, *element2;
Piotr Caban1b462062011-10-31 12:32:39 +01001983 int index;
1984
Detlef Riekenbergf4558c62008-09-21 15:56:18 +02001985 SysFreeString(This->publicId);
1986 SysFreeString(This->systemId);
Piotr Cabana95b35d2011-10-31 12:33:18 +01001987 SysFreeString(This->namespaceUri);
Piotr Caban60ca0a72008-07-17 00:42:01 +02001988
Piotr Caban1b462062011-10-31 12:32:39 +01001989 for(index=0; index<This->nb_attributes; index++)
1990 {
1991 SysFreeString(This->attributes[index].szLocalname);
Piotr Caban1b462062011-10-31 12:32:39 +01001992 SysFreeString(This->attributes[index].szValue);
1993 SysFreeString(This->attributes[index].szQName);
1994 }
1995 heap_free(This->attributes);
1996
Nikolay Sivov9d662922011-12-24 16:10:33 +03001997 /* element stack */
1998 LIST_FOR_EACH_ENTRY_SAFE(element, element2, &This->elements, element_entry, entry)
1999 {
2000 list_remove(&element->entry);
2001 free_element_entry(element);
2002 }
2003
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002004 ISAXXMLReader_Release(&This->saxreader->ISAXXMLReader_iface);
Nikolay Sivovd8a9a692010-02-03 22:47:53 +03002005 heap_free( This );
Piotr Caband3e9ca72008-07-17 00:40:32 +02002006 }
2007
2008 return ref;
2009}
2010
2011/*** ISAXLocator methods ***/
2012static HRESULT WINAPI isaxlocator_getColumnNumber(
2013 ISAXLocator* iface,
2014 int *pnColumn)
2015{
2016 saxlocator *This = impl_from_ISAXLocator( iface );
2017
Piotr Cabanbb849dc2008-07-23 16:39:04 +02002018 *pnColumn = This->column;
Piotr Caban4cdcc022008-07-17 00:41:11 +02002019 return S_OK;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002020}
2021
2022static HRESULT WINAPI isaxlocator_getLineNumber(
2023 ISAXLocator* iface,
2024 int *pnLine)
2025{
2026 saxlocator *This = impl_from_ISAXLocator( iface );
2027
Piotr Cabanbb849dc2008-07-23 16:39:04 +02002028 *pnLine = This->line;
Piotr Caban4cdcc022008-07-17 00:41:11 +02002029 return S_OK;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002030}
2031
2032static HRESULT WINAPI isaxlocator_getPublicId(
2033 ISAXLocator* iface,
2034 const WCHAR ** ppwchPublicId)
2035{
Piotr Caban60ca0a72008-07-17 00:42:01 +02002036 BSTR publicId;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002037 saxlocator *This = impl_from_ISAXLocator( iface );
2038
Detlef Riekenbergf4558c62008-09-21 15:56:18 +02002039 SysFreeString(This->publicId);
Piotr Caban60ca0a72008-07-17 00:42:01 +02002040
2041 publicId = bstr_from_xmlChar(xmlSAX2GetPublicId(This->pParserCtxt));
2042 if(SysStringLen(publicId))
2043 This->publicId = (WCHAR*)&publicId;
2044 else
2045 {
2046 SysFreeString(publicId);
2047 This->publicId = NULL;
2048 }
2049
2050 *ppwchPublicId = This->publicId;
2051 return S_OK;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002052}
2053
2054static HRESULT WINAPI isaxlocator_getSystemId(
2055 ISAXLocator* iface,
2056 const WCHAR ** ppwchSystemId)
2057{
Piotr Caban3196f782008-07-17 00:42:07 +02002058 BSTR systemId;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002059 saxlocator *This = impl_from_ISAXLocator( iface );
2060
Detlef Riekenbergf4558c62008-09-21 15:56:18 +02002061 SysFreeString(This->systemId);
Piotr Caban3196f782008-07-17 00:42:07 +02002062
2063 systemId = bstr_from_xmlChar(xmlSAX2GetSystemId(This->pParserCtxt));
2064 if(SysStringLen(systemId))
2065 This->systemId = (WCHAR*)&systemId;
2066 else
2067 {
2068 SysFreeString(systemId);
2069 This->systemId = NULL;
2070 }
2071
2072 *ppwchSystemId = This->systemId;
2073 return S_OK;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002074}
2075
Nikolay Sivov9d662922011-12-24 16:10:33 +03002076static const struct ISAXLocatorVtbl SAXLocatorVtbl =
Piotr Caband3e9ca72008-07-17 00:40:32 +02002077{
2078 isaxlocator_QueryInterface,
2079 isaxlocator_AddRef,
2080 isaxlocator_Release,
2081 isaxlocator_getColumnNumber,
2082 isaxlocator_getLineNumber,
2083 isaxlocator_getPublicId,
2084 isaxlocator_getSystemId
2085};
2086
Piotr Cabanc52e0912008-07-31 16:48:03 +02002087static HRESULT SAXLocator_create(saxreader *reader, saxlocator **ppsaxlocator, BOOL vbInterface)
Piotr Caband3e9ca72008-07-17 00:40:32 +02002088{
Piotr Cabana95b35d2011-10-31 12:33:18 +01002089 static const WCHAR w3xmlns[] = { 'h','t','t','p',':','/','/', 'w','w','w','.','w','3','.',
2090 'o','r','g','/','2','0','0','0','/','x','m','l','n','s','/',0 };
2091
Piotr Caband3e9ca72008-07-17 00:40:32 +02002092 saxlocator *locator;
2093
Nikolay Sivovd8a9a692010-02-03 22:47:53 +03002094 locator = heap_alloc( sizeof (*locator) );
Piotr Caband3e9ca72008-07-17 00:40:32 +02002095 if( !locator )
2096 return E_OUTOFMEMORY;
2097
Nikolay Sivov9d662922011-12-24 16:10:33 +03002098 locator->IVBSAXLocator_iface.lpVtbl = &VBSAXLocatorVtbl;
2099 locator->ISAXLocator_iface.lpVtbl = &SAXLocatorVtbl;
Piotr Caban1b462062011-10-31 12:32:39 +01002100 locator->IVBSAXAttributes_iface.lpVtbl = &ivbsaxattributes_vtbl;
2101 locator->ISAXAttributes_iface.lpVtbl = &isaxattributes_vtbl;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002102 locator->ref = 1;
Piotr Cabanc52e0912008-07-31 16:48:03 +02002103 locator->vbInterface = vbInterface;
Piotr Caband3e9ca72008-07-17 00:40:32 +02002104
2105 locator->saxreader = reader;
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002106 ISAXXMLReader_AddRef(&reader->ISAXXMLReader_iface);
Piotr Caband3e9ca72008-07-17 00:40:32 +02002107
Piotr Caban51c95272008-07-17 00:40:53 +02002108 locator->pParserCtxt = NULL;
Piotr Caban60ca0a72008-07-17 00:42:01 +02002109 locator->publicId = NULL;
Piotr Caban3196f782008-07-17 00:42:07 +02002110 locator->systemId = NULL;
Nikolay Sivovbcd85852012-04-15 13:47:30 +04002111 locator->line = reader->version < MSXML4 ? 0 : 1;
Piotr Cabanbb849dc2008-07-23 16:39:04 +02002112 locator->column = 0;
Piotr Caban51c95272008-07-17 00:40:53 +02002113 locator->ret = S_OK;
Nikolay Sivovbcd85852012-04-15 13:47:30 +04002114 if (locator->saxreader->version >= MSXML6)
Piotr Cabana95b35d2011-10-31 12:33:18 +01002115 locator->namespaceUri = SysAllocString(w3xmlns);
2116 else
2117 locator->namespaceUri = SysAllocStringLen(NULL, 0);
2118 if(!locator->namespaceUri)
2119 {
2120 ISAXXMLReader_Release(&reader->ISAXXMLReader_iface);
2121 heap_free(locator);
2122 return E_OUTOFMEMORY;
2123 }
Piotr Caban51c95272008-07-17 00:40:53 +02002124
Piotr Caban1b462062011-10-31 12:32:39 +01002125 locator->attributesSize = 8;
2126 locator->nb_attributes = 0;
2127 locator->attributes = heap_alloc(sizeof(struct _attributes)*locator->attributesSize);
2128 if(!locator->attributes)
2129 {
2130 ISAXXMLReader_Release(&reader->ISAXXMLReader_iface);
Piotr Cabana95b35d2011-10-31 12:33:18 +01002131 SysFreeString(locator->namespaceUri);
Piotr Caban1b462062011-10-31 12:32:39 +01002132 heap_free(locator);
2133 return E_OUTOFMEMORY;
2134 }
2135
Nikolay Sivov9d662922011-12-24 16:10:33 +03002136 list_init(&locator->elements);
2137
Piotr Caband3e9ca72008-07-17 00:40:32 +02002138 *ppsaxlocator = locator;
2139
2140 TRACE("returning %p\n", *ppsaxlocator);
2141
2142 return S_OK;
2143}
2144
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002145/*** SAXXMLReader internal functions ***/
2146static HRESULT internal_parseBuffer(saxreader *This, const char *buffer, int size, BOOL vbInterface)
2147{
Nikolay Sivov678fbc12010-10-30 18:49:43 +04002148 xmlCharEncoding encoding = XML_CHAR_ENCODING_NONE;
2149 xmlChar *enc_name = NULL;
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002150 saxlocator *locator;
2151 HRESULT hr;
2152
2153 hr = SAXLocator_create(This, &locator, vbInterface);
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002154 if (FAILED(hr))
Ricardo Filipee5e301b2008-11-13 13:33:27 +00002155 return hr;
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002156
Nikolay Sivov678fbc12010-10-30 18:49:43 +04002157 if (size >= 4)
2158 {
2159 const unsigned char *buff = (unsigned char*)buffer;
2160
2161 encoding = xmlDetectCharEncoding((xmlChar*)buffer, 4);
2162 enc_name = (xmlChar*)xmlGetCharEncodingName(encoding);
2163 TRACE("detected encoding: %s\n", enc_name);
2164 /* skip BOM, parser won't switch encodings and so won't skip it on its own */
2165 if ((encoding == XML_CHAR_ENCODING_UTF8) &&
2166 buff[0] == 0xEF && buff[1] == 0xBB && buff[2] == 0xBF)
2167 {
2168 buffer += 3;
2169 size -= 3;
2170 }
2171 }
2172
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002173 /* if libxml2 detection failed try to guess */
2174 if (encoding == XML_CHAR_ENCODING_NONE)
2175 {
2176 const WCHAR *ptr = (WCHAR*)buffer;
2177 /* xml declaration with possibly specfied encoding will be still handled by parser */
2178 if ((size >= 2) && *ptr == '<' && ptr[1] != '?')
2179 {
2180 enc_name = (xmlChar*)xmlGetCharEncodingName(XML_CHAR_ENCODING_UTF16LE);
2181 encoding = XML_CHAR_ENCODING_UTF16LE;
2182 }
2183 }
2184 else if (encoding == XML_CHAR_ENCODING_UTF8)
2185 enc_name = (xmlChar*)xmlGetCharEncodingName(encoding);
2186 else
2187 enc_name = NULL;
2188
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002189 locator->pParserCtxt = xmlCreateMemoryParserCtxt(buffer, size);
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002190 if (!locator->pParserCtxt)
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002191 {
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002192 ISAXLocator_Release(&locator->ISAXLocator_iface);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002193 return E_FAIL;
2194 }
2195
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002196 if (enc_name)
2197 {
Nikolay Sivov678fbc12010-10-30 18:49:43 +04002198 locator->pParserCtxt->encoding = xmlStrdup(enc_name);
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002199 if (encoding == XML_CHAR_ENCODING_UTF16LE) {
2200 TRACE("switching to %s\n", enc_name);
2201 xmlSwitchEncoding(locator->pParserCtxt, encoding);
2202 }
2203 }
Nikolay Sivov678fbc12010-10-30 18:49:43 +04002204
Nikolay Sivove4d895c2010-01-10 18:17:37 +03002205 xmlFree(locator->pParserCtxt->sax);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002206 locator->pParserCtxt->sax = &locator->saxreader->sax;
2207 locator->pParserCtxt->userData = locator;
2208
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002209 This->isParsing = TRUE;
Nikolay Sivov221f9ef2012-04-21 12:32:15 +04002210 if(xmlParseDocument(locator->pParserCtxt) == -1 && locator->ret == S_OK)
Piotr Caban6927a112011-10-24 12:55:13 +02002211 hr = E_FAIL;
2212 else
2213 hr = locator->ret;
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002214 This->isParsing = FALSE;
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002215
2216 if(locator->pParserCtxt)
2217 {
2218 locator->pParserCtxt->sax = NULL;
2219 xmlFreeParserCtxt(locator->pParserCtxt);
2220 locator->pParserCtxt = NULL;
2221 }
2222
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002223 ISAXLocator_Release(&locator->ISAXLocator_iface);
Ricardo Filipee5e301b2008-11-13 13:33:27 +00002224 return hr;
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002225}
2226
Piotr Caban38b05732008-09-08 15:07:46 +02002227static HRESULT internal_parseStream(saxreader *This, IStream *stream, BOOL vbInterface)
2228{
2229 saxlocator *locator;
2230 HRESULT hr;
2231 ULONG dataRead;
2232 char data[1024];
Nikolay Sivov809959d2011-09-21 13:48:17 +04002233 int ret;
Piotr Caban38b05732008-09-08 15:07:46 +02002234
Nikolay Sivov809959d2011-09-21 13:48:17 +04002235 dataRead = 0;
Piotr Caban38b05732008-09-08 15:07:46 +02002236 hr = IStream_Read(stream, data, sizeof(data), &dataRead);
Nikolay Sivov809959d2011-09-21 13:48:17 +04002237 if(FAILED(hr)) return hr;
Piotr Caban38b05732008-09-08 15:07:46 +02002238
2239 hr = SAXLocator_create(This, &locator, vbInterface);
Nikolay Sivov809959d2011-09-21 13:48:17 +04002240 if(FAILED(hr)) return hr;
Piotr Caban38b05732008-09-08 15:07:46 +02002241
2242 locator->pParserCtxt = xmlCreatePushParserCtxt(
2243 &locator->saxreader->sax, locator,
2244 data, dataRead, NULL);
2245 if(!locator->pParserCtxt)
2246 {
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002247 ISAXLocator_Release(&locator->ISAXLocator_iface);
Piotr Caban38b05732008-09-08 15:07:46 +02002248 return E_FAIL;
2249 }
2250
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002251 This->isParsing = TRUE;
Nikolay Sivov809959d2011-09-21 13:48:17 +04002252
2253 if(dataRead != sizeof(data))
Piotr Caban38b05732008-09-08 15:07:46 +02002254 {
Nikolay Sivov809959d2011-09-21 13:48:17 +04002255 ret = xmlParseChunk(locator->pParserCtxt, data, 0, 1);
Piotr Caban6927a112011-10-24 12:55:13 +02002256 hr = ret!=XML_ERR_OK && locator->ret==S_OK ? E_FAIL : locator->ret;
Nikolay Sivov809959d2011-09-21 13:48:17 +04002257 }
2258 else
2259 {
2260 while(1)
Piotr Caban38b05732008-09-08 15:07:46 +02002261 {
Nikolay Sivov809959d2011-09-21 13:48:17 +04002262 dataRead = 0;
2263 hr = IStream_Read(stream, data, sizeof(data), &dataRead);
2264 if (FAILED(hr)) break;
Piotr Caban38b05732008-09-08 15:07:46 +02002265
Nikolay Sivov809959d2011-09-21 13:48:17 +04002266 ret = xmlParseChunk(locator->pParserCtxt, data, dataRead, 0);
Piotr Caban6927a112011-10-24 12:55:13 +02002267 hr = ret!=XML_ERR_OK && locator->ret==S_OK ? E_FAIL : locator->ret;
Nikolay Sivov809959d2011-09-21 13:48:17 +04002268
2269 if (hr != S_OK) break;
2270
2271 if (dataRead != sizeof(data))
2272 {
2273 ret = xmlParseChunk(locator->pParserCtxt, data, 0, 1);
Piotr Caban6927a112011-10-24 12:55:13 +02002274 hr = ret!=XML_ERR_OK && locator->ret==S_OK ? E_FAIL : locator->ret;
Nikolay Sivov809959d2011-09-21 13:48:17 +04002275 break;
2276 }
Piotr Caban38b05732008-09-08 15:07:46 +02002277 }
2278 }
Nikolay Sivov809959d2011-09-21 13:48:17 +04002279
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002280 This->isParsing = FALSE;
Piotr Caban38b05732008-09-08 15:07:46 +02002281
Piotr Caban38b05732008-09-08 15:07:46 +02002282 xmlFreeParserCtxt(locator->pParserCtxt);
2283 locator->pParserCtxt = NULL;
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002284 ISAXLocator_Release(&locator->ISAXLocator_iface);
Piotr Caban38b05732008-09-08 15:07:46 +02002285 return hr;
2286}
2287
Francois Gouget848b5092008-11-24 17:23:59 +01002288static HRESULT internal_getEntityResolver(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002289 saxreader *This,
2290 void *pEntityResolver,
2291 BOOL vbInterface)
2292{
2293 FIXME("(%p)->(%p) stub\n", This, pEntityResolver);
2294 return E_NOTIMPL;
2295}
2296
Francois Gouget848b5092008-11-24 17:23:59 +01002297static HRESULT internal_putEntityResolver(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002298 saxreader *This,
2299 void *pEntityResolver,
2300 BOOL vbInterface)
2301{
2302 FIXME("(%p)->(%p) stub\n", This, pEntityResolver);
2303 return E_NOTIMPL;
2304}
2305
Francois Gouget848b5092008-11-24 17:23:59 +01002306static HRESULT internal_getContentHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002307 saxreader* This,
2308 void *pContentHandler,
2309 BOOL vbInterface)
2310{
2311 TRACE("(%p)->(%p)\n", This, pContentHandler);
2312 if(pContentHandler == NULL)
2313 return E_POINTER;
Piotr Cabanf3ab2282008-08-20 18:26:25 +02002314 if((vbInterface && This->vbcontentHandler)
2315 || (!vbInterface && This->contentHandler))
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002316 {
2317 if(vbInterface)
2318 IVBSAXContentHandler_AddRef(This->vbcontentHandler);
2319 else
2320 ISAXContentHandler_AddRef(This->contentHandler);
2321 }
2322 if(vbInterface) *(IVBSAXContentHandler**)pContentHandler =
2323 This->vbcontentHandler;
2324 else *(ISAXContentHandler**)pContentHandler = This->contentHandler;
2325
2326 return S_OK;
2327}
2328
Francois Gouget848b5092008-11-24 17:23:59 +01002329static HRESULT internal_putContentHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002330 saxreader* This,
2331 void *contentHandler,
2332 BOOL vbInterface)
2333{
2334 TRACE("(%p)->(%p)\n", This, contentHandler);
2335 if(contentHandler)
2336 {
2337 if(vbInterface)
2338 IVBSAXContentHandler_AddRef((IVBSAXContentHandler*)contentHandler);
2339 else
2340 ISAXContentHandler_AddRef((ISAXContentHandler*)contentHandler);
2341 }
Piotr Cabanf3ab2282008-08-20 18:26:25 +02002342 if((vbInterface && This->vbcontentHandler)
2343 || (!vbInterface && This->contentHandler))
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002344 {
2345 if(vbInterface)
2346 IVBSAXContentHandler_Release(This->vbcontentHandler);
2347 else
2348 ISAXContentHandler_Release(This->contentHandler);
2349 }
2350 if(vbInterface)
2351 This->vbcontentHandler = contentHandler;
2352 else
2353 This->contentHandler = contentHandler;
2354
2355 return S_OK;
2356}
2357
Francois Gouget848b5092008-11-24 17:23:59 +01002358static HRESULT internal_getDTDHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002359 saxreader* This,
2360 void *pDTDHandler,
2361 BOOL vbInterface)
2362{
2363 FIXME("(%p)->(%p) stub\n", This, pDTDHandler);
2364 return E_NOTIMPL;
2365}
2366
Francois Gouget848b5092008-11-24 17:23:59 +01002367static HRESULT internal_putDTDHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002368 saxreader* This,
2369 void *pDTDHandler,
2370 BOOL vbInterface)
2371{
2372 FIXME("(%p)->(%p) stub\n", This, pDTDHandler);
2373 return E_NOTIMPL;
2374}
2375
Francois Gouget848b5092008-11-24 17:23:59 +01002376static HRESULT internal_getErrorHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002377 saxreader* This,
2378 void *pErrorHandler,
2379 BOOL vbInterface)
2380{
2381 TRACE("(%p)->(%p)\n", This, pErrorHandler);
2382 if(pErrorHandler == NULL)
2383 return E_POINTER;
Piotr Caban7b7a1872008-08-20 18:32:23 +02002384
2385 if(vbInterface && This->vberrorHandler)
2386 IVBSAXErrorHandler_AddRef(This->vberrorHandler);
2387 else if(!vbInterface && This->errorHandler)
2388 ISAXErrorHandler_AddRef(This->errorHandler);
2389
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002390 if(vbInterface)
2391 *(IVBSAXErrorHandler**)pErrorHandler = This->vberrorHandler;
2392 else
2393 *(ISAXErrorHandler**)pErrorHandler = This->errorHandler;
2394
2395 return S_OK;
2396
2397}
2398
Francois Gouget848b5092008-11-24 17:23:59 +01002399static HRESULT internal_putErrorHandler(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002400 saxreader* This,
2401 void *errorHandler,
2402 BOOL vbInterface)
2403{
2404 TRACE("(%p)->(%p)\n", This, errorHandler);
2405 if(errorHandler)
2406 {
2407 if(vbInterface)
2408 IVBSAXErrorHandler_AddRef((IVBSAXErrorHandler*)errorHandler);
2409 else
2410 ISAXErrorHandler_AddRef((ISAXErrorHandler*)errorHandler);
2411 }
Piotr Caban7b7a1872008-08-20 18:32:23 +02002412
2413 if(vbInterface && This->vberrorHandler)
2414 IVBSAXErrorHandler_Release(This->vberrorHandler);
2415 else if(!vbInterface && This->errorHandler)
2416 ISAXErrorHandler_Release(This->errorHandler);
2417
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002418 if(vbInterface)
2419 This->vberrorHandler = errorHandler;
2420 else
2421 This->errorHandler = errorHandler;
2422
2423 return S_OK;
2424
2425}
2426
Francois Gouget848b5092008-11-24 17:23:59 +01002427static HRESULT internal_parse(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002428 saxreader* This,
2429 VARIANT varInput,
2430 BOOL vbInterface)
2431{
2432 HRESULT hr;
2433
Nikolay Sivova0a464a2011-02-22 02:04:02 +03002434 TRACE("(%p)->(%s)\n", This, debugstr_variant(&varInput));
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002435
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -05002436 /* Dispose of the BSTRs in the pool from a prior run, if any. */
2437 free_bstr_pool(&This->pool);
2438
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002439 switch(V_VT(&varInput))
2440 {
2441 case VT_BSTR:
2442 hr = internal_parseBuffer(This, (const char*)V_BSTR(&varInput),
2443 SysStringByteLen(V_BSTR(&varInput)), vbInterface);
2444 break;
2445 case VT_ARRAY|VT_UI1: {
2446 void *pSAData;
2447 LONG lBound, uBound;
2448 ULONG dataRead;
2449
2450 hr = SafeArrayGetLBound(V_ARRAY(&varInput), 1, &lBound);
2451 if(hr != S_OK) break;
2452 hr = SafeArrayGetUBound(V_ARRAY(&varInput), 1, &uBound);
2453 if(hr != S_OK) break;
2454 dataRead = (uBound-lBound)*SafeArrayGetElemsize(V_ARRAY(&varInput));
Michael Stefaniuc6a5dda72009-01-26 11:01:02 +01002455 hr = SafeArrayAccessData(V_ARRAY(&varInput), &pSAData);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002456 if(hr != S_OK) break;
2457 hr = internal_parseBuffer(This, pSAData, dataRead, vbInterface);
2458 SafeArrayUnaccessData(V_ARRAY(&varInput));
2459 break;
2460 }
2461 case VT_UNKNOWN:
2462 case VT_DISPATCH: {
2463 IPersistStream *persistStream;
2464 IStream *stream = NULL;
2465 IXMLDOMDocument *xmlDoc;
2466
2467 if(IUnknown_QueryInterface(V_UNKNOWN(&varInput),
Piotr Cabana28b6042008-09-19 16:01:03 +02002468 &IID_IXMLDOMDocument, (void**)&xmlDoc) == S_OK)
2469 {
2470 BSTR bstrData;
2471
2472 IXMLDOMDocument_get_xml(xmlDoc, &bstrData);
2473 hr = internal_parseBuffer(This, (const char*)bstrData,
2474 SysStringByteLen(bstrData), vbInterface);
2475 IXMLDOMDocument_Release(xmlDoc);
Nikolay Sivov73a955e2010-01-10 18:46:30 +03002476 SysFreeString(bstrData);
Piotr Cabana28b6042008-09-19 16:01:03 +02002477 break;
2478 }
Nikolay Sivov850be4d2010-10-28 01:27:06 +04002479
Piotr Cabana28b6042008-09-19 16:01:03 +02002480 if(IUnknown_QueryInterface(V_UNKNOWN(&varInput),
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002481 &IID_IPersistStream, (void**)&persistStream) == S_OK)
2482 {
Nikolay Sivov850be4d2010-10-28 01:27:06 +04002483 hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
2484 if(hr != S_OK)
2485 {
2486 IPersistStream_Release(persistStream);
2487 return hr;
2488 }
2489
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002490 hr = IPersistStream_Save(persistStream, stream, TRUE);
2491 IPersistStream_Release(persistStream);
Nikolay Sivov850be4d2010-10-28 01:27:06 +04002492 if(hr != S_OK)
2493 {
2494 IStream_Release(stream);
Nikolay Sivov012806a2012-01-23 10:31:55 +03002495 stream = NULL;
Nikolay Sivov850be4d2010-10-28 01:27:06 +04002496 }
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002497 }
Nikolay Sivov012806a2012-01-23 10:31:55 +03002498
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002499 if(stream || IUnknown_QueryInterface(V_UNKNOWN(&varInput),
2500 &IID_IStream, (void**)&stream) == S_OK)
2501 {
Piotr Caban38b05732008-09-08 15:07:46 +02002502 hr = internal_parseStream(This, stream, vbInterface);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002503 IStream_Release(stream);
2504 break;
2505 }
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002506 }
2507 default:
2508 WARN("vt %d not implemented\n", V_VT(&varInput));
2509 hr = E_INVALIDARG;
2510 }
2511
2512 return hr;
2513}
2514
2515static HRESULT internal_vbonDataAvailable(void *obj, char *ptr, DWORD len)
2516{
2517 saxreader *This = obj;
2518
2519 return internal_parseBuffer(This, ptr, len, TRUE);
2520}
2521
2522static HRESULT internal_onDataAvailable(void *obj, char *ptr, DWORD len)
2523{
2524 saxreader *This = obj;
2525
2526 return internal_parseBuffer(This, ptr, len, FALSE);
2527}
2528
Francois Gouget848b5092008-11-24 17:23:59 +01002529static HRESULT internal_parseURL(
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002530 saxreader* This,
2531 const WCHAR *url,
2532 BOOL vbInterface)
2533{
Piotr Caban2cbd6a62012-03-12 10:56:59 +01002534 IMoniker *mon;
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002535 bsc_t *bsc;
2536 HRESULT hr;
2537
2538 TRACE("(%p)->(%s)\n", This, debugstr_w(url));
2539
Piotr Caban2cbd6a62012-03-12 10:56:59 +01002540 hr = create_moniker_from_url(url, &mon);
2541 if(FAILED(hr))
2542 return hr;
2543
2544 if(vbInterface) hr = bind_url(mon, internal_vbonDataAvailable, This, &bsc);
2545 else hr = bind_url(mon, internal_onDataAvailable, This, &bsc);
2546 IMoniker_Release(mon);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002547
2548 if(FAILED(hr))
2549 return hr;
2550
Piotr Caban41eba1c2011-10-24 12:55:00 +02002551 return detach_bsc(bsc);
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002552}
2553
Francois Gouget848b5092008-11-24 17:23:59 +01002554static HRESULT internal_putProperty(
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002555 saxreader* This,
Nikolay Sivov864f1702011-09-21 21:01:10 +04002556 const WCHAR *prop,
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002557 VARIANT value,
2558 BOOL vbInterface)
2559{
Nikolay Sivov864f1702011-09-21 21:01:10 +04002560 TRACE("(%p)->(%s %s)\n", This, debugstr_w(prop), debugstr_variant(&value));
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002561
Nikolay Sivov864f1702011-09-21 21:01:10 +04002562 if(!memcmp(prop, PropertyDeclHandlerW, sizeof(PropertyDeclHandlerW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002563 {
2564 if(This->isParsing) return E_FAIL;
2565
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002566 switch (V_VT(&value))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002567 {
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002568 case VT_EMPTY:
2569 if (vbInterface)
2570 {
2571 if (This->vbdeclHandler)
2572 {
2573 IVBSAXDeclHandler_Release(This->vbdeclHandler);
2574 This->vbdeclHandler = NULL;
2575 }
2576 }
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002577 else
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002578 if (This->declHandler)
2579 {
2580 ISAXDeclHandler_Release(This->declHandler);
2581 This->declHandler = NULL;
2582 }
2583 break;
2584 case VT_UNKNOWN:
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002585 if ((vbInterface && This->vbdeclHandler) ||
2586 (!vbInterface && This->declHandler))
2587 {
2588 if (vbInterface)
2589 IVBSAXDeclHandler_Release(This->vbdeclHandler);
2590 else
2591 ISAXDeclHandler_Release(This->declHandler);
2592 }
2593
Nikolay Sivov28335fb2012-04-26 23:54:22 +04002594 if (V_UNKNOWN(&value))
2595 {
2596 return vbInterface ?
2597 IVBSAXDeclHandler_QueryInterface(V_UNKNOWN(&value), &IID_IVBSAXDeclHandler, (void**)&This->vbdeclHandler) :
2598 ISAXDeclHandler_QueryInterface(V_UNKNOWN(&value), &IID_ISAXDeclHandler, (void**)&This->declHandler);
2599 }
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002600 else
Nikolay Sivov28335fb2012-04-26 23:54:22 +04002601 {
2602 This->vbdeclHandler = NULL;
2603 This->declHandler = NULL;
2604 }
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002605 break;
2606 default:
2607 return E_INVALIDARG;
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002608 }
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002609
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002610 return S_OK;
2611 }
2612
Nikolay Sivov864f1702011-09-21 21:01:10 +04002613 if(!memcmp(prop, PropertyLexicalHandlerW, sizeof(PropertyLexicalHandlerW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002614 {
2615 if(This->isParsing) return E_FAIL;
2616
Nikolay Sivov864f1702011-09-21 21:01:10 +04002617 switch (V_VT(&value))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002618 {
Nikolay Sivov864f1702011-09-21 21:01:10 +04002619 case VT_EMPTY:
2620 if (vbInterface)
2621 {
2622 if (This->vblexicalHandler)
2623 {
2624 IVBSAXLexicalHandler_Release(This->vblexicalHandler);
2625 This->vblexicalHandler = NULL;
2626 }
2627 }
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002628 else
Nikolay Sivov864f1702011-09-21 21:01:10 +04002629 if (This->lexicalHandler)
2630 {
2631 ISAXLexicalHandler_Release(This->lexicalHandler);
2632 This->lexicalHandler = NULL;
2633 }
2634 break;
2635 case VT_UNKNOWN:
Nikolay Sivov864f1702011-09-21 21:01:10 +04002636 if ((vbInterface && This->vblexicalHandler) ||
2637 (!vbInterface && This->lexicalHandler))
2638 {
2639 if (vbInterface)
2640 IVBSAXLexicalHandler_Release(This->vblexicalHandler);
2641 else
2642 ISAXLexicalHandler_Release(This->lexicalHandler);
2643 }
2644
Nikolay Sivov28335fb2012-04-26 23:54:22 +04002645 if (V_UNKNOWN(&value))
2646 {
2647 return vbInterface ?
2648 IVBSAXLexicalHandler_QueryInterface(V_UNKNOWN(&value), &IID_IVBSAXLexicalHandler, (void**)&This->vblexicalHandler) :
2649 ISAXLexicalHandler_QueryInterface(V_UNKNOWN(&value), &IID_ISAXLexicalHandler, (void**)&This->lexicalHandler);
2650 }
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002651 else
Nikolay Sivov28335fb2012-04-26 23:54:22 +04002652 {
2653 This->vblexicalHandler = NULL;
2654 This->lexicalHandler = NULL;
2655 }
2656
Nikolay Sivov864f1702011-09-21 21:01:10 +04002657 break;
2658 default:
2659 return E_INVALIDARG;
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002660 }
Nikolay Sivov864f1702011-09-21 21:01:10 +04002661
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002662 return S_OK;
2663 }
2664
Nikolay Sivov393c2322011-10-05 11:19:43 -05002665 if(!memcmp(prop, PropertyMaxXMLSizeW, sizeof(PropertyMaxXMLSizeW)))
2666 {
2667 if (V_VT(&value) == VT_I4 && V_I4(&value) == 0) return S_OK;
2668 FIXME("(%p)->(%s): max-xml-size unsupported\n", This, debugstr_variant(&value));
2669 return E_NOTIMPL;
2670 }
2671
Nikolay Sivov5f68f372011-10-05 11:23:56 -05002672 if(!memcmp(prop, PropertyMaxElementDepthW, sizeof(PropertyMaxElementDepthW)))
2673 {
2674 if (V_VT(&value) == VT_I4 && V_I4(&value) == 0) return S_OK;
2675 FIXME("(%p)->(%s): max-element-depth unsupported\n", This, debugstr_variant(&value));
2676 return E_NOTIMPL;
2677 }
2678
Nikolay Sivov393c2322011-10-05 11:19:43 -05002679 FIXME("(%p)->(%s:%s): unsupported property\n", This, debugstr_w(prop), debugstr_variant(&value));
Nikolay Sivovfd350842010-08-30 01:33:28 +04002680
Nikolay Sivov864f1702011-09-21 21:01:10 +04002681 if(!memcmp(prop, PropertyCharsetW, sizeof(PropertyCharsetW)))
Nikolay Sivovfd350842010-08-30 01:33:28 +04002682 return E_NOTIMPL;
2683
Nikolay Sivov864f1702011-09-21 21:01:10 +04002684 if(!memcmp(prop, PropertyDomNodeW, sizeof(PropertyDomNodeW)))
Nikolay Sivovfd350842010-08-30 01:33:28 +04002685 return E_FAIL;
2686
Nikolay Sivov864f1702011-09-21 21:01:10 +04002687 if(!memcmp(prop, PropertyInputSourceW, sizeof(PropertyInputSourceW)))
Nikolay Sivovfd350842010-08-30 01:33:28 +04002688 return E_NOTIMPL;
2689
Nikolay Sivov864f1702011-09-21 21:01:10 +04002690 if(!memcmp(prop, PropertySchemaDeclHandlerW, sizeof(PropertySchemaDeclHandlerW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002691 return E_NOTIMPL;
2692
Nikolay Sivov864f1702011-09-21 21:01:10 +04002693 if(!memcmp(prop, PropertyXMLDeclEncodingW, sizeof(PropertyXMLDeclEncodingW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002694 return E_FAIL;
2695
Nikolay Sivov864f1702011-09-21 21:01:10 +04002696 if(!memcmp(prop, PropertyXMLDeclStandaloneW, sizeof(PropertyXMLDeclStandaloneW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002697 return E_FAIL;
2698
Nikolay Sivov864f1702011-09-21 21:01:10 +04002699 if(!memcmp(prop, PropertyXMLDeclVersionW, sizeof(PropertyXMLDeclVersionW)))
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002700 return E_FAIL;
2701
2702 return E_INVALIDARG;
2703}
2704
Nikolay Sivov864f1702011-09-21 21:01:10 +04002705static HRESULT internal_getProperty(const saxreader* This, const WCHAR *prop, VARIANT *value, BOOL vb)
2706{
2707 TRACE("(%p)->(%s)\n", This, debugstr_w(prop));
2708
2709 if (!value) return E_POINTER;
2710
2711 if (!memcmp(PropertyLexicalHandlerW, prop, sizeof(PropertyLexicalHandlerW)))
2712 {
2713 V_VT(value) = VT_UNKNOWN;
2714 V_UNKNOWN(value) = vb ? (IUnknown*)This->vblexicalHandler : (IUnknown*)This->lexicalHandler;
2715 if (V_UNKNOWN(value)) IUnknown_AddRef(V_UNKNOWN(value));
2716 return S_OK;
2717 }
2718
Nikolay Sivov7db7d852011-09-22 10:50:41 +04002719 if (!memcmp(PropertyDeclHandlerW, prop, sizeof(PropertyDeclHandlerW)))
2720 {
2721 V_VT(value) = VT_UNKNOWN;
2722 V_UNKNOWN(value) = vb ? (IUnknown*)This->vbdeclHandler : (IUnknown*)This->declHandler;
2723 if (V_UNKNOWN(value)) IUnknown_AddRef(V_UNKNOWN(value));
2724 return S_OK;
2725 }
2726
Nikolay Sivov864f1702011-09-21 21:01:10 +04002727 FIXME("(%p)->(%s) unsupported property\n", This, debugstr_w(prop));
2728
2729 return E_NOTIMPL;
2730}
2731
Piotr Caban19267602008-07-08 20:52:04 +02002732/*** IVBSAXXMLReader interface ***/
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002733/*** IUnknown methods ***/
2734static HRESULT WINAPI saxxmlreader_QueryInterface(IVBSAXXMLReader* iface, REFIID riid, void **ppvObject)
2735{
2736 saxreader *This = impl_from_IVBSAXXMLReader( iface );
2737
2738 TRACE("%p %s %p\n", This, debugstr_guid( riid ), ppvObject );
2739
2740 *ppvObject = NULL;
2741
2742 if ( IsEqualGUID( riid, &IID_IUnknown ) ||
2743 IsEqualGUID( riid, &IID_IDispatch ) ||
2744 IsEqualGUID( riid, &IID_IVBSAXXMLReader ))
2745 {
2746 *ppvObject = iface;
2747 }
Piotr Caban19267602008-07-08 20:52:04 +02002748 else if( IsEqualGUID( riid, &IID_ISAXXMLReader ))
2749 {
Michael Stefaniuc13e19972010-12-27 23:12:06 +01002750 *ppvObject = &This->ISAXXMLReader_iface;
Piotr Caban19267602008-07-08 20:52:04 +02002751 }
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002752 else if (dispex_query_interface(&This->dispex, riid, ppvObject))
2753 {
2754 return *ppvObject ? S_OK : E_NOINTERFACE;
2755 }
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002756 else
2757 {
2758 FIXME("interface %s not implemented\n", debugstr_guid(riid));
2759 return E_NOINTERFACE;
2760 }
2761
2762 IVBSAXXMLReader_AddRef( iface );
2763
2764 return S_OK;
2765}
2766
2767static ULONG WINAPI saxxmlreader_AddRef(IVBSAXXMLReader* iface)
2768{
2769 saxreader *This = impl_from_IVBSAXXMLReader( iface );
2770 TRACE("%p\n", This );
2771 return InterlockedIncrement( &This->ref );
2772}
2773
2774static ULONG WINAPI saxxmlreader_Release(
2775 IVBSAXXMLReader* iface)
2776{
2777 saxreader *This = impl_from_IVBSAXXMLReader( iface );
2778 LONG ref;
2779
2780 TRACE("%p\n", This );
2781
2782 ref = InterlockedDecrement( &This->ref );
2783 if ( ref == 0 )
2784 {
Piotr Caban3fa75f42008-07-08 21:00:20 +02002785 if(This->contentHandler)
2786 ISAXContentHandler_Release(This->contentHandler);
2787
Piotr Caban7b7a1872008-08-20 18:32:23 +02002788 if(This->vbcontentHandler)
2789 IVBSAXContentHandler_Release(This->vbcontentHandler);
2790
Piotr Cabanbe89ee12008-07-08 20:52:41 +02002791 if(This->errorHandler)
2792 ISAXErrorHandler_Release(This->errorHandler);
2793
Piotr Caban7b7a1872008-08-20 18:32:23 +02002794 if(This->vberrorHandler)
2795 IVBSAXErrorHandler_Release(This->vberrorHandler);
2796
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002797 if(This->lexicalHandler)
2798 ISAXLexicalHandler_Release(This->lexicalHandler);
2799
2800 if(This->vblexicalHandler)
2801 IVBSAXLexicalHandler_Release(This->vblexicalHandler);
2802
2803 if(This->declHandler)
2804 ISAXDeclHandler_Release(This->declHandler);
2805
2806 if(This->vbdeclHandler)
2807 IVBSAXDeclHandler_Release(This->vbdeclHandler);
2808
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -05002809 free_bstr_pool(&This->pool);
2810
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002811 release_dispex(&This->dispex);
Nikolay Sivovd8a9a692010-02-03 22:47:53 +03002812 heap_free( This );
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002813 }
2814
2815 return ref;
2816}
2817/*** IDispatch ***/
2818static HRESULT WINAPI saxxmlreader_GetTypeInfoCount( IVBSAXXMLReader *iface, UINT* pctinfo )
2819{
2820 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002821 return IDispatchEx_GetTypeInfoCount(&This->dispex.IDispatchEx_iface, pctinfo);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002822}
2823
2824static HRESULT WINAPI saxxmlreader_GetTypeInfo(
2825 IVBSAXXMLReader *iface,
2826 UINT iTInfo, LCID lcid, ITypeInfo** ppTInfo )
2827{
2828 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002829 return IDispatchEx_GetTypeInfo(&This->dispex.IDispatchEx_iface,
2830 iTInfo, lcid, ppTInfo);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002831}
2832
2833static HRESULT WINAPI saxxmlreader_GetIDsOfNames(
2834 IVBSAXXMLReader *iface,
2835 REFIID riid,
2836 LPOLESTR* rgszNames,
2837 UINT cNames,
2838 LCID lcid,
2839 DISPID* rgDispId)
2840{
2841 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002842 return IDispatchEx_GetIDsOfNames(&This->dispex.IDispatchEx_iface,
2843 riid, rgszNames, cNames, lcid, rgDispId);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002844}
2845
2846static HRESULT WINAPI saxxmlreader_Invoke(
2847 IVBSAXXMLReader *iface,
2848 DISPID dispIdMember,
2849 REFIID riid,
2850 LCID lcid,
2851 WORD wFlags,
2852 DISPPARAMS* pDispParams,
2853 VARIANT* pVarResult,
2854 EXCEPINFO* pExcepInfo,
2855 UINT* puArgErr)
2856{
2857 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03002858 return IDispatchEx_Invoke(&This->dispex.IDispatchEx_iface,
2859 dispIdMember, riid, lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002860}
2861
2862/*** IVBSAXXMLReader methods ***/
2863static HRESULT WINAPI saxxmlreader_getFeature(
2864 IVBSAXXMLReader* iface,
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002865 const WCHAR *feature_name,
Nikolay Sivov3d3786c2011-10-05 13:29:37 -05002866 VARIANT_BOOL *value)
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002867{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002868 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002869 saxreader_feature feature;
Piotr Caban4fb786a2008-04-01 01:45:13 +02002870
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002871 TRACE("(%p)->(%s %p)\n", This, debugstr_w(feature_name), value);
Nikolay Sivov3d3786c2011-10-05 13:29:37 -05002872
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002873 feature = get_saxreader_feature(feature_name);
2874 if (feature == Namespaces || feature == NamespacePrefixes)
2875 return get_feature_value(This, feature, value);
Nikolay Sivov9a5da562012-04-17 09:44:44 +04002876
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002877 FIXME("(%p)->(%s %p) stub\n", This, debugstr_w(feature_name), value);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002878 return E_NOTIMPL;
2879}
2880
2881static HRESULT WINAPI saxxmlreader_putFeature(
2882 IVBSAXXMLReader* iface,
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002883 const WCHAR *feature_name,
Nikolay Sivovad336be2011-10-04 16:27:39 -05002884 VARIANT_BOOL value)
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002885{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002886 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002887 saxreader_feature feature;
Piotr Caban4fb786a2008-04-01 01:45:13 +02002888
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002889 TRACE("(%p)->(%s %x)\n", This, debugstr_w(feature_name), value);
Nikolay Sivovad336be2011-10-04 16:27:39 -05002890
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002891 feature = get_saxreader_feature(feature_name);
Nikolay Sivovad336be2011-10-04 16:27:39 -05002892
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002893 /* accepted cases */
2894 if ((feature == ExternalGeneralEntities && value == VARIANT_FALSE) ||
2895 (feature == ExternalParameterEntities && value == VARIANT_FALSE) ||
Nikolay Sivov4fb58722012-04-20 11:03:18 +04002896 feature == Namespaces ||
2897 feature == NamespacePrefixes)
Nikolay Sivovcc4c6942011-10-05 09:37:40 -05002898 {
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002899 return set_feature_value(This, feature, value);
Nikolay Sivovcc4c6942011-10-05 09:37:40 -05002900 }
2901
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002902 if (feature == LexicalHandlerParEntities || feature == ProhibitDTD)
Nikolay Sivov165d1b52011-10-05 11:09:28 -05002903 {
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002904 FIXME("(%p)->(%s %x) stub\n", This, debugstr_w(feature_name), value);
2905 return set_feature_value(This, feature, value);
Nikolay Sivov165d1b52011-10-05 11:09:28 -05002906 }
2907
Nikolay Sivov137a21d2012-04-18 22:39:38 +04002908 FIXME("(%p)->(%s %x) stub\n", This, debugstr_w(feature_name), value);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002909 return E_NOTIMPL;
2910}
2911
2912static HRESULT WINAPI saxxmlreader_getProperty(
2913 IVBSAXXMLReader* iface,
Nikolay Sivov864f1702011-09-21 21:01:10 +04002914 const WCHAR *prop,
2915 VARIANT *value)
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002916{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002917 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Nikolay Sivov864f1702011-09-21 21:01:10 +04002918 return internal_getProperty(This, prop, value, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002919}
2920
2921static HRESULT WINAPI saxxmlreader_putProperty(
2922 IVBSAXXMLReader* iface,
2923 const WCHAR *pProp,
2924 VARIANT value)
2925{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002926 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabanc7fc9262008-10-01 19:52:36 +02002927 return internal_putProperty(This, pProp, value, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002928}
2929
Rob Shearman60b4fee2008-09-09 10:26:51 +01002930static HRESULT WINAPI saxxmlreader_get_entityResolver(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002931 IVBSAXXMLReader* iface,
2932 IVBSAXEntityResolver **pEntityResolver)
2933{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002934 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002935 return internal_getEntityResolver(This, pEntityResolver, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002936}
2937
Rob Shearman60b4fee2008-09-09 10:26:51 +01002938static HRESULT WINAPI saxxmlreader_put_entityResolver(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002939 IVBSAXXMLReader* iface,
2940 IVBSAXEntityResolver *pEntityResolver)
2941{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002942 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002943 return internal_putEntityResolver(This, pEntityResolver, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002944}
2945
Rob Shearman60b4fee2008-09-09 10:26:51 +01002946static HRESULT WINAPI saxxmlreader_get_contentHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002947 IVBSAXXMLReader* iface,
2948 IVBSAXContentHandler **ppContentHandler)
2949{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002950 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002951 return internal_getContentHandler(This, ppContentHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002952}
2953
Rob Shearman60b4fee2008-09-09 10:26:51 +01002954static HRESULT WINAPI saxxmlreader_put_contentHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002955 IVBSAXXMLReader* iface,
2956 IVBSAXContentHandler *contentHandler)
2957{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002958 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002959 return internal_putContentHandler(This, contentHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002960}
2961
Rob Shearman60b4fee2008-09-09 10:26:51 +01002962static HRESULT WINAPI saxxmlreader_get_dtdHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002963 IVBSAXXMLReader* iface,
2964 IVBSAXDTDHandler **pDTDHandler)
2965{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002966 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002967 return internal_getDTDHandler(This, pDTDHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002968}
2969
Rob Shearman60b4fee2008-09-09 10:26:51 +01002970static HRESULT WINAPI saxxmlreader_put_dtdHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002971 IVBSAXXMLReader* iface,
2972 IVBSAXDTDHandler *pDTDHandler)
2973{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002974 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002975 return internal_putDTDHandler(This, pDTDHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002976}
2977
Rob Shearman60b4fee2008-09-09 10:26:51 +01002978static HRESULT WINAPI saxxmlreader_get_errorHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002979 IVBSAXXMLReader* iface,
2980 IVBSAXErrorHandler **pErrorHandler)
2981{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002982 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002983 return internal_getErrorHandler(This, pErrorHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002984}
2985
Rob Shearman60b4fee2008-09-09 10:26:51 +01002986static HRESULT WINAPI saxxmlreader_put_errorHandler(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002987 IVBSAXXMLReader* iface,
2988 IVBSAXErrorHandler *errorHandler)
2989{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002990 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02002991 return internal_putErrorHandler(This, errorHandler, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002992}
2993
Rob Shearman60b4fee2008-09-09 10:26:51 +01002994static HRESULT WINAPI saxxmlreader_get_baseURL(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11002995 IVBSAXXMLReader* iface,
2996 const WCHAR **pBaseUrl)
2997{
Piotr Caban4fb786a2008-04-01 01:45:13 +02002998 saxreader *This = impl_from_IVBSAXXMLReader( iface );
2999
3000 FIXME("(%p)->(%p) stub\n", This, pBaseUrl);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003001 return E_NOTIMPL;
3002}
3003
Rob Shearman60b4fee2008-09-09 10:26:51 +01003004static HRESULT WINAPI saxxmlreader_put_baseURL(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003005 IVBSAXXMLReader* iface,
3006 const WCHAR *pBaseUrl)
3007{
Piotr Caban4fb786a2008-04-01 01:45:13 +02003008 saxreader *This = impl_from_IVBSAXXMLReader( iface );
3009
3010 FIXME("(%p)->(%s) stub\n", This, debugstr_w(pBaseUrl));
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003011 return E_NOTIMPL;
3012}
3013
Rob Shearman60b4fee2008-09-09 10:26:51 +01003014static HRESULT WINAPI saxxmlreader_get_secureBaseURL(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003015 IVBSAXXMLReader* iface,
3016 const WCHAR **pSecureBaseUrl)
3017{
Piotr Caban4fb786a2008-04-01 01:45:13 +02003018 saxreader *This = impl_from_IVBSAXXMLReader( iface );
3019
3020 FIXME("(%p)->(%p) stub\n", This, pSecureBaseUrl);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003021 return E_NOTIMPL;
3022}
3023
3024
Rob Shearman60b4fee2008-09-09 10:26:51 +01003025static HRESULT WINAPI saxxmlreader_put_secureBaseURL(
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003026 IVBSAXXMLReader* iface,
3027 const WCHAR *secureBaseUrl)
3028{
Piotr Caban4fb786a2008-04-01 01:45:13 +02003029 saxreader *This = impl_from_IVBSAXXMLReader( iface );
3030
3031 FIXME("(%p)->(%s) stub\n", This, debugstr_w(secureBaseUrl));
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003032 return E_NOTIMPL;
3033}
3034
3035static HRESULT WINAPI saxxmlreader_parse(
3036 IVBSAXXMLReader* iface,
3037 VARIANT varInput)
3038{
Piotr Caban4fb786a2008-04-01 01:45:13 +02003039 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003040 return internal_parse(This, varInput, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003041}
3042
3043static HRESULT WINAPI saxxmlreader_parseURL(
3044 IVBSAXXMLReader* iface,
3045 const WCHAR *url)
3046{
Piotr Caban4fb786a2008-04-01 01:45:13 +02003047 saxreader *This = impl_from_IVBSAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003048 return internal_parseURL(This, url, TRUE);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003049}
3050
Nikolay Sivov9d662922011-12-24 16:10:33 +03003051static const struct IVBSAXXMLReaderVtbl VBSAXXMLReaderVtbl =
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003052{
3053 saxxmlreader_QueryInterface,
3054 saxxmlreader_AddRef,
3055 saxxmlreader_Release,
3056 saxxmlreader_GetTypeInfoCount,
3057 saxxmlreader_GetTypeInfo,
3058 saxxmlreader_GetIDsOfNames,
3059 saxxmlreader_Invoke,
3060 saxxmlreader_getFeature,
3061 saxxmlreader_putFeature,
3062 saxxmlreader_getProperty,
3063 saxxmlreader_putProperty,
Rob Shearman60b4fee2008-09-09 10:26:51 +01003064 saxxmlreader_get_entityResolver,
3065 saxxmlreader_put_entityResolver,
3066 saxxmlreader_get_contentHandler,
3067 saxxmlreader_put_contentHandler,
3068 saxxmlreader_get_dtdHandler,
3069 saxxmlreader_put_dtdHandler,
3070 saxxmlreader_get_errorHandler,
3071 saxxmlreader_put_errorHandler,
3072 saxxmlreader_get_baseURL,
3073 saxxmlreader_put_baseURL,
3074 saxxmlreader_get_secureBaseURL,
3075 saxxmlreader_put_secureBaseURL,
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003076 saxxmlreader_parse,
3077 saxxmlreader_parseURL
3078};
3079
Piotr Caban19267602008-07-08 20:52:04 +02003080/*** ISAXXMLReader interface ***/
3081/*** IUnknown methods ***/
3082static HRESULT WINAPI isaxxmlreader_QueryInterface(ISAXXMLReader* iface, REFIID riid, void **ppvObject)
3083{
3084 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003085 return saxxmlreader_QueryInterface(&This->IVBSAXXMLReader_iface, riid, ppvObject);
Piotr Caban19267602008-07-08 20:52:04 +02003086}
3087
3088static ULONG WINAPI isaxxmlreader_AddRef(ISAXXMLReader* iface)
3089{
3090 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003091 return saxxmlreader_AddRef(&This->IVBSAXXMLReader_iface);
Piotr Caban19267602008-07-08 20:52:04 +02003092}
3093
3094static ULONG WINAPI isaxxmlreader_Release(ISAXXMLReader* iface)
3095{
3096 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003097 return saxxmlreader_Release(&This->IVBSAXXMLReader_iface);
Piotr Caban19267602008-07-08 20:52:04 +02003098}
3099
3100/*** ISAXXMLReader methods ***/
3101static HRESULT WINAPI isaxxmlreader_getFeature(
3102 ISAXXMLReader* iface,
3103 const WCHAR *pFeature,
3104 VARIANT_BOOL *pValue)
3105{
3106 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003107 return IVBSAXXMLReader_getFeature(&This->IVBSAXXMLReader_iface, pFeature, pValue);
Piotr Caban19267602008-07-08 20:52:04 +02003108}
3109
3110static HRESULT WINAPI isaxxmlreader_putFeature(
3111 ISAXXMLReader* iface,
3112 const WCHAR *pFeature,
3113 VARIANT_BOOL vfValue)
3114{
3115 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003116 return IVBSAXXMLReader_putFeature(&This->IVBSAXXMLReader_iface, pFeature, vfValue);
Piotr Caban19267602008-07-08 20:52:04 +02003117}
3118
3119static HRESULT WINAPI isaxxmlreader_getProperty(
3120 ISAXXMLReader* iface,
Nikolay Sivov864f1702011-09-21 21:01:10 +04003121 const WCHAR *prop,
3122 VARIANT *value)
Piotr Caban19267602008-07-08 20:52:04 +02003123{
3124 saxreader *This = impl_from_ISAXXMLReader( iface );
Nikolay Sivov864f1702011-09-21 21:01:10 +04003125 return internal_getProperty(This, prop, value, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003126}
3127
3128static HRESULT WINAPI isaxxmlreader_putProperty(
3129 ISAXXMLReader* iface,
3130 const WCHAR *pProp,
3131 VARIANT value)
3132{
3133 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabanc7fc9262008-10-01 19:52:36 +02003134 return internal_putProperty(This, pProp, value, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003135}
3136
3137static HRESULT WINAPI isaxxmlreader_getEntityResolver(
3138 ISAXXMLReader* iface,
3139 ISAXEntityResolver **ppEntityResolver)
3140{
3141 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003142 return internal_getEntityResolver(This, ppEntityResolver, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003143}
3144
3145static HRESULT WINAPI isaxxmlreader_putEntityResolver(
3146 ISAXXMLReader* iface,
3147 ISAXEntityResolver *pEntityResolver)
3148{
3149 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003150 return internal_putEntityResolver(This, pEntityResolver, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003151}
3152
3153static HRESULT WINAPI isaxxmlreader_getContentHandler(
3154 ISAXXMLReader* iface,
3155 ISAXContentHandler **pContentHandler)
3156{
3157 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003158 return internal_getContentHandler(This, pContentHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003159}
3160
3161static HRESULT WINAPI isaxxmlreader_putContentHandler(
3162 ISAXXMLReader* iface,
3163 ISAXContentHandler *contentHandler)
3164{
3165 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003166 return internal_putContentHandler(This, contentHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003167}
3168
3169static HRESULT WINAPI isaxxmlreader_getDTDHandler(
3170 ISAXXMLReader* iface,
3171 ISAXDTDHandler **pDTDHandler)
3172{
3173 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003174 return internal_getDTDHandler(This, pDTDHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003175}
3176
3177static HRESULT WINAPI isaxxmlreader_putDTDHandler(
3178 ISAXXMLReader* iface,
3179 ISAXDTDHandler *pDTDHandler)
3180{
3181 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003182 return internal_putDTDHandler(This, pDTDHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003183}
3184
3185static HRESULT WINAPI isaxxmlreader_getErrorHandler(
3186 ISAXXMLReader* iface,
3187 ISAXErrorHandler **pErrorHandler)
3188{
3189 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003190 return internal_getErrorHandler(This, pErrorHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003191}
3192
3193static HRESULT WINAPI isaxxmlreader_putErrorHandler(
3194 ISAXXMLReader* iface,
3195 ISAXErrorHandler *errorHandler)
3196{
3197 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003198 return internal_putErrorHandler(This, errorHandler, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003199}
3200
3201static HRESULT WINAPI isaxxmlreader_getBaseURL(
3202 ISAXXMLReader* iface,
3203 const WCHAR **pBaseUrl)
3204{
3205 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003206 return IVBSAXXMLReader_get_baseURL(&This->IVBSAXXMLReader_iface, pBaseUrl);
Piotr Caban19267602008-07-08 20:52:04 +02003207}
3208
3209static HRESULT WINAPI isaxxmlreader_putBaseURL(
3210 ISAXXMLReader* iface,
3211 const WCHAR *pBaseUrl)
3212{
3213 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003214 return IVBSAXXMLReader_put_baseURL(&This->IVBSAXXMLReader_iface, pBaseUrl);
Piotr Caban19267602008-07-08 20:52:04 +02003215}
3216
3217static HRESULT WINAPI isaxxmlreader_getSecureBaseURL(
3218 ISAXXMLReader* iface,
3219 const WCHAR **pSecureBaseUrl)
3220{
3221 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003222 return IVBSAXXMLReader_get_secureBaseURL(&This->IVBSAXXMLReader_iface, pSecureBaseUrl);
Piotr Caban19267602008-07-08 20:52:04 +02003223}
3224
3225static HRESULT WINAPI isaxxmlreader_putSecureBaseURL(
3226 ISAXXMLReader* iface,
3227 const WCHAR *secureBaseUrl)
3228{
3229 saxreader *This = impl_from_ISAXXMLReader( iface );
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003230 return IVBSAXXMLReader_put_secureBaseURL(&This->IVBSAXXMLReader_iface, secureBaseUrl);
Piotr Cabanb15345d2008-07-27 19:54:48 +02003231}
3232
3233static HRESULT WINAPI isaxxmlreader_parse(
3234 ISAXXMLReader* iface,
3235 VARIANT varInput)
3236{
3237 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003238 return internal_parse(This, varInput, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003239}
3240
3241static HRESULT WINAPI isaxxmlreader_parseURL(
3242 ISAXXMLReader* iface,
3243 const WCHAR *url)
3244{
3245 saxreader *This = impl_from_ISAXXMLReader( iface );
Piotr Cabancb6bcd02008-07-31 16:48:20 +02003246 return internal_parseURL(This, url, FALSE);
Piotr Caban19267602008-07-08 20:52:04 +02003247}
3248
Nikolay Sivov9d662922011-12-24 16:10:33 +03003249static const struct ISAXXMLReaderVtbl SAXXMLReaderVtbl =
Piotr Caban19267602008-07-08 20:52:04 +02003250{
3251 isaxxmlreader_QueryInterface,
3252 isaxxmlreader_AddRef,
3253 isaxxmlreader_Release,
3254 isaxxmlreader_getFeature,
3255 isaxxmlreader_putFeature,
3256 isaxxmlreader_getProperty,
3257 isaxxmlreader_putProperty,
3258 isaxxmlreader_getEntityResolver,
3259 isaxxmlreader_putEntityResolver,
3260 isaxxmlreader_getContentHandler,
3261 isaxxmlreader_putContentHandler,
3262 isaxxmlreader_getDTDHandler,
3263 isaxxmlreader_putDTDHandler,
3264 isaxxmlreader_getErrorHandler,
3265 isaxxmlreader_putErrorHandler,
3266 isaxxmlreader_getBaseURL,
3267 isaxxmlreader_putBaseURL,
3268 isaxxmlreader_getSecureBaseURL,
3269 isaxxmlreader_putSecureBaseURL,
3270 isaxxmlreader_parse,
3271 isaxxmlreader_parseURL
3272};
3273
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03003274static const tid_t saxreader_iface_tids[] = {
3275 IVBSAXXMLReader_tid,
3276 0
3277};
3278static dispex_static_data_t saxreader_dispex = {
3279 NULL,
3280 IVBSAXXMLReader_tid,
3281 NULL,
3282 saxreader_iface_tids
3283};
3284
Nikolay Sivov9d662922011-12-24 16:10:33 +03003285HRESULT SAXXMLReader_create(MSXML_VERSION version, IUnknown *outer, LPVOID *ppObj)
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003286{
3287 saxreader *reader;
3288
Nikolay Sivov9d662922011-12-24 16:10:33 +03003289 TRACE("(%p, %p)\n", outer, ppObj);
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003290
Nikolay Sivovd8a9a692010-02-03 22:47:53 +03003291 reader = heap_alloc( sizeof (*reader) );
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003292 if( !reader )
3293 return E_OUTOFMEMORY;
3294
Nikolay Sivov9d662922011-12-24 16:10:33 +03003295 reader->IVBSAXXMLReader_iface.lpVtbl = &VBSAXXMLReaderVtbl;
3296 reader->ISAXXMLReader_iface.lpVtbl = &SAXXMLReaderVtbl;
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003297 reader->ref = 1;
Piotr Caban3fa75f42008-07-08 21:00:20 +02003298 reader->contentHandler = NULL;
Piotr Caban7b7a1872008-08-20 18:32:23 +02003299 reader->vbcontentHandler = NULL;
Piotr Cabanbe89ee12008-07-08 20:52:41 +02003300 reader->errorHandler = NULL;
Piotr Caban7b7a1872008-08-20 18:32:23 +02003301 reader->vberrorHandler = NULL;
Piotr Cabanc7fc9262008-10-01 19:52:36 +02003302 reader->lexicalHandler = NULL;
3303 reader->vblexicalHandler = NULL;
3304 reader->declHandler = NULL;
3305 reader->vbdeclHandler = NULL;
3306 reader->isParsing = FALSE;
Andrew Nguyen9c5d5d82011-07-15 07:38:07 -05003307 reader->pool.pool = NULL;
3308 reader->pool.index = 0;
3309 reader->pool.len = 0;
Nikolay Sivov9a5da562012-04-17 09:44:44 +04003310 reader->features = Namespaces | NamespacePrefixes;
Piotr Cabane84c4102011-10-24 20:08:46 +02003311 reader->version = version;
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003312
Nikolay Sivov0ba0f4c2011-11-21 12:23:25 +03003313 init_dispex(&reader->dispex, (IUnknown*)&reader->IVBSAXXMLReader_iface, &saxreader_dispex);
3314
Piotr Caban51c95272008-07-17 00:40:53 +02003315 memset(&reader->sax, 0, sizeof(xmlSAXHandler));
3316 reader->sax.initialized = XML_SAX2_MAGIC;
Piotr Caban072383e2008-07-17 00:41:01 +02003317 reader->sax.startDocument = libxmlStartDocument;
Piotr Cabanc58b24b2008-07-17 00:41:19 +02003318 reader->sax.endDocument = libxmlEndDocument;
Piotr Caban965a2ea2008-07-17 00:41:27 +02003319 reader->sax.startElementNs = libxmlStartElementNS;
Piotr Caban66932632008-07-17 00:41:34 +02003320 reader->sax.endElementNs = libxmlEndElementNS;
Piotr Caban40388632008-07-17 00:41:40 +02003321 reader->sax.characters = libxmlCharacters;
Piotr Cabane47cbd12008-07-17 00:41:52 +02003322 reader->sax.setDocumentLocator = libxmlSetDocumentLocator;
Piotr Cabandb2761a2008-10-07 22:11:04 +02003323 reader->sax.comment = libxmlComment;
Piotr Cabanbba7eb52008-07-19 22:32:10 +02003324 reader->sax.error = libxmlFatalError;
3325 reader->sax.fatalError = libxmlFatalError;
Piotr Cabanb84e4782008-10-07 22:11:11 +02003326 reader->sax.cdataBlock = libxmlCDataBlock;
Piotr Caban51c95272008-07-17 00:40:53 +02003327
Michael Stefaniuc13e19972010-12-27 23:12:06 +01003328 *ppObj = &reader->IVBSAXXMLReader_iface;
Alistair Leslie-Hughesb478ec02008-03-26 14:40:01 +11003329
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003330 TRACE("returning iface %p\n", *ppObj);
Alistair Leslie-Hughesb478ec02008-03-26 14:40:01 +11003331
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003332 return S_OK;
3333}
3334
3335#else
3336
Piotr Cabane84c4102011-10-24 20:08:46 +02003337HRESULT SAXXMLReader_create(MSXML_VERSION version, IUnknown *pUnkOuter, LPVOID *ppObj)
Alistair Leslie-Hughes264be582008-03-25 14:19:10 +11003338{
3339 MESSAGE("This program tried to use a SAX XML Reader object, but\n"
3340 "libxml2 support was not present at compile time.\n");
3341 return E_NOTIMPL;
3342}
3343
3344#endif