| /* |
| * IXmlReader implementation |
| * |
| * Copyright 2010 Nikolay Sivov |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #define COBJMACROS |
| |
| #include <stdarg.h> |
| #include "windef.h" |
| #include "winbase.h" |
| #include "initguid.h" |
| #include "objbase.h" |
| #include "xmllite.h" |
| |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(xmllite); |
| |
| /* not defined in public headers */ |
| DEFINE_GUID(IID_IXmlReaderInput, 0x0b3ccc9b, 0x9214, 0x428b, 0xa2, 0xae, 0xef, 0x3a, 0xa8, 0x71, 0xaf, 0xda); |
| |
| static HRESULT xmlreaderinput_query_for_stream(IXmlReaderInput *iface, void **pObj); |
| |
| typedef struct _xmlreader |
| { |
| IXmlReader IXmlReader_iface; |
| LONG ref; |
| IXmlReaderInput *input; |
| ISequentialStream *stream;/* stored as sequential stream, cause currently |
| optimizations possible with IStream aren't implemented */ |
| XmlReadState state; |
| DtdProcessing dtdmode; |
| UINT line, pos; /* reader position in XML stream */ |
| } xmlreader; |
| |
| typedef struct _xmlreaderinput |
| { |
| IXmlReaderInput IXmlReaderInput_iface; |
| LONG ref; |
| IUnknown *input; /* reference passed on IXmlReaderInput creation */ |
| } xmlreaderinput; |
| |
| static inline xmlreader *impl_from_IXmlReader(IXmlReader *iface) |
| { |
| return CONTAINING_RECORD(iface, xmlreader, IXmlReader_iface); |
| } |
| |
| static inline xmlreaderinput *impl_from_IXmlReaderInput(IXmlReaderInput *iface) |
| { |
| return CONTAINING_RECORD(iface, xmlreaderinput, IXmlReaderInput_iface); |
| } |
| |
| static HRESULT WINAPI xmlreader_QueryInterface(IXmlReader *iface, REFIID riid, void** ppvObject) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| |
| TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject); |
| |
| if (IsEqualGUID(riid, &IID_IUnknown) || |
| IsEqualGUID(riid, &IID_IXmlReader)) |
| { |
| *ppvObject = iface; |
| } |
| else |
| { |
| FIXME("interface %s not implemented\n", debugstr_guid(riid)); |
| return E_NOINTERFACE; |
| } |
| |
| IXmlReader_AddRef(iface); |
| |
| return S_OK; |
| } |
| |
| static ULONG WINAPI xmlreader_AddRef(IXmlReader *iface) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| TRACE("%p\n", This); |
| return InterlockedIncrement(&This->ref); |
| } |
| |
| static ULONG WINAPI xmlreader_Release(IXmlReader *iface) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| LONG ref; |
| |
| TRACE("%p\n", This); |
| |
| ref = InterlockedDecrement(&This->ref); |
| if (ref == 0) |
| { |
| if (This->input) IUnknown_Release(This->input); |
| if (This->stream) ISequentialStream_Release(This->stream); |
| HeapFree(GetProcessHeap(), 0, This); |
| } |
| |
| return ref; |
| } |
| |
| static HRESULT WINAPI xmlreader_SetInput(IXmlReader* iface, IUnknown *input) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| HRESULT hr; |
| |
| TRACE("(%p %p)\n", This, input); |
| |
| if (This->input) |
| { |
| IUnknown_Release(This->input); |
| This->input = NULL; |
| } |
| |
| if (This->stream) |
| { |
| ISequentialStream_Release(This->stream); |
| This->stream = NULL; |
| } |
| |
| This->line = This->pos = 0; |
| |
| /* just reset current input */ |
| if (!input) |
| { |
| This->state = XmlReadState_Initial; |
| return S_OK; |
| } |
| |
| /* now try IXmlReaderInput, ISequentialStream, IStream */ |
| hr = IUnknown_QueryInterface(input, &IID_IXmlReaderInput, (void**)&This->input); |
| if (hr != S_OK) |
| { |
| /* create IXmlReaderInput basing on supplied interface */ |
| hr = CreateXmlReaderInputWithEncodingName(input, |
| NULL, NULL, FALSE, NULL, &This->input); |
| if (hr != S_OK) return hr; |
| } |
| |
| /* set stream for supplied IXmlReaderInput */ |
| hr = xmlreaderinput_query_for_stream(This->input, (void**)&This->stream); |
| if (hr == S_OK) |
| This->state = XmlReadState_Initial; |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetProperty(IXmlReader* iface, UINT property, LONG_PTR *value) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| |
| TRACE("(%p %u %p)\n", This, property, value); |
| |
| if (!value) return E_INVALIDARG; |
| |
| switch (property) |
| { |
| case XmlReaderProperty_DtdProcessing: |
| *value = This->dtdmode; |
| break; |
| case XmlReaderProperty_ReadState: |
| *value = This->state; |
| break; |
| default: |
| FIXME("Unimplemented property (%u)\n", property); |
| return E_NOTIMPL; |
| } |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI xmlreader_SetProperty(IXmlReader* iface, UINT property, LONG_PTR value) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| |
| TRACE("(%p %u %lu)\n", iface, property, value); |
| |
| switch (property) |
| { |
| case XmlReaderProperty_DtdProcessing: |
| if (value < 0 || value > _DtdProcessing_Last) return E_INVALIDARG; |
| This->dtdmode = value; |
| break; |
| default: |
| FIXME("Unimplemented property (%u)\n", property); |
| return E_NOTIMPL; |
| } |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI xmlreader_Read(IXmlReader* iface, XmlNodeType *node_type) |
| { |
| FIXME("(%p %p): stub\n", iface, node_type); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetNodeType(IXmlReader* iface, XmlNodeType *node_type) |
| { |
| FIXME("(%p %p): stub\n", iface, node_type); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_MoveToFirstAttribute(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_MoveToNextAttribute(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_MoveToAttributeByName(IXmlReader* iface, |
| LPCWSTR local_name, |
| LPCWSTR namespaceUri) |
| { |
| FIXME("(%p %p %p): stub\n", iface, local_name, namespaceUri); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_MoveToElement(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetQualifiedName(IXmlReader* iface, LPCWSTR *qualifiedName, |
| UINT *qualifiedName_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, qualifiedName, qualifiedName_length); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetNamespaceUri(IXmlReader* iface, |
| LPCWSTR *namespaceUri, |
| UINT *namespaceUri_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, namespaceUri, namespaceUri_length); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetLocalName(IXmlReader* iface, |
| LPCWSTR *local_name, |
| UINT *local_name_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, local_name, local_name_length); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetPrefix(IXmlReader* iface, |
| LPCWSTR *prefix, |
| UINT *prefix_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, prefix, prefix_length); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetValue(IXmlReader* iface, |
| LPCWSTR *value, |
| UINT *value_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, value, value_length); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_ReadValueChunk(IXmlReader* iface, |
| WCHAR *buffer, |
| UINT chunk_size, |
| UINT *read) |
| { |
| FIXME("(%p %p %u %p): stub\n", iface, buffer, chunk_size, read); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetBaseUri(IXmlReader* iface, |
| LPCWSTR *baseUri, |
| UINT *baseUri_length) |
| { |
| FIXME("(%p %p %p): stub\n", iface, baseUri, baseUri_length); |
| return E_NOTIMPL; |
| } |
| |
| static BOOL WINAPI xmlreader_IsDefault(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static BOOL WINAPI xmlreader_IsEmptyElement(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetLineNumber(IXmlReader* iface, UINT *lineNumber) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| |
| TRACE("(%p %p)\n", This, lineNumber); |
| |
| if (!lineNumber) return E_INVALIDARG; |
| |
| *lineNumber = This->line; |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetLinePosition(IXmlReader* iface, UINT *linePosition) |
| { |
| xmlreader *This = impl_from_IXmlReader(iface); |
| |
| TRACE("(%p %p)\n", This, linePosition); |
| |
| if (!linePosition) return E_INVALIDARG; |
| |
| *linePosition = This->pos; |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetAttributeCount(IXmlReader* iface, UINT *attributeCount) |
| { |
| FIXME("(%p %p): stub\n", iface, attributeCount); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI xmlreader_GetDepth(IXmlReader* iface, UINT *depth) |
| { |
| FIXME("(%p %p): stub\n", iface, depth); |
| return E_NOTIMPL; |
| } |
| |
| static BOOL WINAPI xmlreader_IsEOF(IXmlReader* iface) |
| { |
| FIXME("(%p): stub\n", iface); |
| return E_NOTIMPL; |
| } |
| |
| static const struct IXmlReaderVtbl xmlreader_vtbl = |
| { |
| xmlreader_QueryInterface, |
| xmlreader_AddRef, |
| xmlreader_Release, |
| xmlreader_SetInput, |
| xmlreader_GetProperty, |
| xmlreader_SetProperty, |
| xmlreader_Read, |
| xmlreader_GetNodeType, |
| xmlreader_MoveToFirstAttribute, |
| xmlreader_MoveToNextAttribute, |
| xmlreader_MoveToAttributeByName, |
| xmlreader_MoveToElement, |
| xmlreader_GetQualifiedName, |
| xmlreader_GetNamespaceUri, |
| xmlreader_GetLocalName, |
| xmlreader_GetPrefix, |
| xmlreader_GetValue, |
| xmlreader_ReadValueChunk, |
| xmlreader_GetBaseUri, |
| xmlreader_IsDefault, |
| xmlreader_IsEmptyElement, |
| xmlreader_GetLineNumber, |
| xmlreader_GetLinePosition, |
| xmlreader_GetAttributeCount, |
| xmlreader_GetDepth, |
| xmlreader_IsEOF |
| }; |
| |
| /** IXmlReaderInput **/ |
| |
| /* Queries already stored interface for IStream/ISequentialStream. |
| Interface supplied on creation will be overwritten */ |
| static HRESULT xmlreaderinput_query_for_stream(IXmlReaderInput *iface, void **pObj) |
| { |
| xmlreaderinput *This = impl_from_IXmlReaderInput(iface); |
| HRESULT hr; |
| |
| hr = IUnknown_QueryInterface(This->input, &IID_IStream, pObj); |
| if (hr != S_OK) |
| hr = IUnknown_QueryInterface(This->input, &IID_ISequentialStream, pObj); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI xmlreaderinput_QueryInterface(IXmlReaderInput *iface, REFIID riid, void** ppvObject) |
| { |
| xmlreaderinput *This = impl_from_IXmlReaderInput(iface); |
| |
| TRACE("%p %s %p\n", This, debugstr_guid(riid), ppvObject); |
| |
| if (IsEqualGUID(riid, &IID_IXmlReaderInput) || |
| IsEqualGUID(riid, &IID_IUnknown)) |
| { |
| *ppvObject = iface; |
| } |
| else |
| { |
| FIXME("interface %s not implemented\n", debugstr_guid(riid)); |
| return E_NOINTERFACE; |
| } |
| |
| IUnknown_AddRef(iface); |
| |
| return S_OK; |
| } |
| |
| static ULONG WINAPI xmlreaderinput_AddRef(IXmlReaderInput *iface) |
| { |
| xmlreaderinput *This = impl_from_IXmlReaderInput(iface); |
| TRACE("%p\n", This); |
| return InterlockedIncrement(&This->ref); |
| } |
| |
| static ULONG WINAPI xmlreaderinput_Release(IXmlReaderInput *iface) |
| { |
| xmlreaderinput *This = impl_from_IXmlReaderInput(iface); |
| LONG ref; |
| |
| TRACE("%p\n", This); |
| |
| ref = InterlockedDecrement(&This->ref); |
| if (ref == 0) |
| { |
| if (This->input) IUnknown_Release(This->input); |
| HeapFree(GetProcessHeap(), 0, This); |
| } |
| |
| return ref; |
| } |
| |
| static const struct IUnknownVtbl xmlreaderinput_vtbl = |
| { |
| xmlreaderinput_QueryInterface, |
| xmlreaderinput_AddRef, |
| xmlreaderinput_Release |
| }; |
| |
| HRESULT WINAPI CreateXmlReader(REFIID riid, void **pObject, IMalloc *pMalloc) |
| { |
| xmlreader *reader; |
| |
| TRACE("(%s, %p, %p)\n", wine_dbgstr_guid(riid), pObject, pMalloc); |
| |
| if (pMalloc) FIXME("custom IMalloc not supported yet\n"); |
| |
| if (!IsEqualGUID(riid, &IID_IXmlReader)) |
| { |
| ERR("Unexpected IID requested -> (%s)\n", wine_dbgstr_guid(riid)); |
| return E_FAIL; |
| } |
| |
| reader = HeapAlloc(GetProcessHeap(), 0, sizeof (*reader)); |
| if(!reader) return E_OUTOFMEMORY; |
| |
| reader->IXmlReader_iface.lpVtbl = &xmlreader_vtbl; |
| reader->ref = 1; |
| reader->stream = NULL; |
| reader->input = NULL; |
| reader->state = XmlReadState_Closed; |
| reader->dtdmode = DtdProcessing_Prohibit; |
| reader->line = reader->pos = 0; |
| |
| *pObject = &reader->IXmlReader_iface; |
| |
| TRACE("returning iface %p\n", *pObject); |
| |
| return S_OK; |
| } |
| |
| HRESULT WINAPI CreateXmlReaderInputWithEncodingName(IUnknown *stream, |
| IMalloc *pMalloc, |
| LPCWSTR encoding, |
| BOOL hint, |
| LPCWSTR base_uri, |
| IXmlReaderInput **ppInput) |
| { |
| xmlreaderinput *readerinput; |
| |
| FIXME("%p %p %s %d %s %p: stub\n", stream, pMalloc, wine_dbgstr_w(encoding), |
| hint, wine_dbgstr_w(base_uri), ppInput); |
| |
| if (!stream || !ppInput) return E_INVALIDARG; |
| |
| readerinput = HeapAlloc(GetProcessHeap(), 0, sizeof (*readerinput)); |
| if(!readerinput) return E_OUTOFMEMORY; |
| |
| readerinput->IXmlReaderInput_iface.lpVtbl = &xmlreaderinput_vtbl; |
| readerinput->ref = 1; |
| IUnknown_QueryInterface(stream, &IID_IUnknown, (void**)&readerinput->input); |
| |
| *ppInput = &readerinput->IXmlReaderInput_iface; |
| |
| TRACE("returning iface %p\n", *ppInput); |
| |
| return S_OK; |
| } |