setupapi: Improve parameter validation for SetupCreateDiskSpaceListA/W.
[wine] / dlls / msxml3 / comment.c
1 /*
2  *    DOM comment node implementation
3  *
4  * Copyright 2006 Huw Davies
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 _domcomment
41 {
42     xmlnode node;
43     const struct IXMLDOMCommentVtbl *lpVtbl;
44     LONG ref;
45 } domcomment;
46
47 static inline domcomment *impl_from_IXMLDOMComment( IXMLDOMComment *iface )
48 {
49     return (domcomment *)((char*)iface - FIELD_OFFSET(domcomment, lpVtbl));
50 }
51
52 static HRESULT WINAPI domcomment_QueryInterface(
53     IXMLDOMComment *iface,
54     REFIID riid,
55     void** ppvObject )
56 {
57     domcomment *This = impl_from_IXMLDOMComment( iface );
58     TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppvObject);
59
60     if ( IsEqualGUID( riid, &IID_IXMLDOMComment ) ||
61          IsEqualGUID( riid, &IID_IXMLDOMCharacterData) ||
62          IsEqualGUID( riid, &IID_IXMLDOMNode ) ||
63          IsEqualGUID( riid, &IID_IDispatch ) ||
64          IsEqualGUID( riid, &IID_IUnknown ) )
65     {
66         *ppvObject = iface;
67     }
68     else if(node_query_interface(&This->node, riid, ppvObject))
69     {
70         return *ppvObject ? S_OK : E_NOINTERFACE;
71     }
72     else
73     {
74         FIXME("Unsupported interface %s\n", debugstr_guid(riid));
75         return E_NOINTERFACE;
76     }
77
78     IXMLDOMText_AddRef((IUnknown*)*ppvObject);
79     return S_OK;
80 }
81
82 static ULONG WINAPI domcomment_AddRef(
83     IXMLDOMComment *iface )
84 {
85     domcomment *This = impl_from_IXMLDOMComment( iface );
86     return InterlockedIncrement( &This->ref );
87 }
88
89 static ULONG WINAPI domcomment_Release(
90     IXMLDOMComment *iface )
91 {
92     domcomment *This = impl_from_IXMLDOMComment( iface );
93     ULONG ref;
94
95     ref = InterlockedDecrement( &This->ref );
96     if ( ref == 0 )
97     {
98         destroy_xmlnode(&This->node);
99         heap_free( This );
100     }
101
102     return ref;
103 }
104
105 static HRESULT WINAPI domcomment_GetTypeInfoCount(
106     IXMLDOMComment *iface,
107     UINT* pctinfo )
108 {
109     domcomment *This = impl_from_IXMLDOMComment( iface );
110
111     TRACE("(%p)->(%p)\n", This, pctinfo);
112
113     *pctinfo = 1;
114
115     return S_OK;
116 }
117
118 static HRESULT WINAPI domcomment_GetTypeInfo(
119     IXMLDOMComment *iface,
120     UINT iTInfo, LCID lcid,
121     ITypeInfo** ppTInfo )
122 {
123     domcomment *This = impl_from_IXMLDOMComment( iface );
124     HRESULT hr;
125
126     TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
127
128     hr = get_typeinfo(IXMLDOMComment_tid, ppTInfo);
129
130     return hr;
131 }
132
133 static HRESULT WINAPI domcomment_GetIDsOfNames(
134     IXMLDOMComment *iface,
135     REFIID riid, LPOLESTR* rgszNames,
136     UINT cNames, LCID lcid, DISPID* rgDispId )
137 {
138     domcomment *This = impl_from_IXMLDOMComment( iface );
139     ITypeInfo *typeinfo;
140     HRESULT hr;
141
142     TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames,
143           lcid, rgDispId);
144
145     if(!rgszNames || cNames == 0 || !rgDispId)
146         return E_INVALIDARG;
147
148     hr = get_typeinfo(IXMLDOMComment_tid, &typeinfo);
149     if(SUCCEEDED(hr))
150     {
151         hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
152         ITypeInfo_Release(typeinfo);
153     }
154
155     return hr;
156 }
157
158 static HRESULT WINAPI domcomment_Invoke(
159     IXMLDOMComment *iface,
160     DISPID dispIdMember, REFIID riid, LCID lcid,
161     WORD wFlags, DISPPARAMS* pDispParams, VARIANT* pVarResult,
162     EXCEPINFO* pExcepInfo, UINT* puArgErr )
163 {
164     domcomment *This = impl_from_IXMLDOMComment( iface );
165     ITypeInfo *typeinfo;
166     HRESULT hr;
167
168     TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
169           lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
170
171     hr = get_typeinfo(IXMLDOMComment_tid, &typeinfo);
172     if(SUCCEEDED(hr))
173     {
174         hr = ITypeInfo_Invoke(typeinfo, &(This->lpVtbl), dispIdMember, wFlags, pDispParams,
175                 pVarResult, pExcepInfo, puArgErr);
176         ITypeInfo_Release(typeinfo);
177     }
178
179     return hr;
180 }
181
182 static HRESULT WINAPI domcomment_get_nodeName(
183     IXMLDOMComment *iface,
184     BSTR* p )
185 {
186     domcomment *This = impl_from_IXMLDOMComment( iface );
187
188     static const WCHAR commentW[] = {'#','c','o','m','m','e','n','t',0};
189
190     TRACE("(%p)->(%p)\n", This, p);
191
192     return return_bstr(commentW, p);
193 }
194
195 static HRESULT WINAPI domcomment_get_nodeValue(
196     IXMLDOMComment *iface,
197     VARIANT* value)
198 {
199     domcomment *This = impl_from_IXMLDOMComment( iface );
200
201     TRACE("(%p)->(%p)\n", This, value);
202
203     return node_get_content(&This->node, value);
204 }
205
206 static HRESULT WINAPI domcomment_put_nodeValue(
207     IXMLDOMComment *iface,
208     VARIANT var1 )
209 {
210     domcomment *This = impl_from_IXMLDOMComment( iface );
211     return IXMLDOMNode_put_nodeValue( IXMLDOMNode_from_impl(&This->node), var1 );
212 }
213
214 static HRESULT WINAPI domcomment_get_nodeType(
215     IXMLDOMComment *iface,
216     DOMNodeType* domNodeType )
217 {
218     domcomment *This = impl_from_IXMLDOMComment( iface );
219     return IXMLDOMNode_get_nodeType( IXMLDOMNode_from_impl(&This->node), domNodeType );
220 }
221
222 static HRESULT WINAPI domcomment_get_parentNode(
223     IXMLDOMComment *iface,
224     IXMLDOMNode** parent )
225 {
226     domcomment *This = impl_from_IXMLDOMComment( iface );
227     return IXMLDOMNode_get_parentNode( IXMLDOMNode_from_impl(&This->node), parent );
228 }
229
230 static HRESULT WINAPI domcomment_get_childNodes(
231     IXMLDOMComment *iface,
232     IXMLDOMNodeList** outList)
233 {
234     domcomment *This = impl_from_IXMLDOMComment( iface );
235     return IXMLDOMNode_get_childNodes( IXMLDOMNode_from_impl(&This->node), outList );
236 }
237
238 static HRESULT WINAPI domcomment_get_firstChild(
239     IXMLDOMComment *iface,
240     IXMLDOMNode** domNode)
241 {
242     domcomment *This = impl_from_IXMLDOMComment( iface );
243     return IXMLDOMNode_get_firstChild( IXMLDOMNode_from_impl(&This->node), domNode );
244 }
245
246 static HRESULT WINAPI domcomment_get_lastChild(
247     IXMLDOMComment *iface,
248     IXMLDOMNode** domNode)
249 {
250     domcomment *This = impl_from_IXMLDOMComment( iface );
251     return IXMLDOMNode_get_lastChild( IXMLDOMNode_from_impl(&This->node), domNode );
252 }
253
254 static HRESULT WINAPI domcomment_get_previousSibling(
255     IXMLDOMComment *iface,
256     IXMLDOMNode** domNode)
257 {
258     domcomment *This = impl_from_IXMLDOMComment( iface );
259     return IXMLDOMNode_get_previousSibling( IXMLDOMNode_from_impl(&This->node), domNode );
260 }
261
262 static HRESULT WINAPI domcomment_get_nextSibling(
263     IXMLDOMComment *iface,
264     IXMLDOMNode** domNode)
265 {
266     domcomment *This = impl_from_IXMLDOMComment( iface );
267     return IXMLDOMNode_get_nextSibling( IXMLDOMNode_from_impl(&This->node), domNode );
268 }
269
270 static HRESULT WINAPI domcomment_get_attributes(
271     IXMLDOMComment *iface,
272     IXMLDOMNamedNodeMap** attributeMap)
273 {
274     domcomment *This = impl_from_IXMLDOMComment( iface );
275     return IXMLDOMNode_get_attributes( IXMLDOMNode_from_impl(&This->node), attributeMap );
276 }
277
278 static HRESULT WINAPI domcomment_insertBefore(
279     IXMLDOMComment *iface,
280     IXMLDOMNode* newNode, VARIANT var1,
281     IXMLDOMNode** outOldNode)
282 {
283     domcomment *This = impl_from_IXMLDOMComment( iface );
284     return IXMLDOMNode_insertBefore( IXMLDOMNode_from_impl(&This->node), newNode, var1, outOldNode );
285 }
286
287 static HRESULT WINAPI domcomment_replaceChild(
288     IXMLDOMComment *iface,
289     IXMLDOMNode* newNode,
290     IXMLDOMNode* oldNode,
291     IXMLDOMNode** outOldNode)
292 {
293     domcomment *This = impl_from_IXMLDOMComment( iface );
294     return IXMLDOMNode_replaceChild( IXMLDOMNode_from_impl(&This->node), newNode, oldNode, outOldNode );
295 }
296
297 static HRESULT WINAPI domcomment_removeChild(
298     IXMLDOMComment *iface,
299     IXMLDOMNode* domNode, IXMLDOMNode** oldNode)
300 {
301     domcomment *This = impl_from_IXMLDOMComment( iface );
302     return IXMLDOMNode_removeChild( IXMLDOMNode_from_impl(&This->node), domNode, oldNode );
303 }
304
305 static HRESULT WINAPI domcomment_appendChild(
306     IXMLDOMComment *iface,
307     IXMLDOMNode* newNode, IXMLDOMNode** outNewNode)
308 {
309     domcomment *This = impl_from_IXMLDOMComment( iface );
310     return IXMLDOMNode_appendChild( IXMLDOMNode_from_impl(&This->node), newNode, outNewNode );
311 }
312
313 static HRESULT WINAPI domcomment_hasChildNodes(
314     IXMLDOMComment *iface,
315     VARIANT_BOOL* pbool)
316 {
317     domcomment *This = impl_from_IXMLDOMComment( iface );
318     return IXMLDOMNode_hasChildNodes( IXMLDOMNode_from_impl(&This->node), pbool );
319 }
320
321 static HRESULT WINAPI domcomment_get_ownerDocument(
322     IXMLDOMComment *iface,
323     IXMLDOMDocument** domDocument)
324 {
325     domcomment *This = impl_from_IXMLDOMComment( iface );
326     return IXMLDOMNode_get_ownerDocument( IXMLDOMNode_from_impl(&This->node), domDocument );
327 }
328
329 static HRESULT WINAPI domcomment_cloneNode(
330     IXMLDOMComment *iface,
331     VARIANT_BOOL pbool, IXMLDOMNode** outNode)
332 {
333     domcomment *This = impl_from_IXMLDOMComment( iface );
334     return IXMLDOMNode_cloneNode( IXMLDOMNode_from_impl(&This->node), pbool, outNode );
335 }
336
337 static HRESULT WINAPI domcomment_get_nodeTypeString(
338     IXMLDOMComment *iface,
339     BSTR* p)
340 {
341     domcomment *This = impl_from_IXMLDOMComment( iface );
342     return IXMLDOMNode_get_nodeTypeString( IXMLDOMNode_from_impl(&This->node), p );
343 }
344
345 static HRESULT WINAPI domcomment_get_text(
346     IXMLDOMComment *iface,
347     BSTR* p)
348 {
349     domcomment *This = impl_from_IXMLDOMComment( iface );
350     return IXMLDOMNode_get_text( IXMLDOMNode_from_impl(&This->node), p );
351 }
352
353 static HRESULT WINAPI domcomment_put_text(
354     IXMLDOMComment *iface,
355     BSTR p)
356 {
357     domcomment *This = impl_from_IXMLDOMComment( iface );
358     return IXMLDOMNode_put_text( IXMLDOMNode_from_impl(&This->node), p );
359 }
360
361 static HRESULT WINAPI domcomment_get_specified(
362     IXMLDOMComment *iface,
363     VARIANT_BOOL* pbool)
364 {
365     domcomment *This = impl_from_IXMLDOMComment( iface );
366     return IXMLDOMNode_get_specified( IXMLDOMNode_from_impl(&This->node), pbool );
367 }
368
369 static HRESULT WINAPI domcomment_get_definition(
370     IXMLDOMComment *iface,
371     IXMLDOMNode** domNode)
372 {
373     domcomment *This = impl_from_IXMLDOMComment( iface );
374     return IXMLDOMNode_get_definition( IXMLDOMNode_from_impl(&This->node), domNode );
375 }
376
377 static HRESULT WINAPI domcomment_get_nodeTypedValue(
378     IXMLDOMComment *iface,
379     VARIANT* var1)
380 {
381     domcomment *This = impl_from_IXMLDOMComment( iface );
382     return IXMLDOMNode_get_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
383 }
384
385 static HRESULT WINAPI domcomment_put_nodeTypedValue(
386     IXMLDOMComment *iface,
387     VARIANT var1)
388 {
389     domcomment *This = impl_from_IXMLDOMComment( iface );
390     return IXMLDOMNode_put_nodeTypedValue( IXMLDOMNode_from_impl(&This->node), var1 );
391 }
392
393 static HRESULT WINAPI domcomment_get_dataType(
394     IXMLDOMComment *iface,
395     VARIANT* var1)
396 {
397     domcomment *This = impl_from_IXMLDOMComment( iface );
398     return IXMLDOMNode_get_dataType( IXMLDOMNode_from_impl(&This->node), var1 );
399 }
400
401 static HRESULT WINAPI domcomment_put_dataType(
402     IXMLDOMComment *iface,
403     BSTR p)
404 {
405     domcomment *This = impl_from_IXMLDOMComment( iface );
406     return IXMLDOMNode_put_dataType( IXMLDOMNode_from_impl(&This->node), p );
407 }
408
409 static HRESULT WINAPI domcomment_get_xml(
410     IXMLDOMComment *iface,
411     BSTR* p)
412 {
413     domcomment *This = impl_from_IXMLDOMComment( iface );
414     return IXMLDOMNode_get_xml( IXMLDOMNode_from_impl(&This->node), p );
415 }
416
417 static HRESULT WINAPI domcomment_transformNode(
418     IXMLDOMComment *iface,
419     IXMLDOMNode* domNode, BSTR* p)
420 {
421     domcomment *This = impl_from_IXMLDOMComment( iface );
422     return IXMLDOMNode_transformNode( IXMLDOMNode_from_impl(&This->node), domNode, p );
423 }
424
425 static HRESULT WINAPI domcomment_selectNodes(
426     IXMLDOMComment *iface,
427     BSTR p, IXMLDOMNodeList** outList)
428 {
429     domcomment *This = impl_from_IXMLDOMComment( iface );
430     return IXMLDOMNode_selectNodes( IXMLDOMNode_from_impl(&This->node), p, outList );
431 }
432
433 static HRESULT WINAPI domcomment_selectSingleNode(
434     IXMLDOMComment *iface,
435     BSTR p, IXMLDOMNode** outNode)
436 {
437     domcomment *This = impl_from_IXMLDOMComment( iface );
438     return IXMLDOMNode_selectSingleNode( IXMLDOMNode_from_impl(&This->node), p, outNode );
439 }
440
441 static HRESULT WINAPI domcomment_get_parsed(
442     IXMLDOMComment *iface,
443     VARIANT_BOOL* pbool)
444 {
445     domcomment *This = impl_from_IXMLDOMComment( iface );
446     return IXMLDOMNode_get_parsed( IXMLDOMNode_from_impl(&This->node), pbool );
447 }
448
449 static HRESULT WINAPI domcomment_get_namespaceURI(
450     IXMLDOMComment *iface,
451     BSTR* p)
452 {
453     domcomment *This = impl_from_IXMLDOMComment( iface );
454     return IXMLDOMNode_get_namespaceURI( IXMLDOMNode_from_impl(&This->node), p );
455 }
456
457 static HRESULT WINAPI domcomment_get_prefix(
458     IXMLDOMComment *iface,
459     BSTR* p)
460 {
461     domcomment *This = impl_from_IXMLDOMComment( iface );
462     return IXMLDOMNode_get_prefix( IXMLDOMNode_from_impl(&This->node), p );
463 }
464
465 static HRESULT WINAPI domcomment_get_baseName(
466     IXMLDOMComment *iface,
467     BSTR* p)
468 {
469     domcomment *This = impl_from_IXMLDOMComment( iface );
470     return IXMLDOMNode_get_baseName( IXMLDOMNode_from_impl(&This->node), p );
471 }
472
473 static HRESULT WINAPI domcomment_transformNodeToObject(
474     IXMLDOMComment *iface,
475     IXMLDOMNode* domNode, VARIANT var1)
476 {
477     domcomment *This = impl_from_IXMLDOMComment( iface );
478     return IXMLDOMNode_transformNodeToObject( IXMLDOMNode_from_impl(&This->node), domNode, var1 );
479 }
480
481 static HRESULT WINAPI domcomment_get_data(
482     IXMLDOMComment *iface,
483     BSTR *p)
484 {
485     domcomment *This = impl_from_IXMLDOMComment( iface );
486     HRESULT hr;
487     VARIANT vRet;
488
489     TRACE("(%p)->(%p)\n", This, p);
490
491     if(!p)
492         return E_INVALIDARG;
493
494     hr = IXMLDOMComment_get_nodeValue( iface, &vRet );
495     if(hr == S_OK)
496     {
497         *p = V_BSTR(&vRet);
498     }
499
500     return hr;
501 }
502
503 static HRESULT WINAPI domcomment_put_data(
504     IXMLDOMComment *iface,
505     BSTR data)
506 {
507     domcomment *This = impl_from_IXMLDOMComment( iface );
508     VARIANT val;
509
510     TRACE("(%p)->(%s)\n", This, debugstr_w(data) );
511
512     V_VT(&val) = VT_BSTR;
513     V_BSTR(&val) = data;
514
515     return IXMLDOMNode_put_nodeValue( IXMLDOMNode_from_impl(&This->node), val );
516 }
517
518 static HRESULT WINAPI domcomment_get_length(
519     IXMLDOMComment *iface,
520     LONG *len)
521 {
522     domcomment *This = impl_from_IXMLDOMComment( iface );
523     HRESULT hr;
524     BSTR data;
525
526     TRACE("(%p)->(%p)\n", This, len);
527
528     if(!len)
529         return E_INVALIDARG;
530
531     hr = IXMLDOMComment_get_data(iface, &data);
532     if(hr == S_OK)
533     {
534         *len = SysStringLen(data);
535         SysFreeString(data);
536     }
537
538     return hr;
539 }
540
541 static HRESULT WINAPI domcomment_substringData(
542     IXMLDOMComment *iface,
543     LONG offset, LONG count, BSTR *p)
544 {
545     domcomment *This = impl_from_IXMLDOMComment( iface );
546     HRESULT hr;
547     BSTR data;
548
549     TRACE("(%p)->(%d %d %p)\n", This, offset, count, p);
550
551     if(!p)
552         return E_INVALIDARG;
553
554     *p = NULL;
555     if(offset < 0 || count < 0)
556         return E_INVALIDARG;
557
558     if(count == 0)
559         return S_FALSE;
560
561     hr = IXMLDOMComment_get_data(iface, &data);
562     if(hr == S_OK)
563     {
564         LONG len = SysStringLen(data);
565
566         if(offset < len)
567         {
568             if(offset + count > len)
569                 *p = SysAllocString(&data[offset]);
570             else
571                 *p = SysAllocStringLen(&data[offset], count);
572         }
573         else
574             hr = S_FALSE;
575
576         SysFreeString(data);
577     }
578
579     return hr;
580 }
581
582 static HRESULT WINAPI domcomment_appendData(
583     IXMLDOMComment *iface,
584     BSTR p)
585 {
586     domcomment *This = impl_from_IXMLDOMComment( iface );
587     HRESULT hr;
588     BSTR data;
589     LONG p_len;
590
591     TRACE("(%p)->(%s)\n", This, debugstr_w(p));
592
593     /* Nothing to do if NULL or an Empty string passed in. */
594     if((p_len = SysStringLen(p)) == 0) return S_OK;
595
596     hr = IXMLDOMComment_get_data(iface, &data);
597     if(hr == S_OK)
598     {
599         LONG len = SysStringLen(data);
600         BSTR str = SysAllocStringLen(NULL, p_len + len);
601
602         memcpy(str, data, len*sizeof(WCHAR));
603         memcpy(&str[len], p, p_len*sizeof(WCHAR));
604         str[len+p_len] = 0;
605
606         hr = IXMLDOMComment_put_data(iface, str);
607
608         SysFreeString(str);
609         SysFreeString(data);
610     }
611
612     return hr;
613 }
614
615 static HRESULT WINAPI domcomment_insertData(
616     IXMLDOMComment *iface,
617     LONG offset, BSTR p)
618 {
619     domcomment *This = impl_from_IXMLDOMComment( iface );
620     HRESULT hr;
621     BSTR data;
622     LONG p_len;
623
624     TRACE("(%p)->(%d %s)\n", This, offset, debugstr_w(p));
625
626     /* If have a NULL or empty string, don't do anything. */
627     if((p_len = SysStringLen(p)) == 0)
628         return S_OK;
629
630     if(offset < 0)
631     {
632         return E_INVALIDARG;
633     }
634
635     hr = IXMLDOMComment_get_data(iface, &data);
636     if(hr == S_OK)
637     {
638         LONG len = SysStringLen(data);
639         BSTR str;
640
641         if(len < offset)
642         {
643             SysFreeString(data);
644             return E_INVALIDARG;
645         }
646
647         str = SysAllocStringLen(NULL, len + p_len);
648         /* start part, supplied string and end part */
649         memcpy(str, data, offset*sizeof(WCHAR));
650         memcpy(&str[offset], p, p_len*sizeof(WCHAR));
651         memcpy(&str[offset+p_len], &data[offset], (len-offset)*sizeof(WCHAR));
652         str[len+p_len] = 0;
653
654         hr = IXMLDOMComment_put_data(iface, str);
655
656         SysFreeString(str);
657         SysFreeString(data);
658     }
659
660     return hr;
661 }
662
663 static HRESULT WINAPI domcomment_deleteData(
664     IXMLDOMComment *iface,
665     LONG offset, LONG count)
666 {
667     HRESULT hr;
668     LONG len = -1;
669     BSTR str;
670
671     TRACE("(%p)->(%d %d)\n", iface, offset, count);
672
673     hr = IXMLDOMComment_get_length(iface, &len);
674     if(hr != S_OK) return hr;
675
676     if((offset < 0) || (offset > len) || (count < 0))
677         return E_INVALIDARG;
678
679     if(len == 0) return S_OK;
680
681     /* cutting start or end */
682     if((offset == 0) || ((count + offset) >= len))
683     {
684         if(offset == 0)
685             IXMLDOMComment_substringData(iface, count, len - count, &str);
686         else
687             IXMLDOMComment_substringData(iface, 0, offset, &str);
688         hr = IXMLDOMComment_put_data(iface, str);
689     }
690     else
691     /* cutting from the inside */
692     {
693         BSTR str_end;
694
695         IXMLDOMComment_substringData(iface, 0, offset, &str);
696         IXMLDOMComment_substringData(iface, offset + count, len - count, &str_end);
697
698         hr = IXMLDOMComment_put_data(iface, str);
699         if(hr == S_OK)
700             hr = IXMLDOMComment_appendData(iface, str_end);
701
702         SysFreeString(str_end);
703     }
704
705     SysFreeString(str);
706
707     return hr;
708 }
709
710 static HRESULT WINAPI domcomment_replaceData(
711     IXMLDOMComment *iface,
712     LONG offset, LONG count, BSTR p)
713 {
714     domcomment *This = impl_from_IXMLDOMComment( iface );
715     HRESULT hr;
716
717     TRACE("(%p)->(%d %d %s)\n", This, offset, count, debugstr_w(p));
718
719     hr = IXMLDOMComment_deleteData(iface, offset, count);
720
721     if (hr == S_OK)
722        hr = IXMLDOMComment_insertData(iface, offset, p);
723
724     return hr;
725 }
726
727 static const struct IXMLDOMCommentVtbl domcomment_vtbl =
728 {
729     domcomment_QueryInterface,
730     domcomment_AddRef,
731     domcomment_Release,
732     domcomment_GetTypeInfoCount,
733     domcomment_GetTypeInfo,
734     domcomment_GetIDsOfNames,
735     domcomment_Invoke,
736     domcomment_get_nodeName,
737     domcomment_get_nodeValue,
738     domcomment_put_nodeValue,
739     domcomment_get_nodeType,
740     domcomment_get_parentNode,
741     domcomment_get_childNodes,
742     domcomment_get_firstChild,
743     domcomment_get_lastChild,
744     domcomment_get_previousSibling,
745     domcomment_get_nextSibling,
746     domcomment_get_attributes,
747     domcomment_insertBefore,
748     domcomment_replaceChild,
749     domcomment_removeChild,
750     domcomment_appendChild,
751     domcomment_hasChildNodes,
752     domcomment_get_ownerDocument,
753     domcomment_cloneNode,
754     domcomment_get_nodeTypeString,
755     domcomment_get_text,
756     domcomment_put_text,
757     domcomment_get_specified,
758     domcomment_get_definition,
759     domcomment_get_nodeTypedValue,
760     domcomment_put_nodeTypedValue,
761     domcomment_get_dataType,
762     domcomment_put_dataType,
763     domcomment_get_xml,
764     domcomment_transformNode,
765     domcomment_selectNodes,
766     domcomment_selectSingleNode,
767     domcomment_get_parsed,
768     domcomment_get_namespaceURI,
769     domcomment_get_prefix,
770     domcomment_get_baseName,
771     domcomment_transformNodeToObject,
772     domcomment_get_data,
773     domcomment_put_data,
774     domcomment_get_length,
775     domcomment_substringData,
776     domcomment_appendData,
777     domcomment_insertData,
778     domcomment_deleteData,
779     domcomment_replaceData
780 };
781
782 IUnknown* create_comment( xmlNodePtr comment )
783 {
784     domcomment *This;
785
786     This = heap_alloc( sizeof *This );
787     if ( !This )
788         return NULL;
789
790     This->lpVtbl = &domcomment_vtbl;
791     This->ref = 1;
792
793     init_xmlnode(&This->node, comment, (IXMLDOMNode*)&This->lpVtbl, NULL);
794
795     return (IUnknown*) &This->lpVtbl;
796 }
797
798 #endif