xmllite: Initial support for just created reader state.
[wine.git] / dlls / xmllite / reader.c
blobee4a00cdf7c1795792bf38b9e6f1440d7a9dcb6a
1 /*
2 * IXmlReader implementation
4 * Copyright 2010 Nikolay Sivov
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #define COBJMACROS
23 #include <stdarg.h>
24 #include "windef.h"
25 #include "winbase.h"
26 #include "initguid.h"
27 #include "objbase.h"
28 #include "xmllite.h"
30 #include "wine/debug.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(xmllite);
34 /* not defined in public headers */
35 DEFINE_GUID(IID_IXmlReaderInput, 0x0b3ccc9b, 0x9214, 0x428b, 0xa2, 0xae, 0xef, 0x3a, 0xa8, 0x71, 0xaf, 0xda);
37 static HRESULT xmlreaderinput_query_for_stream(IXmlReaderInput *iface, void **pObj);
39 typedef struct _xmlreader
41 const IXmlReaderVtbl *lpVtbl;
42 LONG ref;
43 IXmlReaderInput *input;
44 ISequentialStream *stream;/* stored as sequential stream, cause currently
45 optimizations possible with IStream aren't implemented */
46 XmlReadState state;
47 } xmlreader;
49 typedef struct _xmlreaderinput
51 const IUnknownVtbl *lpVtbl;
52 LONG ref;
53 IUnknown *input; /* reference passed on IXmlReaderInput creation */
54 } xmlreaderinput;
56 static inline xmlreader *impl_from_IXmlReader(IXmlReader *iface)
58 return (xmlreader *)((char*)iface - FIELD_OFFSET(xmlreader, lpVtbl));
61 static inline xmlreaderinput *impl_from_IXmlReaderInput(IXmlReaderInput *iface)
63 return (xmlreaderinput *)((char*)iface - FIELD_OFFSET(xmlreaderinput, lpVtbl));
66 static HRESULT WINAPI xmlreader_QueryInterface(IXmlReader *iface, REFIID riid, void** ppvObject)
68 xmlreader *This = impl_from_IXmlReader(iface);
70 TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject);
72 if (IsEqualGUID(riid, &IID_IUnknown) ||
73 IsEqualGUID(riid, &IID_IXmlReader))
75 *ppvObject = iface;
77 else
79 FIXME("interface %s not implemented\n", debugstr_guid(riid));
80 return E_NOINTERFACE;
83 IXmlReader_AddRef(iface);
85 return S_OK;
88 static ULONG WINAPI xmlreader_AddRef(IXmlReader *iface)
90 xmlreader *This = impl_from_IXmlReader(iface);
91 TRACE("%p\n", This);
92 return InterlockedIncrement(&This->ref);
95 static ULONG WINAPI xmlreader_Release(IXmlReader *iface)
97 xmlreader *This = impl_from_IXmlReader(iface);
98 LONG ref;
100 TRACE("%p\n", This);
102 ref = InterlockedDecrement(&This->ref);
103 if (ref == 0)
105 if (This->input)
107 IUnknown_Release(This->stream);
108 IUnknown_Release(This->input);
110 HeapFree(GetProcessHeap(), 0, This);
113 return ref;
116 static HRESULT WINAPI xmlreader_SetInput(IXmlReader* iface, IUnknown *input)
118 xmlreader *This = impl_from_IXmlReader(iface);
119 HRESULT hr;
121 TRACE("(%p %p)\n", This, input);
123 if (This->input)
125 IUnknown_Release(This->input);
126 This->input = NULL;
129 if (This->stream)
131 IUnknown_Release(This->stream);
132 This->stream = NULL;
135 /* just reset current input */
136 if (!input)
138 This->state = XmlReadState_Closed;
139 return S_OK;
142 /* now try IXmlReaderInput, ISequentialStream, IStream */
143 hr = IUnknown_QueryInterface(input, &IID_IXmlReaderInput, (void**)&This->input);
144 if (hr != S_OK)
146 /* create IXmlReaderInput basing on supplied interface */
147 hr = CreateXmlReaderInputWithEncodingName(input,
148 NULL, NULL, FALSE, NULL, &This->input);
149 if (hr != S_OK) return hr;
152 /* set stream for supplied IXmlReaderInput */
153 hr = xmlreaderinput_query_for_stream(This->input, (void**)&This->stream);
154 if (hr != S_OK)
156 /* IXmlReaderInput doesn't provide streaming interface */
157 IUnknown_Release(This->input);
158 This->input = NULL;
160 else
161 This->state = XmlReadState_Initial;
163 return hr;
166 static HRESULT WINAPI xmlreader_GetProperty(IXmlReader* iface, UINT property, LONG_PTR *value)
168 xmlreader *This = impl_from_IXmlReader(iface);
170 TRACE("(%p %u %p)\n", This, property, value);
172 if (!value) return E_INVALIDARG;
174 switch (property)
176 case XmlReaderProperty_ReadState:
177 *value = This->state;
178 break;
179 default:
180 FIXME("Unimplemented property (%u)\n", property);
181 return E_NOTIMPL;
184 return S_OK;
187 static HRESULT WINAPI xmlreader_SetProperty(IXmlReader* iface, UINT property, LONG_PTR value)
189 FIXME("(%p %u %lu): stub\n", iface, property, value);
190 return E_NOTIMPL;
193 static HRESULT WINAPI xmlreader_Read(IXmlReader* iface, XmlNodeType *node_type)
195 FIXME("(%p %p): stub\n", iface, node_type);
196 return E_NOTIMPL;
199 static HRESULT WINAPI xmlreader_GetNodeType(IXmlReader* iface, XmlNodeType *node_type)
201 FIXME("(%p %p): stub\n", iface, node_type);
202 return E_NOTIMPL;
205 static HRESULT WINAPI xmlreader_MoveToFirstAttribute(IXmlReader* iface)
207 FIXME("(%p): stub\n", iface);
208 return E_NOTIMPL;
211 static HRESULT WINAPI xmlreader_MoveToNextAttribute(IXmlReader* iface)
213 FIXME("(%p): stub\n", iface);
214 return E_NOTIMPL;
217 static HRESULT WINAPI xmlreader_MoveToAttributeByName(IXmlReader* iface,
218 LPCWSTR local_name,
219 LPCWSTR namespaceUri)
221 FIXME("(%p %p %p): stub\n", iface, local_name, namespaceUri);
222 return E_NOTIMPL;
225 static HRESULT WINAPI xmlreader_MoveToElement(IXmlReader* iface)
227 FIXME("(%p): stub\n", iface);
228 return E_NOTIMPL;
231 static HRESULT WINAPI xmlreader_GetQualifiedName(IXmlReader* iface, LPCWSTR *qualifiedName,
232 UINT *qualifiedName_length)
234 FIXME("(%p %p %p): stub\n", iface, qualifiedName, qualifiedName_length);
235 return E_NOTIMPL;
238 static HRESULT WINAPI xmlreader_GetNamespaceUri(IXmlReader* iface,
239 LPCWSTR *namespaceUri,
240 UINT *namespaceUri_length)
242 FIXME("(%p %p %p): stub\n", iface, namespaceUri, namespaceUri_length);
243 return E_NOTIMPL;
246 static HRESULT WINAPI xmlreader_GetLocalName(IXmlReader* iface,
247 LPCWSTR *local_name,
248 UINT *local_name_length)
250 FIXME("(%p %p %p): stub\n", iface, local_name, local_name_length);
251 return E_NOTIMPL;
254 static HRESULT WINAPI xmlreader_GetPrefix(IXmlReader* iface,
255 LPCWSTR *prefix,
256 UINT *prefix_length)
258 FIXME("(%p %p %p): stub\n", iface, prefix, prefix_length);
259 return E_NOTIMPL;
262 static HRESULT WINAPI xmlreader_GetValue(IXmlReader* iface,
263 LPCWSTR *value,
264 UINT *value_length)
266 FIXME("(%p %p %p): stub\n", iface, value, value_length);
267 return E_NOTIMPL;
270 static HRESULT WINAPI xmlreader_ReadValueChunk(IXmlReader* iface,
271 WCHAR *buffer,
272 UINT chunk_size,
273 UINT *read)
275 FIXME("(%p %p %u %p): stub\n", iface, buffer, chunk_size, read);
276 return E_NOTIMPL;
279 static HRESULT WINAPI xmlreader_GetBaseUri(IXmlReader* iface,
280 LPCWSTR *baseUri,
281 UINT *baseUri_length)
283 FIXME("(%p %p %p): stub\n", iface, baseUri, baseUri_length);
284 return E_NOTIMPL;
287 static BOOL WINAPI xmlreader_IsDefault(IXmlReader* iface)
289 FIXME("(%p): stub\n", iface);
290 return E_NOTIMPL;
293 static BOOL WINAPI xmlreader_IsEmptyElement(IXmlReader* iface)
295 FIXME("(%p): stub\n", iface);
296 return E_NOTIMPL;
299 static HRESULT WINAPI xmlreader_GetLineNumber(IXmlReader* iface, UINT *lineNumber)
301 FIXME("(%p %p): stub\n", iface, lineNumber);
302 return E_NOTIMPL;
305 static HRESULT WINAPI xmlreader_GetLinePosition(IXmlReader* iface, UINT *linePosition)
307 FIXME("(%p %p): stub\n", iface, linePosition);
308 return E_NOTIMPL;
311 static HRESULT WINAPI xmlreader_GetAttributeCount(IXmlReader* iface, UINT *attributeCount)
313 FIXME("(%p %p): stub\n", iface, attributeCount);
314 return E_NOTIMPL;
317 static HRESULT WINAPI xmlreader_GetDepth(IXmlReader* iface, UINT *depth)
319 FIXME("(%p %p): stub\n", iface, depth);
320 return E_NOTIMPL;
323 static BOOL WINAPI xmlreader_IsEOF(IXmlReader* iface)
325 FIXME("(%p): stub\n", iface);
326 return E_NOTIMPL;
329 static const struct IXmlReaderVtbl xmlreader_vtbl =
331 xmlreader_QueryInterface,
332 xmlreader_AddRef,
333 xmlreader_Release,
334 xmlreader_SetInput,
335 xmlreader_GetProperty,
336 xmlreader_SetProperty,
337 xmlreader_Read,
338 xmlreader_GetNodeType,
339 xmlreader_MoveToFirstAttribute,
340 xmlreader_MoveToNextAttribute,
341 xmlreader_MoveToAttributeByName,
342 xmlreader_MoveToElement,
343 xmlreader_GetQualifiedName,
344 xmlreader_GetNamespaceUri,
345 xmlreader_GetLocalName,
346 xmlreader_GetPrefix,
347 xmlreader_GetValue,
348 xmlreader_ReadValueChunk,
349 xmlreader_GetBaseUri,
350 xmlreader_IsDefault,
351 xmlreader_IsEmptyElement,
352 xmlreader_GetLineNumber,
353 xmlreader_GetLinePosition,
354 xmlreader_GetAttributeCount,
355 xmlreader_GetDepth,
356 xmlreader_IsEOF
359 /** IXmlReaderInput **/
361 /* Queries already stored interface for IStream/ISequentialStream.
362 Interface supplied on creation will be overwritten */
363 static HRESULT xmlreaderinput_query_for_stream(IXmlReaderInput *iface, void **pObj)
365 xmlreaderinput *This = impl_from_IXmlReaderInput(iface);
366 HRESULT hr;
368 hr = IUnknown_QueryInterface(This->input, &IID_IStream, pObj);
369 if (hr != S_OK)
370 hr = IUnknown_QueryInterface(This->input, &IID_ISequentialStream, pObj);
372 return hr;
375 static HRESULT WINAPI xmlreaderinput_QueryInterface(IXmlReaderInput *iface, REFIID riid, void** ppvObject)
377 xmlreaderinput *This = impl_from_IXmlReaderInput(iface);
379 TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject);
381 if (IsEqualGUID(riid, &IID_IXmlReaderInput) ||
382 IsEqualGUID(riid, &IID_IUnknown))
384 *ppvObject = iface;
386 else
388 FIXME("interface %s not implemented\n", debugstr_guid(riid));
389 return E_NOINTERFACE;
392 IUnknown_AddRef(iface);
394 return S_OK;
397 static ULONG WINAPI xmlreaderinput_AddRef(IXmlReaderInput *iface)
399 xmlreaderinput *This = impl_from_IXmlReaderInput(iface);
400 TRACE("%p\n", This);
401 return InterlockedIncrement(&This->ref);
404 static ULONG WINAPI xmlreaderinput_Release(IXmlReaderInput *iface)
406 xmlreaderinput *This = impl_from_IXmlReaderInput(iface);
407 LONG ref;
409 TRACE("%p\n", This);
411 ref = InterlockedDecrement(&This->ref);
412 if (ref == 0)
414 if (This->input) IUnknown_Release(This->input);
415 HeapFree(GetProcessHeap(), 0, This);
418 return ref;
421 static const struct IUnknownVtbl xmlreaderinput_vtbl =
423 xmlreaderinput_QueryInterface,
424 xmlreaderinput_AddRef,
425 xmlreaderinput_Release
428 HRESULT WINAPI CreateXmlReader(REFIID riid, void **pObject, IMalloc *pMalloc)
430 xmlreader *reader;
432 TRACE("(%s, %p, %p)\n", wine_dbgstr_guid(riid), pObject, pMalloc);
434 if (pMalloc) FIXME("custom IMalloc not supported yet\n");
436 if (!IsEqualGUID(riid, &IID_IXmlReader))
438 ERR("Unexpected IID requested -> (%s)\n", wine_dbgstr_guid(riid));
439 return E_FAIL;
442 reader = HeapAlloc(GetProcessHeap(), 0, sizeof (*reader));
443 if(!reader) return E_OUTOFMEMORY;
445 reader->lpVtbl = &xmlreader_vtbl;
446 reader->ref = 1;
447 reader->stream = NULL;
448 reader->input = NULL;
449 reader->state = XmlReadState_Closed;
451 *pObject = &reader->lpVtbl;
453 TRACE("returning iface %p\n", *pObject);
455 return S_OK;
458 HRESULT WINAPI CreateXmlReaderInputWithEncodingName(IUnknown *stream,
459 IMalloc *pMalloc,
460 LPCWSTR encoding,
461 BOOL hint,
462 LPCWSTR base_uri,
463 IXmlReaderInput **ppInput)
465 xmlreaderinput *readerinput;
467 FIXME("%p %p %s %d %s %p: stub\n", stream, pMalloc, wine_dbgstr_w(encoding),
468 hint, wine_dbgstr_w(base_uri), ppInput);
470 if (!stream || !ppInput) return E_INVALIDARG;
472 readerinput = HeapAlloc(GetProcessHeap(), 0, sizeof (*readerinput));
473 if(!readerinput) return E_OUTOFMEMORY;
475 readerinput->lpVtbl = &xmlreaderinput_vtbl;
476 readerinput->ref = 1;
477 IUnknown_QueryInterface(stream, &IID_IUnknown, (void**)&readerinput->input);
479 *ppInput = (IXmlReaderInput*)&readerinput->lpVtbl;
481 TRACE("returning iface %p\n", *ppInput);
483 return S_OK;