ntdll: Check and fix PE header protections.
[wine] / dlls / msxml3 / docfrag.c
1 /*
2  *    DOM Document Fragment implementation
3  *
4  * Copyright 2007 Alistair Leslie-Hughes
5  *
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.
10  *
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.
15  *
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
19  */
20
21 #define COBJMACROS
22
23 #include "config.h"
24
25 #include <stdarg.h>
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winuser.h"
29 #include "ole2.h"
30 #include "msxml6.h"
31
32 #include "msxml_private.h"
33
34 #include "wine/debug.h"
35
36 WINE_DEFAULT_DEBUG_CHANNEL(msxml);
37
38 #ifdef HAVE_LIBXML2
39
40 typedef struct _domfrag
41 {
42     xmlnode node;
43     const struct IXMLDOMDocumentFragmentVtbl *lpVtbl;
44     LONG ref;
45 } domfrag;
46
47 static inline domfrag *impl_from_IXMLDOMDocumentFragment( IXMLDOMDocumentFragment *iface )
48 {
49     return (domfrag *)((char*)iface - FIELD_OFFSET(domfrag, lpVtbl));
50 }
51
52 static HRESULT WINAPI domfrag_QueryInterface(
53     IXMLDOMDocumentFragment *iface,
54     REFIID riid,
55     void** ppvObject )
56 {
57     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
58     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject);
59
60     if ( IsEqualGUID( riid, &IID_IXMLDOMDocumentFragment ) ||
61          IsEqualGUID( riid, &IID_IXMLDOMNode ) ||
62          IsEqualGUID( riid, &IID_IDispatch ) ||
63          IsEqualGUID( riid, &IID_IUnknown ) )
64     {
65         *ppvObject = iface;
66     }
67     else if(node_query_interface(&This->node, riid, ppvObject))
68     {
69         return *ppvObject ? S_OK : E_NOINTERFACE;
70     }
71     else
72     {
73         FIXME("Unsupported interface %s\n", debugstr_guid(riid));
74         return E_NOINTERFACE;
75     }
76
77     IXMLDOMText_AddRef((IUnknown*)*ppvObject);
78     return S_OK;
79 }
80
81 static ULONG WINAPI domfrag_AddRef(
82     IXMLDOMDocumentFragment *iface )
83 {
84     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
85     return InterlockedIncrement( &This->ref );
86 }
87
88 static ULONG WINAPI domfrag_Release(
89     IXMLDOMDocumentFragment *iface )
90 {
91     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
92     ULONG ref;
93
94     ref = InterlockedDecrement( &This->ref );
95     if ( ref == 0 )
96     {
97         destroy_xmlnode(&This->node);
98         heap_free( This );
99     }
100
101     return ref;
102 }
103
104 static HRESULT WINAPI domfrag_GetTypeInfoCount(
105     IXMLDOMDocumentFragment *iface,
106     UINT* pctinfo )
107 {
108     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
109
110     TRACE("(%p)->(%p)\n", This, pctinfo);
111
112     *pctinfo = 1;
113
114     return S_OK;
115 }
116
117 static HRESULT WINAPI domfrag_GetTypeInfo(
118     IXMLDOMDocumentFragment *iface,
119     UINT iTInfo, LCID lcid,
120     ITypeInfo** ppTInfo )
121 {
122     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
123     HRESULT hr;
124
125     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
126
127     hr = get_typeinfo(IXMLDOMDocumentFragment_tid, ppTInfo);
128
129     return hr;
130 }
131
132 static HRESULT WINAPI domfrag_GetIDsOfNames(
133     IXMLDOMDocumentFragment *iface,
134     REFIID riid, LPOLESTR* rgszNames,
135     UINT cNames, LCID lcid, DISPID* rgDispId )
136 {
137     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
138     ITypeInfo *typeinfo;
139     HRESULT hr;
140
141     TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
142           lcid, rgDispId);
143
144     if(!rgszNames || cNames == 0 || !rgDispId)
145         return E_INVALIDARG;
146
147     hr = get_typeinfo(IXMLDOMDocumentFragment_tid, &typeinfo);
148     if(SUCCEEDED(hr))
149     {
150         hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
151         ITypeInfo_Release(typeinfo);
152     }
153
154     return hr;
155 }
156
157 static HRESULT WINAPI domfrag_Invoke(
158     IXMLDOMDocumentFragment *iface,
159     DISPID dispIdMember, REFIID riid, LCID lcid,
160     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
161     EXCEPINFO* pExcepInfo, UINT* puArgErr )
162 {
163     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
164     ITypeInfo *typeinfo;
165     HRESULT hr;
166
167     TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
168           lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
169
170     hr = get_typeinfo(IXMLDOMDocumentFragment_tid, &typeinfo);
171     if(SUCCEEDED(hr))
172     {
173         hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
174                 pVarResult, pExcepInfo, puArgErr);
175         ITypeInfo_Release(typeinfo);
176     }
177
178     return hr;
179 }
180
181 static HRESULT WINAPI domfrag_get_nodeName(
182     IXMLDOMDocumentFragment *iface,
183     BSTR* p )
184 {
185     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
186
187     static const WCHAR document_fragmentW[] =
188         {'#','d','o','c','u','m','e','n','t','-','f','r','a','g','m','e','n','t',0};
189
190     TRACE("(%p)->(%p)\n", This, p);
191
192     return return_bstr(document_fragmentW, p);
193 }
194
195 static HRESULT WINAPI domfrag_get_nodeValue(
196     IXMLDOMDocumentFragment *iface,
197     VARIANT* value)
198 {
199     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
200
201     FIXME("(%p)->(%p)\n", This, value);
202
203     if(!value)
204         return E_INVALIDARG;
205
206     V_VT(value) = VT_NULL;
207     return S_FALSE;
208 }
209
210 static HRESULT WINAPI domfrag_put_nodeValue(
211     IXMLDOMDocumentFragment *iface,
212     VARIANT value)
213 {
214     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
215     TRACE("(%p)->(v%d)\n", This, V_VT(&value));
216     return E_FAIL;
217 }
218
219 static HRESULT WINAPI domfrag_get_nodeType(
220     IXMLDOMDocumentFragment *iface,
221     DOMNodeType* domNodeType )
222 {
223     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
224
225     TRACE("(%p)->(%p)\n", This, domNodeType);
226
227     *domNodeType = NODE_DOCUMENT_FRAGMENT;
228     return S_OK;
229 }
230
231 static HRESULT WINAPI domfrag_get_parentNode(
232     IXMLDOMDocumentFragment *iface,
233     IXMLDOMNode** parent )
234 {
235     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
236
237     TRACE("(%p)->(%p)\n", This, parent);
238
239     return node_get_parent(&This->node, parent);
240 }
241
242 static HRESULT WINAPI domfrag_get_childNodes(
243     IXMLDOMDocumentFragment *iface,
244     IXMLDOMNodeList** outList)
245 {
246     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
247
248     TRACE("(%p)->(%p)\n", This, outList);
249
250     return node_get_child_nodes(&This->node, outList);
251 }
252
253 static HRESULT WINAPI domfrag_get_firstChild(
254     IXMLDOMDocumentFragment *iface,
255     IXMLDOMNode** domNode)
256 {
257     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
258
259     TRACE("(%p)->(%p)\n", This, domNode);
260
261     return node_get_first_child(&This->node, domNode);
262 }
263
264 static HRESULT WINAPI domfrag_get_lastChild(
265     IXMLDOMDocumentFragment *iface,
266     IXMLDOMNode** domNode)
267 {
268     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
269
270     TRACE("(%p)->(%p)\n", This, domNode);
271
272     return node_get_last_child(&This->node, domNode);
273 }
274
275 static HRESULT WINAPI domfrag_get_previousSibling(
276     IXMLDOMDocumentFragment *iface,
277     IXMLDOMNode** domNode)
278 {
279     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
280
281     TRACE("(%p)->(%p)\n", This, domNode);
282
283     return return_null_node(domNode);
284 }
285
286 static HRESULT WINAPI domfrag_get_nextSibling(
287     IXMLDOMDocumentFragment *iface,
288     IXMLDOMNode** domNode)
289 {
290     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
291
292     TRACE("(%p)->(%p)\n", This, domNode);
293
294     return return_null_node(domNode);
295 }
296
297 static HRESULT WINAPI domfrag_get_attributes(
298     IXMLDOMDocumentFragment *iface,
299     IXMLDOMNamedNodeMap** attributeMap)
300 {
301     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
302
303     TRACE("(%p)->(%p)\n", This, attributeMap);
304
305     return return_null_ptr((void**)attributeMap);
306 }
307
308 static HRESULT WINAPI domfrag_insertBefore(
309     IXMLDOMDocumentFragment *iface,
310     IXMLDOMNode* newNode, VARIANT refChild,
311     IXMLDOMNode** outOldNode)
312 {
313     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
314
315     TRACE("(%p)->(%p x%d %p)\n", This, newNode, V_VT(&refChild), outOldNode);
316
317     /* TODO: test */
318     return node_insert_before(&This->node, newNode, &refChild, outOldNode);
319 }
320
321 static HRESULT WINAPI domfrag_replaceChild(
322     IXMLDOMDocumentFragment *iface,
323     IXMLDOMNode* newNode,
324     IXMLDOMNode* oldNode,
325     IXMLDOMNode** outOldNode)
326 {
327     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
328
329     TRACE("(%p)->(%p %p %p)\n", This, newNode, oldNode, outOldNode);
330
331     /* TODO: test */
332     return node_replace_child(&This->node, newNode, oldNode, outOldNode);
333 }
334
335 static HRESULT WINAPI domfrag_removeChild(
336     IXMLDOMDocumentFragment *iface,
337     IXMLDOMNode* domNode, IXMLDOMNode** oldNode)
338 {
339     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
340     return IXMLDOMNode_removeChild( IXMLDOMNode_from_impl(&This->node), domNode, oldNode );
341 }
342
343 static HRESULT WINAPI domfrag_appendChild(
344     IXMLDOMDocumentFragment *iface,
345     IXMLDOMNode* newNode, IXMLDOMNode** outNewNode)
346 {
347     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
348     return IXMLDOMNode_appendChild( IXMLDOMNode_from_impl(&This->node), newNode, outNewNode );
349 }
350
351 static HRESULT WINAPI domfrag_hasChildNodes(
352     IXMLDOMDocumentFragment *iface,
353     VARIANT_BOOL* pbool)
354 {
355     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
356     return IXMLDOMNode_hasChildNodes( IXMLDOMNode_from_impl(&This->node), pbool );
357 }
358
359 static HRESULT WINAPI domfrag_get_ownerDocument(
360     IXMLDOMDocumentFragment *iface,
361     IXMLDOMDocument** domDocument)
362 {
363     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
364     return IXMLDOMNode_get_ownerDocument( IXMLDOMNode_from_impl(&This->node), domDocument );
365 }
366
367 static HRESULT WINAPI domfrag_cloneNode(
368     IXMLDOMDocumentFragment *iface,
369     VARIANT_BOOL deep, IXMLDOMNode** outNode)
370 {
371     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
372     TRACE("(%p)->(%d %p)\n", This, deep, outNode);
373     return node_clone( &This->node, deep, outNode );
374 }
375
376 static HRESULT WINAPI domfrag_get_nodeTypeString(
377     IXMLDOMDocumentFragment *iface,
378     BSTR* p)
379 {
380     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
381     static const WCHAR documentfragmentW[] = {'d','o','c','u','m','e','n','t','f','r','a','g','m','e','n','t',0};
382
383     TRACE("(%p)->(%p)\n", This, p);
384
385     return return_bstr(documentfragmentW, p);
386 }
387
388 static HRESULT WINAPI domfrag_get_text(
389     IXMLDOMDocumentFragment *iface,
390     BSTR* p)
391 {
392     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
393     return IXMLDOMNode_get_text( IXMLDOMNode_from_impl(&This->node), p );
394 }
395
396 static HRESULT WINAPI domfrag_put_text(
397     IXMLDOMDocumentFragment *iface,
398     BSTR p)
399 {
400     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
401     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
402     return node_put_text( &This->node, p );
403 }
404
405 static HRESULT WINAPI domfrag_get_specified(
406     IXMLDOMDocumentFragment *iface,
407     VARIANT_BOOL* isSpecified)
408 {
409     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
410     FIXME("(%p)->(%p) stub!\n", This, isSpecified);
411     *isSpecified = VARIANT_TRUE;
412     return S_OK;
413 }
414
415 static HRESULT WINAPI domfrag_get_definition(
416     IXMLDOMDocumentFragment *iface,
417     IXMLDOMNode** definitionNode)
418 {
419     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
420     FIXME("(%p)->(%p)\n", This, definitionNode);
421     return E_NOTIMPL;
422 }
423
424 static HRESULT WINAPI domfrag_get_nodeTypedValue(
425     IXMLDOMDocumentFragment *iface,
426     VARIANT* var1)
427 {
428     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
429     return IXMLDOMNode_get_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
430 }
431
432 static HRESULT WINAPI domfrag_put_nodeTypedValue(
433     IXMLDOMDocumentFragment *iface,
434     VARIANT var1)
435 {
436     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
437     return IXMLDOMNode_put_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
438 }
439
440 static HRESULT WINAPI domfrag_get_dataType(
441     IXMLDOMDocumentFragment *iface,
442     VARIANT* typename)
443 {
444     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
445     TRACE("(%p)->(%p)\n", This, typename);
446     return return_null_var( typename );
447 }
448
449 static HRESULT WINAPI domfrag_put_dataType(
450     IXMLDOMDocumentFragment *iface,
451     BSTR p)
452 {
453     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
454     return IXMLDOMNode_put_dataType( IXMLDOMNode_from_impl(&This->node), p );
455 }
456
457 static HRESULT WINAPI domfrag_get_xml(
458     IXMLDOMDocumentFragment *iface,
459     BSTR* p)
460 {
461     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
462
463     TRACE("(%p)->(%p)\n", This, p);
464
465     return node_get_xml(&This->node, FALSE, FALSE, p);
466 }
467
468 static HRESULT WINAPI domfrag_transformNode(
469     IXMLDOMDocumentFragment *iface,
470     IXMLDOMNode* domNode, BSTR* p)
471 {
472     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
473     return IXMLDOMNode_transformNode( IXMLDOMNode_from_impl(&This->node), domNode, p );
474 }
475
476 static HRESULT WINAPI domfrag_selectNodes(
477     IXMLDOMDocumentFragment *iface,
478     BSTR p, IXMLDOMNodeList** outList)
479 {
480     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
481     return IXMLDOMNode_selectNodes( IXMLDOMNode_from_impl(&This->node), p, outList );
482 }
483
484 static HRESULT WINAPI domfrag_selectSingleNode(
485     IXMLDOMDocumentFragment *iface,
486     BSTR p, IXMLDOMNode** outNode)
487 {
488     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
489     return IXMLDOMNode_selectSingleNode( IXMLDOMNode_from_impl(&This->node), p, outNode );
490 }
491
492 static HRESULT WINAPI domfrag_get_parsed(
493     IXMLDOMDocumentFragment *iface,
494     VARIANT_BOOL* isParsed)
495 {
496     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
497     FIXME("(%p)->(%p) stub!\n", This, isParsed);
498     *isParsed = VARIANT_TRUE;
499     return S_OK;
500 }
501
502 static HRESULT WINAPI domfrag_get_namespaceURI(
503     IXMLDOMDocumentFragment *iface,
504     BSTR* p)
505 {
506     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
507     return IXMLDOMNode_get_namespaceURI( IXMLDOMNode_from_impl(&This->node), p );
508 }
509
510 static HRESULT WINAPI domfrag_get_prefix(
511     IXMLDOMDocumentFragment *iface,
512     BSTR* prefix)
513 {
514     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
515     TRACE("(%p)->(%p)\n", This, prefix);
516     return return_null_bstr( prefix );
517 }
518
519 static HRESULT WINAPI domfrag_get_baseName(
520     IXMLDOMDocumentFragment *iface,
521     BSTR* name)
522 {
523     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
524     FIXME("(%p)->(%p): needs test\n", This, name);
525     return return_null_bstr( name );
526 }
527
528 static HRESULT WINAPI domfrag_transformNodeToObject(
529     IXMLDOMDocumentFragment *iface,
530     IXMLDOMNode* domNode, VARIANT var1)
531 {
532     domfrag *This = impl_from_IXMLDOMDocumentFragment( iface );
533     return IXMLDOMNode_transformNodeToObject( IXMLDOMNode_from_impl(&This->node), domNode, var1 );
534 }
535
536 static const struct IXMLDOMDocumentFragmentVtbl domfrag_vtbl =
537 {
538     domfrag_QueryInterface,
539     domfrag_AddRef,
540     domfrag_Release,
541     domfrag_GetTypeInfoCount,
542     domfrag_GetTypeInfo,
543     domfrag_GetIDsOfNames,
544     domfrag_Invoke,
545     domfrag_get_nodeName,
546     domfrag_get_nodeValue,
547     domfrag_put_nodeValue,
548     domfrag_get_nodeType,
549     domfrag_get_parentNode,
550     domfrag_get_childNodes,
551     domfrag_get_firstChild,
552     domfrag_get_lastChild,
553     domfrag_get_previousSibling,
554     domfrag_get_nextSibling,
555     domfrag_get_attributes,
556     domfrag_insertBefore,
557     domfrag_replaceChild,
558     domfrag_removeChild,
559     domfrag_appendChild,
560     domfrag_hasChildNodes,
561     domfrag_get_ownerDocument,
562     domfrag_cloneNode,
563     domfrag_get_nodeTypeString,
564     domfrag_get_text,
565     domfrag_put_text,
566     domfrag_get_specified,
567     domfrag_get_definition,
568     domfrag_get_nodeTypedValue,
569     domfrag_put_nodeTypedValue,
570     domfrag_get_dataType,
571     domfrag_put_dataType,
572     domfrag_get_xml,
573     domfrag_transformNode,
574     domfrag_selectNodes,
575     domfrag_selectSingleNode,
576     domfrag_get_parsed,
577     domfrag_get_namespaceURI,
578     domfrag_get_prefix,
579     domfrag_get_baseName,
580     domfrag_transformNodeToObject
581 };
582
583 IUnknown* create_doc_fragment( xmlNodePtr fragment )
584 {
585     domfrag *This;
586
587     This = heap_alloc( sizeof *This );
588     if ( !This )
589         return NULL;
590
591     This->lpVtbl = &domfrag_vtbl;
592     This->ref = 1;
593
594     init_xmlnode(&This->node, fragment, (IXMLDOMNode*)&This->lpVtbl, NULL);
595
596     return (IUnknown*) &This->lpVtbl;
597 }
598
599 #endif